mirror of
https://github.com/darkrenaissance/darkfi.git
synced 2026-01-10 07:08:05 -05:00
305 lines
8.9 KiB
Rust
305 lines
8.9 KiB
Rust
use std::iter::FromIterator;
|
|
use std::str::FromStr;
|
|
|
|
use num_bigint::BigUint;
|
|
use sha2::{Digest, Sha256};
|
|
|
|
use crate::{
|
|
serial::{deserialize, serialize},
|
|
util::{NetworkName, TokenList},
|
|
Error, Result,
|
|
};
|
|
|
|
// hash the external token ID and NetworkName param.
|
|
// if fails, change the last 4 bytes and hash it again. keep repeating until it works.
|
|
pub fn generate_id(tkn_str: &str, network: &NetworkName) -> Result<jubjub::Fr> {
|
|
let mut id_string = network.to_string();
|
|
|
|
id_string.push_str(tkn_str);
|
|
|
|
let mut data: Vec<u8> = serialize(&id_string);
|
|
|
|
let token_id = match deserialize::<jubjub::Fr>(&data) {
|
|
Ok(v) => v,
|
|
Err(_) => {
|
|
let mut counter = 0;
|
|
loop {
|
|
data.truncate(28);
|
|
let serialized_counter = serialize(&counter);
|
|
data.extend(serialized_counter.iter());
|
|
let mut hasher = Sha256::new();
|
|
hasher.update(&data);
|
|
let hash = hasher.finalize();
|
|
let token_id = deserialize::<jubjub::Fr>(&hash);
|
|
if token_id.is_err() {
|
|
counter += 1;
|
|
continue;
|
|
}
|
|
return Ok(token_id.unwrap());
|
|
}
|
|
}
|
|
};
|
|
|
|
Ok(token_id)
|
|
}
|
|
|
|
pub fn assign_id(
|
|
network: &NetworkName,
|
|
token: &str,
|
|
sol_tokenlist: &TokenList,
|
|
eth_tokenlist: &TokenList,
|
|
btc_tokenlist: &TokenList,
|
|
) -> Result<String> {
|
|
match network {
|
|
#[cfg(feature = "sol")]
|
|
NetworkName::Solana => {
|
|
// (== 44) can represent a Solana base58 token mint address
|
|
if token.len() == 44 {
|
|
Ok(token.to_string())
|
|
} else {
|
|
let tok_lower = token.to_lowercase();
|
|
symbol_to_id(&tok_lower, sol_tokenlist)
|
|
}
|
|
}
|
|
#[cfg(feature = "btc")]
|
|
NetworkName::Bitcoin => {
|
|
if token.len() == 34 {
|
|
Ok(token.to_string())
|
|
} else {
|
|
let tok_lower = token.to_lowercase();
|
|
symbol_to_id(&tok_lower, btc_tokenlist)
|
|
}
|
|
}
|
|
#[cfg(feature = "eth")]
|
|
NetworkName::Ethereum => {
|
|
use crate::service::eth::ETH_NATIVE_TOKEN_ID;
|
|
// (== 42) can represent a erc20 token mint address
|
|
if token.len() == 42 {
|
|
Ok(token.to_string())
|
|
}
|
|
else if token == "eth" {
|
|
Ok(ETH_NATIVE_TOKEN_ID.to_string())
|
|
} else {
|
|
let tok_lower = token.to_lowercase();
|
|
symbol_to_id(&tok_lower, eth_tokenlist)
|
|
}
|
|
}
|
|
_ => Err(Error::NotSupportedNetwork),
|
|
}
|
|
}
|
|
|
|
pub fn symbol_to_id(token: &str, tokenlist: &TokenList) -> Result<String> {
|
|
let vec: Vec<char> = token.chars().collect();
|
|
let mut counter = 0;
|
|
for c in vec {
|
|
if c.is_alphabetic() {
|
|
counter += 1;
|
|
}
|
|
}
|
|
if counter == token.len() {
|
|
if let Some(id) = tokenlist.search_id(token)? {
|
|
Ok(id)
|
|
} else {
|
|
Err(Error::TokenParseError)
|
|
}
|
|
} else {
|
|
Ok(token.to_string())
|
|
}
|
|
}
|
|
|
|
fn is_digit(c: char) -> bool {
|
|
('0'..='9').contains(&c)
|
|
}
|
|
|
|
fn char_eq(a: char, b: char) -> bool {
|
|
a == b
|
|
}
|
|
|
|
pub fn decode_base10(amount: &str, decimal_places: usize, strict: bool) -> Result<BigUint> {
|
|
let mut s: Vec<char> = amount.to_string().chars().collect();
|
|
|
|
// Get rid of the decimal point:
|
|
let point: usize;
|
|
if let Some(p) = amount.find('.') {
|
|
s.remove(p);
|
|
point = p;
|
|
} else {
|
|
point = s.len();
|
|
}
|
|
|
|
// Only digits should remain
|
|
for i in &s {
|
|
if !is_digit(*i) {
|
|
return Err(Error::ParseFailed("Found non-digits"));
|
|
}
|
|
}
|
|
|
|
// Add digits to the end if there are too few:
|
|
let actual_places = s.len() - point;
|
|
if actual_places < decimal_places {
|
|
s.extend(vec!['0'; decimal_places - actual_places])
|
|
}
|
|
|
|
// Remove digits from the end if there are too many:
|
|
let mut round = false;
|
|
if actual_places > decimal_places {
|
|
let end = point + decimal_places;
|
|
for i in &s[end..s.len()] {
|
|
if !char_eq(*i, '0') {
|
|
round = true;
|
|
break;
|
|
}
|
|
}
|
|
s.truncate(end);
|
|
}
|
|
|
|
if strict && round {
|
|
return Err(Error::ParseFailed("Would end up rounding while strict"));
|
|
}
|
|
|
|
// Convert to an integer
|
|
let number = BigUint::from_str(&String::from_iter(&s))?;
|
|
|
|
// Round and return
|
|
/*
|
|
if round && number == u64::MAX {
|
|
return Err(Error::ParseFailed("u64 overflow"));
|
|
}
|
|
*/
|
|
|
|
Ok(number + round as u64)
|
|
}
|
|
|
|
pub fn encode_base10(amount: BigUint, decimal_places: usize) -> String {
|
|
let mut s: Vec<char> = format!("{:0width$}", amount, width = 1 + decimal_places)
|
|
.chars()
|
|
.collect();
|
|
s.insert(s.len() - decimal_places, '.');
|
|
|
|
String::from_iter(&s)
|
|
.trim_end_matches('0')
|
|
.trim_end_matches('.')
|
|
.to_string()
|
|
}
|
|
|
|
pub fn truncate(amount: u64, decimals: u16, token_decimals: u16) -> Result<u64> {
|
|
let mut amount: Vec<char> = amount.to_string().chars().collect();
|
|
|
|
if token_decimals > decimals {
|
|
if amount.len() <= (token_decimals - decimals) as usize {
|
|
return Ok(0);
|
|
}
|
|
amount.truncate(amount.len() - (token_decimals - decimals) as usize);
|
|
}
|
|
|
|
if token_decimals < decimals {
|
|
amount.resize(amount.len() + (decimals - token_decimals) as usize, '0');
|
|
}
|
|
|
|
let amount = u64::from_str(&String::from_iter(amount))?;
|
|
Ok(amount)
|
|
}
|
|
|
|
#[allow(unused_imports)]
|
|
mod tests {
|
|
use num_bigint::ToBigUint;
|
|
|
|
use super::{decode_base10, encode_base10, truncate};
|
|
|
|
#[test]
|
|
fn test_decode_base10() {
|
|
assert_eq!(
|
|
124.to_biguint().unwrap(),
|
|
decode_base10("12.33", 1, false).unwrap()
|
|
);
|
|
assert_eq!(
|
|
1233000.to_biguint().unwrap(),
|
|
decode_base10("12.33", 5, false).unwrap()
|
|
);
|
|
assert_eq!(
|
|
1200000.to_biguint().unwrap(),
|
|
decode_base10("12.", 5, false).unwrap()
|
|
);
|
|
assert_eq!(
|
|
1200000.to_biguint().unwrap(),
|
|
decode_base10("12", 5, false).unwrap()
|
|
);
|
|
assert!(decode_base10("12.33", 1, true).is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn test_encode_base10() {
|
|
assert_eq!(
|
|
"23.4321111",
|
|
&encode_base10(234321111_u64.to_biguint().unwrap(), 7)
|
|
);
|
|
assert_eq!(
|
|
"23432111.1",
|
|
&encode_base10(234321111_u64.to_biguint().unwrap(), 1)
|
|
);
|
|
assert_eq!(
|
|
"234321.1",
|
|
&encode_base10(2343211_u64.to_biguint().unwrap(), 1)
|
|
);
|
|
assert_eq!(
|
|
"2343211",
|
|
&encode_base10(2343211_u64.to_biguint().unwrap(), 0)
|
|
);
|
|
assert_eq!(
|
|
"0.00002343",
|
|
&encode_base10(2343_u64.to_biguint().unwrap(), 8)
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn test_truncate() {
|
|
// Token decimals is equal to 8
|
|
assert_eq!(100, truncate(100, 8, 8).unwrap());
|
|
assert_eq!(12, truncate(12, 8, 8).unwrap());
|
|
|
|
// Token decimals is bigger than 8
|
|
assert_eq!(100000000, truncate(1000000000, 8, 9).unwrap());
|
|
assert_eq!(10, truncate(100, 8, 9).unwrap());
|
|
assert_eq!(1, truncate(12, 8, 9).unwrap());
|
|
assert_eq!(10, truncate(102, 8, 9).unwrap());
|
|
assert_eq!(0, truncate(1, 8, 9).unwrap());
|
|
assert_eq!(1, truncate(100000000, 8, 16).unwrap());
|
|
assert_eq!(10, truncate(100000000, 8, 15).unwrap());
|
|
assert_eq!(0, truncate(100000000, 8, 17).unwrap());
|
|
assert_eq!(0, truncate(10, 8, 16).unwrap());
|
|
|
|
// Token decimals is less than 8
|
|
assert_eq!(1000, truncate(100, 8, 7).unwrap());
|
|
assert_eq!(12000, truncate(120, 8, 6).unwrap());
|
|
assert_eq!(1000000, truncate(100, 8, 4).unwrap());
|
|
|
|
// token decimals is 0
|
|
assert_eq!(00000000, truncate(0, 8, 0).unwrap());
|
|
assert_eq!(100000000, truncate(1, 8, 0).unwrap());
|
|
|
|
//
|
|
// reverse truncate
|
|
//
|
|
|
|
// Token decimals is less than decimals
|
|
assert_eq!(1000000000, truncate(100000000, 9, 8).unwrap());
|
|
assert_eq!(100000000, truncate(10000000, 9, 8).unwrap());
|
|
assert_eq!(100, truncate(10, 9, 8).unwrap());
|
|
assert_eq!(10, truncate(1, 9, 8).unwrap());
|
|
assert_eq!(100, truncate(10, 9, 8).unwrap());
|
|
assert_eq!(0, truncate(0, 9, 8).unwrap());
|
|
assert_eq!(100000000, truncate(1, 16, 8).unwrap());
|
|
assert_eq!(100000000, truncate(10, 15, 8).unwrap());
|
|
assert_eq!(0, truncate(0, 17, 8).unwrap());
|
|
|
|
// Token decimals is bigger than decimals
|
|
assert_eq!(100, truncate(1000, 7, 8).unwrap());
|
|
assert_eq!(120, truncate(12000, 6, 8).unwrap());
|
|
assert_eq!(100, truncate(1000000, 4, 8).unwrap());
|
|
|
|
// token decimals is 0
|
|
assert_eq!(0, truncate(00000000, 0, 8).unwrap());
|
|
assert_eq!(1, truncate(100000000, 0, 8).unwrap());
|
|
}
|
|
}
|