2018-01-20 01:43:02 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2016-10-22 21:35:48 +03:00
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2016-10-21 03:06:12 +03:00
|
|
|
//! Core types
|
|
|
|
|
2016-10-23 01:54:42 +03:00
|
|
|
pub mod block;
|
2018-09-20 11:19:32 +03:00
|
|
|
pub mod block_sums;
|
2018-06-02 21:00:44 +03:00
|
|
|
pub mod committed;
|
2018-08-22 22:19:37 +03:00
|
|
|
pub mod compact_block;
|
2016-10-23 01:08:53 +03:00
|
|
|
pub mod hash;
|
2018-01-20 01:43:02 +03:00
|
|
|
pub mod id;
|
2018-06-20 22:18:52 +03:00
|
|
|
pub mod merkle_proof;
|
2017-08-04 22:50:32 +03:00
|
|
|
pub mod pmmr;
|
2016-10-23 16:18:44 +03:00
|
|
|
pub mod transaction;
|
2018-08-30 17:44:34 +03:00
|
|
|
pub mod verifier_cache;
|
2018-06-18 18:18:38 +03:00
|
|
|
|
2018-12-08 02:59:40 +03:00
|
|
|
use crate::consensus::GRIN_BASE;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2018-12-08 02:59:40 +03:00
|
|
|
use crate::util::secp::pedersen::Commitment;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2017-09-22 19:44:12 +03:00
|
|
|
pub use self::block::*;
|
2018-09-20 11:19:32 +03:00
|
|
|
pub use self::block_sums::*;
|
2018-06-02 21:00:44 +03:00
|
|
|
pub use self::committed::Committed;
|
2018-08-22 22:19:37 +03:00
|
|
|
pub use self::compact_block::*;
|
2018-01-20 01:43:02 +03:00
|
|
|
pub use self::id::ShortId;
|
2018-06-01 22:41:26 +03:00
|
|
|
pub use self::transaction::*;
|
2018-06-29 20:41:28 +03:00
|
|
|
|
2018-09-03 22:55:04 +03:00
|
|
|
/// Common errors
|
|
|
|
#[derive(Fail, Debug)]
|
|
|
|
pub enum Error {
|
|
|
|
/// Human readable represenation of amount is invalid
|
|
|
|
#[fail(display = "Amount string was invalid")]
|
|
|
|
InvalidAmountString,
|
|
|
|
}
|
|
|
|
|
2017-11-08 00:20:36 +03:00
|
|
|
/// Common method for parsing an amount from human-readable, and converting
|
|
|
|
/// to internally-compatible u64
|
|
|
|
|
2018-09-03 22:55:04 +03:00
|
|
|
pub fn amount_from_hr_string(amount: &str) -> Result<u64, Error> {
|
|
|
|
// no i18n yet, make sure we use '.' as the separator
|
|
|
|
if amount.find(',').is_some() {
|
|
|
|
return Err(Error::InvalidAmountString);
|
|
|
|
}
|
|
|
|
let (grins, ngrins) = match amount.find('.') {
|
|
|
|
None => (parse_grins(amount)?, 0),
|
|
|
|
Some(pos) => {
|
|
|
|
let (gs, tail) = amount.split_at(pos);
|
|
|
|
(parse_grins(gs)?, parse_ngrins(&tail[1..])?)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
Ok(grins * GRIN_BASE + ngrins)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_grins(amount: &str) -> Result<u64, Error> {
|
|
|
|
if amount == "" {
|
|
|
|
Ok(0)
|
|
|
|
} else {
|
|
|
|
amount
|
|
|
|
.parse::<u64>()
|
|
|
|
.map_err(|_| Error::InvalidAmountString)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lazy_static! {
|
|
|
|
static ref WIDTH: usize = (GRIN_BASE as f64).log(10.0) as usize + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_ngrins(amount: &str) -> Result<u64, Error> {
|
|
|
|
let amount = if amount.len() > *WIDTH {
|
|
|
|
&amount[..*WIDTH]
|
|
|
|
} else {
|
|
|
|
amount
|
|
|
|
};
|
|
|
|
format!("{:0<width$}", amount, width = WIDTH)
|
|
|
|
.parse::<u64>()
|
|
|
|
.map_err(|_| Error::InvalidAmountString)
|
2017-11-08 00:20:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Common method for converting an amount to a human-readable string
|
|
|
|
|
2018-08-17 20:05:35 +03:00
|
|
|
pub fn amount_to_hr_string(amount: u64, truncate: bool) -> String {
|
2017-11-08 00:20:36 +03:00
|
|
|
let amount = (amount as f64 / GRIN_BASE as f64) as f64;
|
2018-09-03 22:55:04 +03:00
|
|
|
let hr = format!("{:.*}", WIDTH, amount);
|
2018-08-17 20:05:35 +03:00
|
|
|
if truncate {
|
|
|
|
let nzeros = hr.chars().rev().take_while(|x| x == &'0').count();
|
2018-09-03 22:55:04 +03:00
|
|
|
if nzeros < *WIDTH {
|
2018-08-17 20:05:35 +03:00
|
|
|
return hr.trim_right_matches('0').to_string();
|
|
|
|
} else {
|
|
|
|
return format!("{}0", hr.trim_right_matches('0'));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
hr
|
2017-11-08 00:20:36 +03:00
|
|
|
}
|
|
|
|
|
2016-10-21 03:06:12 +03:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use super::*;
|
|
|
|
|
2017-11-08 00:20:36 +03:00
|
|
|
#[test]
|
2018-09-03 22:55:04 +03:00
|
|
|
pub fn test_amount_from_hr() {
|
2017-11-08 00:20:36 +03:00
|
|
|
assert!(50123456789 == amount_from_hr_string("50.123456789").unwrap());
|
2018-09-03 22:55:04 +03:00
|
|
|
assert!(50123456789 == amount_from_hr_string("50.1234567899").unwrap());
|
2017-11-08 00:20:36 +03:00
|
|
|
assert!(50 == amount_from_hr_string(".000000050").unwrap());
|
|
|
|
assert!(1 == amount_from_hr_string(".000000001").unwrap());
|
|
|
|
assert!(0 == amount_from_hr_string(".0000000009").unwrap());
|
|
|
|
assert!(500_000_000_000 == amount_from_hr_string("500").unwrap());
|
|
|
|
assert!(
|
|
|
|
5_000_000_000_000_000_000 == amount_from_hr_string("5000000000.00000000000").unwrap()
|
|
|
|
);
|
2018-09-03 22:55:04 +03:00
|
|
|
assert!(66_600_000_000 == amount_from_hr_string("66.6").unwrap());
|
|
|
|
assert!(66_000_000_000 == amount_from_hr_string("66.").unwrap());
|
2017-11-08 00:20:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2018-09-03 22:55:04 +03:00
|
|
|
pub fn test_amount_to_hr() {
|
2018-08-17 20:05:35 +03:00
|
|
|
assert!("50.123456789" == amount_to_hr_string(50123456789, false));
|
|
|
|
assert!("50.123456789" == amount_to_hr_string(50123456789, true));
|
|
|
|
assert!("0.000000050" == amount_to_hr_string(50, false));
|
|
|
|
assert!("0.00000005" == amount_to_hr_string(50, true));
|
|
|
|
assert!("0.000000001" == amount_to_hr_string(1, false));
|
|
|
|
assert!("0.000000001" == amount_to_hr_string(1, true));
|
|
|
|
assert!("500.000000000" == amount_to_hr_string(500_000_000_000, false));
|
|
|
|
assert!("500.0" == amount_to_hr_string(500_000_000_000, true));
|
|
|
|
assert!("5000000000.000000000" == amount_to_hr_string(5_000_000_000_000_000_000, false));
|
|
|
|
assert!("5000000000.0" == amount_to_hr_string(5_000_000_000_000_000_000, true));
|
2018-09-03 22:55:04 +03:00
|
|
|
assert!("66.6" == amount_to_hr_string(66600000000, true));
|
2017-11-08 00:20:36 +03:00
|
|
|
}
|
|
|
|
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|