|
@@ -9,6 +9,9 @@ pub enum Error {
|
|
|
#[error("{0} is not a valid number")]
|
|
|
NoANumber(String),
|
|
|
|
|
|
+ #[error("Overflow")]
|
|
|
+ Overflow,
|
|
|
+
|
|
|
#[error("Invalid asset name: {0}")]
|
|
|
InvalidAssetName(String),
|
|
|
}
|
|
@@ -161,16 +164,39 @@ impl Amount {
|
|
|
|
|
|
impl ToString for Amount {
|
|
|
fn to_string(&self) -> String {
|
|
|
+ self.try_into().unwrap()
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl TryInto<String> for &Amount {
|
|
|
+ type Error = Error;
|
|
|
+
|
|
|
+ fn try_into(self) -> Result<String, Error> {
|
|
|
let str = self.cents.abs().to_string();
|
|
|
- let precision: usize = self.asset.precision.into();
|
|
|
- let str = if str.len() < precision + 1 {
|
|
|
- format!("{}{}", "0".repeat(precision - str.len() + 1), str)
|
|
|
+ let precision = self.asset.precision;
|
|
|
+ let len = u8::try_from(str.len()).map_err(|_| Error::Overflow)?;
|
|
|
+
|
|
|
+ let (str, len) = if len < precision.checked_add(1).ok_or(Error::Overflow)? {
|
|
|
+ (
|
|
|
+ format!(
|
|
|
+ "{}{}",
|
|
|
+ "0".repeat(
|
|
|
+ precision
|
|
|
+ .checked_sub(len)
|
|
|
+ .and_then(|x| x.checked_add(1))
|
|
|
+ .ok_or(Error::Overflow)?
|
|
|
+ .into()
|
|
|
+ ),
|
|
|
+ str
|
|
|
+ ),
|
|
|
+ precision.checked_add(1).ok_or(Error::Overflow)?,
|
|
|
+ )
|
|
|
} else {
|
|
|
- str
|
|
|
+ (str, len)
|
|
|
};
|
|
|
|
|
|
- let (left, right) = str.split_at(str.len() - precision);
|
|
|
- format!(
|
|
|
+ let (left, right) = str.split_at(len.checked_sub(precision).ok_or(Error::Overflow)?.into());
|
|
|
+ Ok(format!(
|
|
|
"{}{}.{}",
|
|
|
if self.cents.is_negative() { "-" } else { "" },
|
|
|
left,
|
|
@@ -178,7 +204,7 @@ impl ToString for Amount {
|
|
|
)
|
|
|
.trim_end_matches('0')
|
|
|
.trim_end_matches('.')
|
|
|
- .to_owned()
|
|
|
+ .to_owned())
|
|
|
}
|
|
|
}
|
|
|
|