From 78983b28afb1beaaa02e920d4125ccbe71d77a94 Mon Sep 17 00:00:00 2001 From: akildemir Date: Tue, 15 Oct 2024 13:14:26 +0300 Subject: [PATCH] add oraclize values tests --- Cargo.lock | 4 + substrate/genesis-liquidity/pallet/Cargo.toml | 5 + substrate/genesis-liquidity/pallet/src/lib.rs | 4 +- .../genesis-liquidity/pallet/src/tests.rs | 140 +++++++++++++++++- 4 files changed, 143 insertions(+), 10 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 47088261f..bca0f74f7 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -8199,8 +8199,11 @@ dependencies = [ name = "serai-genesis-liquidity-pallet" version = "0.1.0" dependencies = [ + "ciphersuite", "frame-support", "frame-system", + "frost-schnorrkel", + "modular-frost", "pallet-babe", "pallet-grandpa", "pallet-timestamp", @@ -8219,6 +8222,7 @@ dependencies = [ "sp-io", "sp-runtime", "sp-std", + "zeroize", ] [[package]] diff --git a/substrate/genesis-liquidity/pallet/Cargo.toml b/substrate/genesis-liquidity/pallet/Cargo.toml index 8d8e9d093..59cafb91f 100644 --- a/substrate/genesis-liquidity/pallet/Cargo.toml +++ b/substrate/genesis-liquidity/pallet/Cargo.toml @@ -47,6 +47,11 @@ pallet-timestamp = { git = "https://github.com/serai-dex/substrate", default-fea sp-io = { git = "https://github.com/serai-dex/substrate", default-features = false } sp-runtime = { git = "https://github.com/serai-dex/substrate", default-features = false } +ciphersuite = { path = "../../../crypto/ciphersuite", features = ["ristretto"] } +frost = { package = "modular-frost", path = "../../../crypto/frost", features = ["tests"] } +schnorrkel = { path = "../../../crypto/schnorrkel", package = "frost-schnorrkel" } + +zeroize = "^1.5" rand_core = "0.6" [features] diff --git a/substrate/genesis-liquidity/pallet/src/lib.rs b/substrate/genesis-liquidity/pallet/src/lib.rs index c99d880c2..7cd5c1cd1 100644 --- a/substrate/genesis-liquidity/pallet/src/lib.rs +++ b/substrate/genesis-liquidity/pallet/src/lib.rs @@ -437,9 +437,7 @@ pub mod pallet { Err(InvalidTransaction::Custom(1))?; } - // make sure signers settings the value at the end of the genesis period. - // we don't need this check for tests. - #[cfg(not(feature = "fast-epoch"))] + // check we waited for a month before setting the values if >::block_number().saturated_into::() < MONTHS { Err(InvalidTransaction::Custom(2))?; } diff --git a/substrate/genesis-liquidity/pallet/src/tests.rs b/substrate/genesis-liquidity/pallet/src/tests.rs index 30e263168..fa234a7de 100644 --- a/substrate/genesis-liquidity/pallet/src/tests.rs +++ b/substrate/genesis-liquidity/pallet/src/tests.rs @@ -1,16 +1,28 @@ -use crate::{mock::*, primitives::*}; +use crate::{mock::*, pallet, primitives::*}; use std::collections::HashMap; +use ciphersuite::{Ciphersuite, Ristretto}; +use frost::dkg::musig::musig; +use schnorrkel::Schnorrkel; + use rand_core::{RngCore, OsRng}; +use zeroize::Zeroizing; use frame_system::RawOrigin; -use frame_support::{assert_noop, assert_ok, traits::Hooks}; - -use sp_core::Pair; -use sp_runtime::BoundedVec; - -use validator_sets_primitives::{ValidatorSet, Session, KeyPair}; +use frame_support::{ + assert_noop, assert_ok, + pallet_prelude::{TransactionSource, InvalidTransaction}, + traits::Hooks, +}; + +use sp_core::{ + sr25519::{Pair, Signature}, + Pair as PairTrait, +}; +use sp_runtime::{traits::ValidateUnsigned, BoundedVec}; + +use validator_sets_primitives::{ValidatorSet, Session, KeyPair, musig_context}; use serai_primitives::*; fn set_up_genesis( @@ -139,6 +151,56 @@ fn make_networks_reach_economic_security(block_number: u64) { } } +fn oraclize_values_signature(set: ValidatorSet, values: &Values, pairs: &[Pair]) -> Signature { + let mut pub_keys = vec![]; + for pair in pairs { + let public_key = + ::read_G::<&[u8]>(&mut pair.public().0.as_ref()).unwrap(); + pub_keys.push(public_key); + } + + let mut threshold_keys = vec![]; + for i in 0 .. pairs.len() { + let secret_key = ::read_F::<&[u8]>( + &mut pairs[i].as_ref().secret.to_bytes()[.. 32].as_ref(), + ) + .unwrap(); + assert_eq!(Ristretto::generator() * secret_key, pub_keys[i]); + + threshold_keys.push( + musig::(&musig_context(set), &Zeroizing::new(secret_key), &pub_keys).unwrap(), + ); + } + + let mut musig_keys = HashMap::new(); + for tk in threshold_keys { + musig_keys.insert(tk.params().i(), tk.into()); + } + + let sig = frost::tests::sign_without_caching( + &mut OsRng, + frost::tests::algorithm_machines(&mut OsRng, &Schnorrkel::new(b"substrate"), &musig_keys), + &oraclize_values_message(&set, values), + ); + + Signature(sig.to_bytes()) +} + +fn get_ordered_keys(network: NetworkId, participants: &[Pair]) -> Vec { + // retrieve the current session validators so that we know the order of the keys + // that is necessary for the correct musig signature. + let validators = ValidatorSets::participants_for_latest_decided_set(network).unwrap(); + + // collect the pairs of the validators + let mut pairs = vec![]; + for (v, _) in validators { + let p = participants.iter().find(|pair| pair.public() == v).unwrap().clone(); + pairs.push(p); + } + + pairs +} + #[test] fn genesis_liquidity() { new_test_ext().execute_with(|| { @@ -332,3 +394,67 @@ fn remove_coin_liquidity_after_genesis_period() { assert!(Coins::balance(account, coin.into()).0 > account_coin_balance); }) } + +#[test] +fn validate_oraclize_values_already_done() { + new_test_ext().execute_with(|| { + let values = Values { monero: 184100, ether: 4785000, dai: 1500 }; + + // set the oraclization + GenesisLiquidity::oraclize_values(RawOrigin::None.into(), values, Signature([0u8; 64])) + .unwrap(); + + // trying to oraclize again should fail + let call = pallet::Call::::oraclize_values { values, signature: Signature([0u8; 64]) }; + assert_eq!( + GenesisLiquidity::validate_unsigned(TransactionSource::External, &call), + InvalidTransaction::Custom(1).into() + ); + }) +} + +#[test] +fn validate_oraclize_values_submit_before_a_month() { + new_test_ext().execute_with(|| { + let values = Values { monero: 184100, ether: 4785000, dai: 1500 }; + let call = pallet::Call::::oraclize_values { values, signature: Signature([0u8; 64]) }; + + // we should wait for a month before setting the values + assert_eq!( + GenesisLiquidity::validate_unsigned(TransactionSource::External, &call), + InvalidTransaction::Custom(2).into() + ); + }) +} + +#[test] +fn validate_oraclize_values_invalid_signature() { + new_test_ext().execute_with(|| { + let genesis_participants = vec![ + insecure_pair_from_name("Alice"), + insecure_pair_from_name("Bob"), + insecure_pair_from_name("Charlie"), + insecure_pair_from_name("Dave"), + insecure_pair_from_name("Eve"), + insecure_pair_from_name("Ferdie"), + ]; + let network = NetworkId::Serai; + let values = Values { monero: 184100, ether: 4785000, dai: 1500 }; + + // invalid signature should fail + System::set_block_number(MONTHS); + let call = pallet::Call::::oraclize_values { values, signature: Signature([0u8; 64]) }; + assert_eq!( + GenesisLiquidity::validate_unsigned(TransactionSource::External, &call), + InvalidTransaction::BadProof.into() + ); + + let pairs = get_ordered_keys(network, &genesis_participants); + let signature = + oraclize_values_signature(ValidatorSet { session: Session(0), network }, &values, &pairs); + let call = pallet::Call::::oraclize_values { values, signature }; + + // valid signature should pass + GenesisLiquidity::validate_unsigned(TransactionSource::External, &call).unwrap(); + }) +}