1pub(crate) fn parse_decimal_number(
11 s: &str,
12 pref_const: u64,
13) -> Result<u64, DecimalNumberParsingError> {
14 let (int, fract) = if let Some((whole, fractional)) = s.trim().split_once('.') {
15 let int: u64 = whole
16 .parse()
17 .map_err(|_| DecimalNumberParsingError::InvalidNumber(s.to_owned()))?;
18 let mut fract: u64 = fractional
19 .parse()
20 .map_err(|_| DecimalNumberParsingError::InvalidNumber(s.to_owned()))?;
21 let len = u32::try_from(fractional.len())
22 .map_err(|_| DecimalNumberParsingError::InvalidNumber(s.to_owned()))?;
23 fract = fract
24 .checked_mul(
25 pref_const
26 .checked_div(10u64.checked_pow(len).ok_or_else(|| {
27 DecimalNumberParsingError::LongFractional(fractional.to_owned())
28 })?)
29 .filter(|n| *n != 0u64)
30 .ok_or_else(|| {
31 DecimalNumberParsingError::LongFractional(fractional.to_owned())
32 })?,
33 )
34 .ok_or_else(|| DecimalNumberParsingError::LongFractional(fractional.to_owned()))?;
35 (int, fract)
36 } else {
37 let int: u64 = s
38 .parse()
39 .map_err(|_| DecimalNumberParsingError::InvalidNumber(s.to_owned()))?;
40 (int, 0)
41 };
42 let result = fract
43 .checked_add(
44 int.checked_mul(pref_const)
45 .ok_or_else(|| DecimalNumberParsingError::LongWhole(int.to_string()))?,
46 )
47 .ok_or_else(|| DecimalNumberParsingError::LongWhole(int.to_string()))?;
48 Ok(result)
49}
50
51#[derive(Debug, Clone, PartialEq, Eq)]
52pub enum DecimalNumberParsingError {
53 InvalidNumber(String),
54 LongWhole(String),
55 LongFractional(String),
56}
57
58impl std::error::Error for DecimalNumberParsingError {
59 fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
60 None
61 }
62
63 fn description(&self) -> &str {
64 "description() is deprecated; use Display"
65 }
66
67 fn cause(&self) -> Option<&dyn std::error::Error> {
68 self.source()
69 }
70}
71
72impl std::fmt::Display for DecimalNumberParsingError {
73 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
74 match self {
75 DecimalNumberParsingError::InvalidNumber(s) => {
76 write!(f, "Invalid number: {}", s)
77 }
78 DecimalNumberParsingError::LongWhole(s) => {
79 write!(f, "Long whole part: {}", s)
80 }
81 DecimalNumberParsingError::LongFractional(s) => {
82 write!(f, "Long fractional part: {}", s)
83 }
84 }
85 }
86}
87
88#[cfg(test)]
89mod tests {
90 use super::*;
91
92 const TEST: [(u64, &'static str, u64); 6] = [
93 (129380_000_001u64, "129.380000001", 10u64.pow(9)),
94 (12938_000_000_100_000_000u64, "12938000000.1", 10u64.pow(9)),
95 (129380_000_001u64, "0.129380000001", 10u64.pow(12)),
96 (129380_000_001_000u64, "129.380000001000", 10u64.pow(12)),
97 (9488129380_000_001u64, "9488.129380000001", 10u64.pow(12)),
98 (129380_000_001u64, "00.129380000001", 10u64.pow(12)),
99 ];
100
101 #[test]
102 fn parse_test() {
103 for (expected_value, str_value, precision) in TEST {
104 let parsed_value = parse_decimal_number(str_value, precision).unwrap();
105 assert_eq!(parsed_value, expected_value)
106 }
107 }
108
109 #[test]
110 fn test_long_fract() {
111 let data = "1.23456";
112 let prefix = 10000u64;
113 assert_eq!(
114 parse_decimal_number(data, prefix),
115 Err(DecimalNumberParsingError::LongFractional(23456.to_string()))
116 );
117 }
118
119 #[test]
120 fn invalidnumber_whole() {
121 let num = "1h4.7859";
122 let prefix: u64 = 10000;
123 assert_eq!(
124 parse_decimal_number(num, prefix),
125 Err(DecimalNumberParsingError::InvalidNumber(
126 "1h4.7859".to_owned()
127 ))
128 );
129 }
130 #[test]
131 fn invalidnumber_fract() {
132 let num = "14.785h9";
133 let prefix: u64 = 10000;
134 assert_eq!(
135 parse_decimal_number(num, prefix),
136 Err(DecimalNumberParsingError::InvalidNumber(
137 "14.785h9".to_owned()
138 ))
139 );
140 }
141
142 #[test]
143 fn max_long_fract() {
144 let max_data = 10u64.pow(17) + 1;
145 let data = "1.".to_string() + max_data.to_string().as_str();
146 let prefix = 10u64.pow(17);
147 assert_eq!(
148 parse_decimal_number(data.as_str(), prefix),
149 Err(DecimalNumberParsingError::LongFractional(
150 max_data.to_string()
151 ))
152 );
153 }
154
155 #[test]
156 fn long_whole_test() {
157 let data = 10u64.pow(17) + 1;
158 let prefix = 10u64.pow(12);
159 let s = data.to_string() + "." + "1";
160 assert_eq!(
161 parse_decimal_number(s.as_str(), prefix),
162 Err(DecimalNumberParsingError::LongWhole(data.to_string()))
163 );
164 }
165
166 #[test]
167 fn parse_u64_errortest() {
168 let test_data = u64::MAX.to_string();
169 let gas = parse_decimal_number(&test_data, 10u64.pow(9));
170 assert_eq!(
171 gas,
172 Err(DecimalNumberParsingError::LongWhole(u64::MAX.to_string()))
173 );
174 }
175
176 #[test]
177 fn test() {
178 let data = "1.000000000000000000000000000000000000001";
179 let prefix = 100u64;
180 assert_eq!(
181 parse_decimal_number(data, prefix),
182 Err(DecimalNumberParsingError::LongFractional(
183 "000000000000000000000000000000000000001".to_string()
184 ))
185 );
186 }
187}