2024-07-18 23:30:19 +00:00
|
|
|
use std::{time::Duration, collections::HashMap};
|
|
|
|
|
2024-08-15 03:12:04 +00:00
|
|
|
use serai_client::Serai;
|
2024-07-18 23:30:19 +00:00
|
|
|
|
2024-08-15 03:12:04 +00:00
|
|
|
use serai_abi::primitives::{Coin, COINS, Amount, GENESIS_SRI};
|
2024-07-18 23:30:19 +00:00
|
|
|
|
2024-08-15 03:12:04 +00:00
|
|
|
use serai_client::genesis_liquidity::primitives::{
|
|
|
|
GENESIS_LIQUIDITY_ACCOUNT, INITIAL_GENESIS_LP_SHARES,
|
2024-07-18 23:30:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
mod common;
|
2024-08-15 03:12:04 +00:00
|
|
|
use common::genesis_liquidity::set_up_genesis;
|
2024-07-18 23:30:19 +00:00
|
|
|
|
|
|
|
serai_test_fast_epoch!(
|
|
|
|
genesis_liquidity: (|serai: Serai| async move {
|
|
|
|
test_genesis_liquidity(serai).await;
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
2024-08-15 03:12:04 +00:00
|
|
|
pub async fn test_genesis_liquidity(serai: Serai) {
|
|
|
|
// set up the genesis
|
2024-07-18 23:30:19 +00:00
|
|
|
let coins = COINS.into_iter().filter(|c| *c != Coin::native()).collect::<Vec<_>>();
|
2024-08-15 03:12:04 +00:00
|
|
|
let values = HashMap::from([(Coin::Monero, 184100), (Coin::Ether, 4785000), (Coin::Dai, 1500)]);
|
|
|
|
let (accounts, _) = set_up_genesis(&serai, &coins, &values).await;
|
|
|
|
|
|
|
|
// wait until genesis is complete
|
|
|
|
while !serai
|
|
|
|
.as_of_latest_finalized_block()
|
|
|
|
.await
|
|
|
|
.unwrap()
|
|
|
|
.genesis_liquidity()
|
|
|
|
.genesis_complete()
|
|
|
|
.await
|
|
|
|
.unwrap()
|
|
|
|
{
|
|
|
|
tokio::time::sleep(Duration::from_secs(1)).await;
|
2024-07-18 23:30:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// check total SRI supply is +100M
|
|
|
|
// there are 6 endowed accounts in dev-net. Take this into consideration when checking
|
|
|
|
// for the total sri minted at this time.
|
|
|
|
let serai = serai.as_of_latest_finalized_block().await.unwrap();
|
|
|
|
let sri = serai.coins().coin_supply(Coin::Serai).await.unwrap();
|
|
|
|
let endowed_amount: u64 = 1 << 60;
|
|
|
|
let total_sri = (6 * endowed_amount) + GENESIS_SRI;
|
|
|
|
assert_eq!(sri, Amount(total_sri));
|
|
|
|
|
|
|
|
// check genesis account has no coins, all transferred to pools.
|
|
|
|
for coin in COINS {
|
|
|
|
let amount = serai.coins().coin_balance(coin, GENESIS_LIQUIDITY_ACCOUNT).await.unwrap();
|
|
|
|
assert_eq!(amount.0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// check pools has proper liquidity
|
|
|
|
let mut pool_amounts = HashMap::new();
|
|
|
|
let mut total_value = 0u128;
|
|
|
|
for coin in coins.clone() {
|
|
|
|
let total_coin = accounts[&coin].iter().fold(0u128, |acc, value| acc + u128::from(value.1 .0));
|
|
|
|
let value = if coin != Coin::Bitcoin {
|
2024-08-15 03:12:04 +00:00
|
|
|
(total_coin * u128::from(values[&coin])) / 10u128.pow(coin.decimals())
|
2024-07-18 23:30:19 +00:00
|
|
|
} else {
|
|
|
|
total_coin
|
|
|
|
};
|
|
|
|
|
|
|
|
total_value += value;
|
|
|
|
pool_amounts.insert(coin, (total_coin, value));
|
|
|
|
}
|
|
|
|
|
|
|
|
// check distributed SRI per pool
|
|
|
|
let mut total_sri_distributed = 0u128;
|
|
|
|
for coin in coins.clone() {
|
|
|
|
let sri = if coin == *COINS.last().unwrap() {
|
|
|
|
u128::from(GENESIS_SRI).checked_sub(total_sri_distributed).unwrap()
|
|
|
|
} else {
|
|
|
|
(pool_amounts[&coin].1 * u128::from(GENESIS_SRI)) / total_value
|
|
|
|
};
|
|
|
|
total_sri_distributed += sri;
|
|
|
|
|
|
|
|
let reserves = serai.dex().get_reserves(coin).await.unwrap().unwrap();
|
|
|
|
assert_eq!(u128::from(reserves.0 .0), pool_amounts[&coin].0); // coin side
|
|
|
|
assert_eq!(u128::from(reserves.1 .0), sri); // SRI side
|
|
|
|
}
|
|
|
|
|
|
|
|
// check each liquidity provider got liquidity tokens proportional to their value
|
|
|
|
for coin in coins {
|
|
|
|
let liq_supply = serai.genesis_liquidity().supply(coin).await.unwrap();
|
|
|
|
for (acc, amount) in &accounts[&coin] {
|
|
|
|
let acc_liq_shares = serai.genesis_liquidity().liquidity(acc, coin).await.unwrap().shares;
|
|
|
|
|
|
|
|
// since we can't test the ratios directly(due to integer division giving 0)
|
|
|
|
// we test whether they give the same result when multiplied by another constant.
|
|
|
|
// Following test ensures the account in fact has the right amount of shares.
|
|
|
|
let mut shares_ratio = (INITIAL_GENESIS_LP_SHARES * acc_liq_shares) / liq_supply.shares;
|
|
|
|
let amounts_ratio =
|
|
|
|
(INITIAL_GENESIS_LP_SHARES * amount.0) / u64::try_from(pool_amounts[&coin].0).unwrap();
|
|
|
|
|
|
|
|
// we can tolerate 1 unit diff between them due to integer division.
|
|
|
|
if shares_ratio.abs_diff(amounts_ratio) == 1 {
|
|
|
|
shares_ratio = amounts_ratio;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert_eq!(shares_ratio, amounts_ratio);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: test remove the liq before/after genesis ended.
|
|
|
|
}
|