diff --git a/frame/staking/src/mock.rs b/frame/staking/src/mock.rs index 4e88b3061..6cde6b8d4 100644 --- a/frame/staking/src/mock.rs +++ b/frame/staking/src/mock.rs @@ -416,6 +416,14 @@ pub fn check_exposure(stash: AccountId) { // ); } +pub fn assert_ledger_consistent(stash: u64) { + assert_is_stash(stash); + let ledger = Staking::ledger(stash - 1).unwrap(); + + assert_eq!(ledger.active_ring, ledger.ring_staking_lock.staking_amount); + assert_eq!(ledger.active_kton, ledger.kton_staking_lock.staking_amount); +} + /// Check that for each nominator: slashable_balance > sum(used_balance) /// Note: we might not consume all of a nominator's balance, but we MUST NOT over spend it. pub fn check_nominator_exposure(stash: AccountId) { diff --git a/frame/staking/src/tests.rs b/frame/staking/src/tests.rs index a4bb3b4f7..b87c93ba2 100644 --- a/frame/staking/src/tests.rs +++ b/frame/staking/src/tests.rs @@ -1,4536 +1,4506 @@ //! Tests for the module. +use crate::sp_api_hidden_includes_decl_storage::hidden_include::sp_runtime::traits::OnInitialize; +use crate::sp_api_hidden_includes_decl_storage::hidden_include::traits::ReservableCurrency; +use darwinia_support::{BalanceLock, NormalLock, StakingLock, WithdrawLock, WithdrawReasons}; +use frame_support::{assert_err, assert_noop, assert_ok}; +use sp_runtime::{ + assert_eq_error_rate, + traits::{ + Bounded, CheckedAdd, CheckedSub, MaybeSerializeDeserialize, Member, Saturating, SimpleArithmetic, StaticLookup, + Zero, + }, + DispatchError, DispatchResult, RuntimeDebug, +}; +use substrate_test_utils::assert_eq_uvec; + +use crate::StakingBalance::RingBalance; use substrate_test_utils::assert_eq_uvec; use crate::{mock::*, *}; -// #[test] -// fn force_unstake_works() { -// // Verifies initial conditions of mock -// ExtBuilder::default().build().execute_with(|| { -// // Account 11 is stashed and locked, and account 10 is the controller -// assert_eq!(Staking::bonded(&11), Some(10)); -// // Cant transfer -// assert_noop!( -// Balances::transfer(Origin::signed(11), 1, 10), -// DispatchError::Module { -// index: 0, -// error: 1, -// message: Some("LiquidityRestrictions"), -// } -// ); -// // Force unstake requires root. -// assert_noop!(Staking::force_unstake(Origin::signed(11), 11), BadOrigin); -// // We now force them to unstake -// assert_ok!(Staking::force_unstake(Origin::ROOT, 11)); -// // No longer bonded. -// assert_eq!(Staking::bonded(&11), None); -// // Transfer works. -// assert_ok!(Balances::transfer(Origin::signed(11), 1, 10)); -// }); -// } -// -// #[test] -// fn basic_setup_works() { -// // Verifies initial conditions of mock -// ExtBuilder::default().build().execute_with(|| { -// // Account 11 is stashed and locked, and account 10 is the controller -// assert_eq!(Staking::bonded(&11), Some(10)); -// // Account 21 is stashed and locked, and account 20 is the controller -// assert_eq!(Staking::bonded(&21), Some(20)); -// // Account 1 is not a stashed -// assert_eq!(Staking::bonded(&1), None); -// -// // Account 10 controls the stash from account 11, which is 100 * balance_factor units -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000, -// active: 1000, -// unlocking: vec![] -// }) -// ); -// // Account 20 controls the stash from account 21, which is 200 * balance_factor units -// assert_eq!( -// Staking::ledger(&20), -// Some(StakingLedger { -// stash: 21, -// total: 1000, -// active: 1000, -// unlocking: vec![] -// }) -// ); -// // Account 1 does not control any stash -// assert_eq!(Staking::ledger(&1), None); -// -// // ValidatorPrefs are default -// assert_eq!( -// >::enumerate().collect::>(), -// vec![ -// (31, ValidatorPrefs::default()), -// (21, ValidatorPrefs::default()), -// (11, ValidatorPrefs::default()) -// ] -// ); -// -// assert_eq!( -// Staking::ledger(100), -// Some(StakingLedger { -// stash: 101, -// total: 500, -// active: 500, -// unlocking: vec![] -// }) -// ); -// assert_eq!(Staking::nominators(101).unwrap().targets, vec![11, 21]); -// -// if cfg!(feature = "equalize") { -// assert_eq!( -// Staking::stakers(11), -// Exposure { -// total: 1250, -// own: 1000, -// others: vec![IndividualExposure { who: 101, value: 250 }] -// } -// ); -// assert_eq!( -// Staking::stakers(21), -// Exposure { -// total: 1250, -// own: 1000, -// others: vec![IndividualExposure { who: 101, value: 250 }] -// } -// ); -// // initial slot_stake -// assert_eq!(Staking::slot_stake(), 1250); -// } else { -// assert_eq!( -// Staking::stakers(11), -// Exposure { -// total: 1125, -// own: 1000, -// others: vec![IndividualExposure { who: 101, value: 125 }] -// } -// ); -// assert_eq!( -// Staking::stakers(21), -// Exposure { -// total: 1375, -// own: 1000, -// others: vec![IndividualExposure { who: 101, value: 375 }] -// } -// ); -// // initial slot_stake -// assert_eq!(Staking::slot_stake(), 1125); -// } -// -// // The number of validators required. -// assert_eq!(Staking::validator_count(), 2); -// -// // Initial Era and session -// assert_eq!(Staking::current_era(), 0); -// -// // Account 10 has `balance_factor` free balance -// assert_eq!(Balances::free_balance(&10), 1); -// assert_eq!(Balances::free_balance(&10), 1); -// -// // New era is not being forced -// assert_eq!(Staking::force_era(), Forcing::NotForcing); -// -// // All exposures must be correct. -// check_exposure_all(); -// check_nominator_all(); -// }); -// } -// -// #[test] -// fn change_controller_works() { -// ExtBuilder::default().build().execute_with(|| { -// assert_eq!(Staking::bonded(&11), Some(10)); -// -// assert!(>::enumerate() -// .map(|(c, _)| c) -// .collect::>() -// .contains(&11)); -// // 10 can control 11 who is initially a validator. -// assert_ok!(Staking::chill(Origin::signed(10))); -// assert!(!>::enumerate() -// .map(|(c, _)| c) -// .collect::>() -// .contains(&11)); -// -// assert_ok!(Staking::set_controller(Origin::signed(11), 5)); -// -// start_era(1); -// -// assert_noop!( -// Staking::validate(Origin::signed(10), ValidatorPrefs::default()), -// Error::::NotController, -// ); -// assert_ok!(Staking::validate(Origin::signed(5), ValidatorPrefs::default())); -// }) -// } -// -// #[test] -// fn rewards_should_work() { -// // should check that: -// // * rewards get recorded per session -// // * rewards get paid per Era -// // * Check that nominators are also rewarded -// ExtBuilder::default().nominate(false).build().execute_with(|| { -// // Init some balances -// let _ = Balances::make_free_balance_be(&2, 500); -// -// let delay = 1000; -// let init_balance_2 = Balances::total_balance(&2); -// let init_balance_10 = Balances::total_balance(&10); -// let init_balance_11 = Balances::total_balance(&11); -// -// // Set payee to controller -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); -// -// // Initial config should be correct -// assert_eq!(Staking::current_era(), 0); -// assert_eq!(Session::current_index(), 0); -// -// // Add a dummy nominator. -// // -// // Equal division indicates that the reward will be equally divided among validator and -// // nominator. -// >::insert( -// &11, -// Exposure { -// own: 500, -// total: 1000, -// others: vec![IndividualExposure { who: 2, value: 500 }], -// }, -// ); -// -// >::insert(&2, RewardDestination::Stash); -// assert_eq!(Staking::payee(2), RewardDestination::Stash); -// assert_eq!(Staking::payee(11), RewardDestination::Controller); -// -// let mut block = 3; // Block 3 => Session 1 => Era 0 -// System::set_block_number(block); -// Timestamp::set_timestamp(block * 5000); // on time. -// Session::on_initialize(System::block_number()); -// assert_eq!(Staking::current_era(), 0); -// assert_eq!(Session::current_index(), 1); -// >::reward_by_ids(vec![(11, 50)]); -// >::reward_by_ids(vec![(11, 50)]); -// // This is the second validator of the current elected set. -// >::reward_by_ids(vec![(21, 50)]); -// // This must be no-op as it is not an elected validator. -// >::reward_by_ids(vec![(1001, 10_000)]); -// -// // Compute total payout now for whole duration as other parameter won't change -// let total_payout = current_total_payout_for_duration(9 * 5 * 1000); -// assert!(total_payout > 10); // Test is meaningful if reward something -// -// // No reward yet -// assert_eq!(Balances::total_balance(&2), init_balance_2); -// assert_eq!(Balances::total_balance(&10), init_balance_10); -// assert_eq!(Balances::total_balance(&11), init_balance_11); -// -// block = 6; // Block 6 => Session 2 => Era 0 -// System::set_block_number(block); -// Timestamp::set_timestamp(block * 5000 + delay); // a little late. -// Session::on_initialize(System::block_number()); -// assert_eq!(Staking::current_era(), 0); -// assert_eq!(Session::current_index(), 2); -// -// block = 9; // Block 9 => Session 3 => Era 1 -// System::set_block_number(block); -// Timestamp::set_timestamp(block * 5000); // back to being on time. no delays -// Session::on_initialize(System::block_number()); -// assert_eq!(Staking::current_era(), 1); -// assert_eq!(Session::current_index(), 3); -// -// // 11 validator has 2/3 of the total rewards and half half for it and its nominator -// assert_eq_error_rate!(Balances::total_balance(&2), init_balance_2 + total_payout / 3, 1); -// assert_eq_error_rate!(Balances::total_balance(&10), init_balance_10 + total_payout / 3, 1); -// assert_eq!(Balances::total_balance(&11), init_balance_11); -// }); -// } -// -// #[test] -// fn multi_era_reward_should_work() { -// // Should check that: -// // The value of current_session_reward is set at the end of each era, based on -// // slot_stake and session_reward. -// ExtBuilder::default().nominate(false).build().execute_with(|| { -// let init_balance_10 = Balances::total_balance(&10); -// -// // Set payee to controller -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); -// -// // Compute now as other parameter won't change -// let total_payout_0 = current_total_payout_for_duration(3000); -// assert!(total_payout_0 > 10); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(11, 1)]); -// -// start_session(0); -// start_session(1); -// start_session(2); -// start_session(3); -// -// assert_eq!(Staking::current_era(), 1); -// assert_eq!(Balances::total_balance(&10), init_balance_10 + total_payout_0); -// -// start_session(4); -// -// let total_payout_1 = current_total_payout_for_duration(3000); -// assert!(total_payout_1 > 10); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(11, 101)]); -// -// // new era is triggered here. -// start_session(5); -// -// // pay time -// assert_eq!( -// Balances::total_balance(&10), -// init_balance_10 + total_payout_0 + total_payout_1 -// ); -// }); -// } -// -// #[test] -// fn staking_should_work() { -// // should test: -// // * new validators can be added to the default set -// // * new ones will be chosen per era -// // * either one can unlock the stash and back-down from being a validator via `chill`ing. -// ExtBuilder::default() -// .nominate(false) -// .fair(false) // to give 20 more staked value -// .build() -// .execute_with(|| { -// Timestamp::set_timestamp(1); // Initialize time. -// -// // remember + compare this along with the test. -// assert_eq_uvec!(validator_controllers(), vec![20, 10]); -// -// // put some money in account that we'll use. -// for i in 1..5 { -// let _ = Balances::make_free_balance_be(&i, 2000); -// } -// -// // --- Block 1: -// start_session(1); -// // add a new candidate for being a validator. account 3 controlled by 4. -// assert_ok!(Staking::bond(Origin::signed(3), 4, 1500, RewardDestination::Controller)); -// assert_ok!(Staking::validate(Origin::signed(4), ValidatorPrefs::default())); -// -// // No effects will be seen so far. -// assert_eq_uvec!(validator_controllers(), vec![20, 10]); -// -// // --- Block 2: -// start_session(2); -// -// // No effects will be seen so far. Era has not been yet triggered. -// assert_eq_uvec!(validator_controllers(), vec![20, 10]); -// -// // --- Block 3: the validators will now be queued. -// start_session(3); -// assert_eq!(Staking::current_era(), 1); -// -// // --- Block 4: the validators will now be changed. -// start_session(4); -// -// assert_eq_uvec!(validator_controllers(), vec![20, 4]); -// // --- Block 4: Unstake 4 as a validator, freeing up the balance stashed in 3 -// // 4 will chill -// Staking::chill(Origin::signed(4)).unwrap(); -// -// // --- Block 5: nothing. 4 is still there. -// start_session(5); -// assert_eq_uvec!(validator_controllers(), vec![20, 4]); -// -// // --- Block 6: 4 will not be a validator. -// start_session(7); -// assert_eq_uvec!(validator_controllers(), vec![20, 10]); -// -// // Note: the stashed value of 4 is still lock -// assert_eq!( -// Staking::ledger(&4), -// Some(StakingLedger { -// stash: 3, -// total: 1500, -// active: 1500, -// unlocking: vec![] -// }) -// ); -// // e.g. it cannot spend more than 500 that it has free from the total 2000 -// assert_noop!( -// Balances::reserve(&3, 501), -// DispatchError::Module { -// index: 0, -// error: 1, -// message: Some("LiquidityRestrictions"), -// } -// ); -// assert_ok!(Balances::reserve(&3, 409)); -// }); -// } -// -// #[test] -// fn less_than_needed_candidates_works() { -// ExtBuilder::default() -// .minimum_validator_count(1) -// .validator_count(4) -// .nominate(false) -// .num_validators(3) -// .build() -// .execute_with(|| { -// assert_eq!(Staking::validator_count(), 4); -// assert_eq!(Staking::minimum_validator_count(), 1); -// assert_eq_uvec!(validator_controllers(), vec![30, 20, 10]); -// -// start_era(1); -// -// // Previous set is selected. NO election algorithm is even executed. -// assert_eq_uvec!(validator_controllers(), vec![30, 20, 10]); -// -// // But the exposure is updated in a simple way. No external votes exists. -// // This is purely self-vote. -// assert_eq!(Staking::stakers(10).others.len(), 0); -// assert_eq!(Staking::stakers(20).others.len(), 0); -// assert_eq!(Staking::stakers(30).others.len(), 0); -// check_exposure_all(); -// check_nominator_all(); -// }); -// } -// -// #[test] -// fn no_candidate_emergency_condition() { -// ExtBuilder::default() -// .minimum_validator_count(10) -// .validator_count(15) -// .num_validators(4) -// .validator_pool(true) -// .nominate(false) -// .build() -// .execute_with(|| { -// // initial validators -// assert_eq_uvec!(validator_controllers(), vec![10, 20, 30, 40]); -// -// // set the minimum validator count. -// ::MinimumValidatorCount::put(10); -// ::ValidatorCount::put(15); -// assert_eq!(Staking::validator_count(), 15); -// -// let _ = Staking::chill(Origin::signed(10)); -// -// // trigger era -// System::set_block_number(1); -// Session::on_initialize(System::block_number()); -// -// // Previous ones are elected. chill is invalidates. TODO: #2494 -// assert_eq_uvec!(validator_controllers(), vec![10, 20, 30, 40]); -// assert_eq!(Staking::current_elected().len(), 0); -// }); -// } -// -// #[test] -// fn nominating_and_rewards_should_work() { -// // PHRAGMEN OUTPUT: running this test with the reference impl gives: -// // -// // Sequential Phragmén gives -// // 10 is elected with stake 2200.0 and score 0.0003333333333333333 -// // 20 is elected with stake 1800.0 and score 0.0005555555555555556 -// -// // 10 has load 0.0003333333333333333 and supported -// // 10 with stake 1000.0 -// // 20 has load 0.0005555555555555556 and supported -// // 20 with stake 1000.0 -// // 30 has load 0 and supported -// // 30 with stake 0 -// // 40 has load 0 and supported -// // 40 with stake 0 -// // 2 has load 0.0005555555555555556 and supported -// // 10 with stake 600.0 20 with stake 400.0 30 with stake 0.0 -// // 4 has load 0.0005555555555555556 and supported -// // 10 with stake 600.0 20 with stake 400.0 40 with stake 0.0 -// -// // Sequential Phragmén with post processing gives -// // 10 is elected with stake 2000.0 and score 0.0003333333333333333 -// // 20 is elected with stake 2000.0 and score 0.0005555555555555556 -// -// // 10 has load 0.0003333333333333333 and supported -// // 10 with stake 1000.0 -// // 20 has load 0.0005555555555555556 and supported -// // 20 with stake 1000.0 -// // 30 has load 0 and supported -// // 30 with stake 0 -// // 40 has load 0 and supported -// // 40 with stake 0 -// // 2 has load 0.0005555555555555556 and supported -// // 10 with stake 400.0 20 with stake 600.0 30 with stake 0 -// // 4 has load 0.0005555555555555556 and supported -// // 10 with stake 600.0 20 with stake 400.0 40 with stake 0.0 -// ExtBuilder::default() -// .nominate(false) -// .validator_pool(true) -// .build() -// .execute_with(|| { -// // initial validators -- everyone is actually even. -// assert_eq_uvec!(validator_controllers(), vec![40, 30]); -// -// // Set payee to controller -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); -// assert_ok!(Staking::set_payee(Origin::signed(20), RewardDestination::Controller)); -// assert_ok!(Staking::set_payee(Origin::signed(30), RewardDestination::Controller)); -// assert_ok!(Staking::set_payee(Origin::signed(40), RewardDestination::Controller)); -// -// // give the man some money -// let initial_balance = 1000; -// for i in [1, 2, 3, 4, 5, 10, 11, 20, 21].iter() { -// let _ = Balances::make_free_balance_be(i, initial_balance); -// } -// -// // bond two account pairs and state interest in nomination. -// // 2 will nominate for 10, 20, 30 -// assert_ok!(Staking::bond(Origin::signed(1), 2, 1000, RewardDestination::Controller)); -// assert_ok!(Staking::nominate(Origin::signed(2), vec![11, 21, 31])); -// // 4 will nominate for 10, 20, 40 -// assert_ok!(Staking::bond(Origin::signed(3), 4, 1000, RewardDestination::Controller)); -// assert_ok!(Staking::nominate(Origin::signed(4), vec![11, 21, 41])); -// -// // the total reward for era 0 -// let total_payout_0 = current_total_payout_for_duration(3000); -// assert!(total_payout_0 > 100); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(41, 1)]); -// >::reward_by_ids(vec![(31, 1)]); -// >::reward_by_ids(vec![(21, 10)]); // must be no-op -// >::reward_by_ids(vec![(11, 10)]); // must be no-op -// -// start_era(1); -// -// // 10 and 20 have more votes, they will be chosen by phragmen. -// assert_eq_uvec!(validator_controllers(), vec![20, 10]); -// -// // OLD validators must have already received some rewards. -// assert_eq!(Balances::total_balance(&40), 1 + total_payout_0 / 2); -// assert_eq!(Balances::total_balance(&30), 1 + total_payout_0 / 2); -// -// // ------ check the staked value of all parties. -// -// if cfg!(feature = "equalize") { -// // total expo of 10, with 1200 coming from nominators (externals), according to phragmen. -// assert_eq!(Staking::stakers(11).own, 1000); -// assert_eq_error_rate!(Staking::stakers(11).total, 1000 + 1000, 2); -// // 2 and 4 supported 10, each with stake 600, according to phragmen. -// assert_eq!( -// Staking::stakers(11) -// .others -// .iter() -// .map(|e| e.value) -// .collect::>>(), -// vec![600, 400] -// ); -// assert_eq!( -// Staking::stakers(11).others.iter().map(|e| e.who).collect::>(), -// vec![3, 1] -// ); -// // total expo of 20, with 500 coming from nominators (externals), according to phragmen. -// assert_eq!(Staking::stakers(21).own, 1000); -// assert_eq_error_rate!(Staking::stakers(21).total, 1000 + 1000, 2); -// // 2 and 4 supported 20, each with stake 250, according to phragmen. -// assert_eq!( -// Staking::stakers(21) -// .others -// .iter() -// .map(|e| e.value) -// .collect::>>(), -// vec![400, 600] -// ); -// assert_eq!( -// Staking::stakers(21).others.iter().map(|e| e.who).collect::>(), -// vec![3, 1] -// ); -// } else { -// // total expo of 10, with 1200 coming from nominators (externals), according to phragmen. -// assert_eq!(Staking::stakers(11).own, 1000); -// assert_eq!(Staking::stakers(11).total, 1000 + 800); -// // 2 and 4 supported 10, each with stake 600, according to phragmen. -// assert_eq!( -// Staking::stakers(11) -// .others -// .iter() -// .map(|e| e.value) -// .collect::>>(), -// vec![400, 400] -// ); -// assert_eq!( -// Staking::stakers(11).others.iter().map(|e| e.who).collect::>(), -// vec![3, 1] -// ); -// // total expo of 20, with 500 coming from nominators (externals), according to phragmen. -// assert_eq!(Staking::stakers(21).own, 1000); -// assert_eq_error_rate!(Staking::stakers(21).total, 1000 + 1200, 2); -// // 2 and 4 supported 20, each with stake 250, according to phragmen. -// assert_eq!( -// Staking::stakers(21) -// .others -// .iter() -// .map(|e| e.value) -// .collect::>>(), -// vec![600, 600] -// ); -// assert_eq!( -// Staking::stakers(21).others.iter().map(|e| e.who).collect::>(), -// vec![3, 1] -// ); -// } -// -// // They are not chosen anymore -// assert_eq!(Staking::stakers(31).total, 0); -// assert_eq!(Staking::stakers(41).total, 0); -// -// // the total reward for era 1 -// let total_payout_1 = current_total_payout_for_duration(3000); -// assert!(total_payout_1 > 100); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(41, 10)]); // must be no-op -// >::reward_by_ids(vec![(31, 10)]); // must be no-op -// >::reward_by_ids(vec![(21, 2)]); -// >::reward_by_ids(vec![(11, 1)]); -// -// start_era(2); -// -// // nothing else will happen, era ends and rewards are paid again, -// // it is expected that nominators will also be paid. See below -// -// let payout_for_10 = total_payout_1 / 3; -// let payout_for_20 = 2 * total_payout_1 / 3; -// if cfg!(feature = "equalize") { -// // Nominator 2: has [400 / 2000 ~ 1 / 5 from 10] + [600 / 2000 ~ 3 / 10 from 20]'s reward. -// assert_eq_error_rate!( -// Balances::total_balance(&2), -// initial_balance + payout_for_10 / 5 + payout_for_20 * 3 / 10, -// 2, -// ); -// // Nominator 4: has [400 / 2000 ~ 1 / 5 from 20] + [600 / 2000 ~ 3 / 10 from 10]'s reward. -// assert_eq_error_rate!( -// Balances::total_balance(&4), -// initial_balance + payout_for_20 / 5 + payout_for_10 * 3 / 10, -// 2, -// ); -// -// // Validator 10: got 1000 / 2000 external stake. -// assert_eq_error_rate!(Balances::total_balance(&10), initial_balance + payout_for_10 / 2, 1,); -// // Validator 20: got 1000 / 2000 external stake. -// assert_eq_error_rate!(Balances::total_balance(&20), initial_balance + payout_for_20 / 2, 1,); -// } else { -// // Nominator 2: has [400/1800 ~ 2/9 from 10] + [600/2200 ~ 3/11 from 20]'s reward. ==> 2/9 + 3/11 -// assert_eq_error_rate!( -// Balances::total_balance(&2), -// initial_balance + (2 * payout_for_10 / 9 + 3 * payout_for_20 / 11), -// 1, -// ); -// // Nominator 4: has [400/1800 ~ 2/9 from 10] + [600/2200 ~ 3/11 from 20]'s reward. ==> 2/9 + 3/11 -// assert_eq_error_rate!( -// Balances::total_balance(&4), -// initial_balance + (2 * payout_for_10 / 9 + 3 * payout_for_20 / 11), -// 1, -// ); -// -// // Validator 10: got 800 / 1800 external stake => 8/18 =? 4/9 => Validator's share = 5/9 -// assert_eq_error_rate!(Balances::total_balance(&10), initial_balance + 5 * payout_for_10 / 9, 1,); -// // Validator 20: got 1200 / 2200 external stake => 12/22 =? 6/11 => Validator's share = 5/11 -// assert_eq_error_rate!( -// Balances::total_balance(&20), -// initial_balance + 5 * payout_for_20 / 11, -// 1, -// ); -// } -// -// check_exposure_all(); -// check_nominator_all(); -// }); -// } -// -// #[test] -// fn nominators_also_get_slashed() { -// // A nominator should be slashed if the validator they nominated is slashed -// // Here is the breakdown of roles: -// // 10 - is the controller of 11 -// // 11 - is the stash. -// // 2 - is the nominator of 20, 10 -// ExtBuilder::default().nominate(false).build().execute_with(|| { -// assert_eq!(Staking::validator_count(), 2); -// -// // Set payee to controller -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); -// -// // give the man some money. -// let initial_balance = 1000; -// for i in [1, 2, 3, 10].iter() { -// let _ = Balances::make_free_balance_be(i, initial_balance); -// } -// -// // 2 will nominate for 10, 20 -// let nominator_stake = 500; -// assert_ok!(Staking::bond( -// Origin::signed(1), -// 2, -// nominator_stake, -// RewardDestination::default() -// )); -// assert_ok!(Staking::nominate(Origin::signed(2), vec![20, 10])); -// -// let total_payout = current_total_payout_for_duration(3000); -// assert!(total_payout > 100); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(11, 1)]); -// -// // new era, pay rewards, -// start_era(1); -// -// // Nominator stash didn't collect any. -// assert_eq!(Balances::total_balance(&2), initial_balance); -// -// // 10 goes offline -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(5)], -// ); -// let expo = Staking::stakers(11); -// let slash_value = 50; -// let total_slash = expo.total.min(slash_value); -// let validator_slash = expo.own.min(total_slash); -// let nominator_slash = nominator_stake.min(total_slash - validator_slash); -// -// // initial + first era reward + slash -// assert_eq!(Balances::total_balance(&11), initial_balance - validator_slash); -// assert_eq!(Balances::total_balance(&2), initial_balance - nominator_slash); -// check_exposure_all(); -// check_nominator_all(); -// // Because slashing happened. -// assert!(is_disabled(10)); -// }); -// } -// -// #[test] -// fn double_staking_should_fail() { -// // should test (in the same order): -// // * an account already bonded as stash cannot be be stashed again. -// // * an account already bonded as stash cannot nominate. -// // * an account already bonded as controller can nominate. -// ExtBuilder::default().build().execute_with(|| { -// let arbitrary_value = 5; -// // 2 = controller, 1 stashed => ok -// assert_ok!(Staking::bond( -// Origin::signed(1), -// 2, -// arbitrary_value, -// RewardDestination::default() -// )); -// // 4 = not used so far, 1 stashed => not allowed. -// assert_noop!( -// Staking::bond(Origin::signed(1), 4, arbitrary_value, RewardDestination::default()), -// Error::::AlreadyBonded, -// ); -// // 1 = stashed => attempting to nominate should fail. -// assert_noop!( -// Staking::nominate(Origin::signed(1), vec![1]), -// Error::::NotController -// ); -// // 2 = controller => nominating should work. -// assert_ok!(Staking::nominate(Origin::signed(2), vec![1])); -// }); -// } -// -// #[test] -// fn double_controlling_should_fail() { -// // should test (in the same order): -// // * an account already bonded as controller CANNOT be reused as the controller of another account. -// ExtBuilder::default().build().execute_with(|| { -// let arbitrary_value = 5; -// // 2 = controller, 1 stashed => ok -// assert_ok!(Staking::bond( -// Origin::signed(1), -// 2, -// arbitrary_value, -// RewardDestination::default(), -// )); -// // 2 = controller, 3 stashed (Note that 2 is reused.) => no-op -// assert_noop!( -// Staking::bond(Origin::signed(3), 2, arbitrary_value, RewardDestination::default()), -// Error::::AlreadyPaired, -// ); -// }); -// } -// -// #[test] -// fn session_and_eras_work() { -// ExtBuilder::default().build().execute_with(|| { -// assert_eq!(Staking::current_era(), 0); -// -// // Block 1: No change. -// start_session(0); -// assert_eq!(Session::current_index(), 1); -// assert_eq!(Staking::current_era(), 0); -// -// // Block 2: Simple era change. -// start_session(2); -// assert_eq!(Session::current_index(), 3); -// assert_eq!(Staking::current_era(), 1); -// -// // Block 3: Schedule an era length change; no visible changes. -// start_session(3); -// assert_eq!(Session::current_index(), 4); -// assert_eq!(Staking::current_era(), 1); -// -// // Block 4: Era change kicks in. -// start_session(5); -// assert_eq!(Session::current_index(), 6); -// assert_eq!(Staking::current_era(), 2); -// -// // Block 5: No change. -// start_session(6); -// assert_eq!(Session::current_index(), 7); -// assert_eq!(Staking::current_era(), 2); -// -// // Block 6: No change. -// start_session(7); -// assert_eq!(Session::current_index(), 8); -// assert_eq!(Staking::current_era(), 2); -// -// // Block 7: Era increment. -// start_session(8); -// assert_eq!(Session::current_index(), 9); -// assert_eq!(Staking::current_era(), 3); -// }); -// } -// -// #[test] -// fn forcing_new_era_works() { -// ExtBuilder::default().build().execute_with(|| { -// // normal flow of session. -// assert_eq!(Staking::current_era(), 0); -// start_session(0); -// assert_eq!(Staking::current_era(), 0); -// start_session(1); -// assert_eq!(Staking::current_era(), 0); -// start_session(2); -// assert_eq!(Staking::current_era(), 1); -// -// // no era change. -// ForceEra::put(Forcing::ForceNone); -// start_session(3); -// assert_eq!(Staking::current_era(), 1); -// start_session(4); -// assert_eq!(Staking::current_era(), 1); -// start_session(5); -// assert_eq!(Staking::current_era(), 1); -// start_session(6); -// assert_eq!(Staking::current_era(), 1); -// -// // back to normal. -// // this immediately starts a new session. -// ForceEra::put(Forcing::NotForcing); -// start_session(7); -// assert_eq!(Staking::current_era(), 2); -// start_session(8); -// assert_eq!(Staking::current_era(), 2); -// -// // forceful change -// ForceEra::put(Forcing::ForceAlways); -// start_session(9); -// assert_eq!(Staking::current_era(), 3); -// start_session(10); -// assert_eq!(Staking::current_era(), 4); -// start_session(11); -// assert_eq!(Staking::current_era(), 5); -// -// // just one forceful change -// ForceEra::put(Forcing::ForceNew); -// start_session(12); -// assert_eq!(Staking::current_era(), 6); -// -// assert_eq!(ForceEra::get(), Forcing::NotForcing); -// start_session(13); -// assert_eq!(Staking::current_era(), 6); -// }); -// } -// -// #[test] -// fn cannot_transfer_staked_balance() { -// // Tests that a stash account cannot transfer funds -// ExtBuilder::default().nominate(false).build().execute_with(|| { -// // Confirm account 11 is stashed -// assert_eq!(Staking::bonded(&11), Some(10)); -// // Confirm account 11 has some free balance -// assert_eq!(Balances::free_balance(&11), 1000); -// // Confirm account 11 (via controller 10) is totally staked -// assert_eq!(Staking::stakers(&11).total, 1000); -// // Confirm account 11 cannot transfer as a result -// assert_noop!( -// Balances::transfer(Origin::signed(11), 20, 1), -// DispatchError::Module { -// index: 0, -// error: 1, -// message: Some("LiquidityRestrictions"), -// } -// ); -// -// // Give account 11 extra free balance -// let _ = Balances::make_free_balance_be(&11, 10000); -// // Confirm that account 11 can now transfer some balance -// assert_ok!(Balances::transfer(Origin::signed(11), 20, 1)); -// }); -// } -// -// #[test] -// fn cannot_transfer_staked_balance_2() { -// // Tests that a stash account cannot transfer funds -// // Same test as above but with 20, and more accurate. -// // 21 has 2000 free balance but 1000 at stake -// ExtBuilder::default() -// .nominate(false) -// .fair(true) -// .build() -// .execute_with(|| { -// // Confirm account 21 is stashed -// assert_eq!(Staking::bonded(&21), Some(20)); -// // Confirm account 21 has some free balance -// assert_eq!(Balances::free_balance(&21), 2000); -// // Confirm account 21 (via controller 20) is totally staked -// assert_eq!(Staking::stakers(&21).total, 1000); -// // Confirm account 21 can transfer at most 1000 -// assert_noop!( -// Balances::transfer(Origin::signed(21), 20, 1001), -// DispatchError::Module { -// index: 0, -// error: 1, -// message: Some("LiquidityRestrictions"), -// } -// ); -// assert_ok!(Balances::transfer(Origin::signed(21), 20, 1000)); -// }); -// } -// -// #[test] -// fn cannot_reserve_staked_balance() { -// // Checks that a bonded account cannot reserve balance from free balance -// ExtBuilder::default().build().execute_with(|| { -// // Confirm account 11 is stashed -// assert_eq!(Staking::bonded(&11), Some(10)); -// // Confirm account 11 has some free balance -// assert_eq!(Balances::free_balance(&11), 1000); -// // Confirm account 11 (via controller 10) is totally staked -// assert_eq!(Staking::stakers(&11).own, 1000); -// // Confirm account 11 cannot transfer as a result -// assert_noop!( -// Balances::reserve(&11, 1), -// DispatchError::Module { -// index: 0, -// error: 1, -// message: Some("LiquidityRestrictions"), -// } -// ); -// -// // Give account 11 extra free balance -// let _ = Balances::make_free_balance_be(&11, 10000); -// // Confirm account 11 can now reserve balance -// assert_ok!(Balances::reserve(&11, 1)); -// }); -// } -// -// #[test] -// fn reward_destination_works() { -// // Rewards go to the correct destination as determined in Payee -// ExtBuilder::default().nominate(false).build().execute_with(|| { -// // Check that account 11 is a validator -// assert!(Staking::current_elected().contains(&11)); -// // Check the balance of the validator account -// assert_eq!(Balances::free_balance(&10), 1); -// // Check the balance of the stash account -// assert_eq!(Balances::free_balance(&11), 1000); -// // Check how much is at stake -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000, -// active: 1000, -// unlocking: vec![], -// }) -// ); -// -// // Compute total payout now for whole duration as other parameter won't change -// let total_payout_0 = current_total_payout_for_duration(3000); -// assert!(total_payout_0 > 100); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(11, 1)]); -// -// start_era(1); -// -// // Check that RewardDestination is Staked (default) -// assert_eq!(Staking::payee(&11), RewardDestination::Staked); -// // Check that reward went to the stash account of validator -// assert_eq!(Balances::free_balance(&11), 1000 + total_payout_0); -// // Check that amount at stake increased accordingly -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000 + total_payout_0, -// active: 1000 + total_payout_0, -// unlocking: vec![], -// }) -// ); -// -// //Change RewardDestination to Stash -// >::insert(&11, RewardDestination::Stash); -// -// // Compute total payout now for whole duration as other parameter won't change -// let total_payout_1 = current_total_payout_for_duration(3000); -// assert!(total_payout_1 > 100); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(11, 1)]); -// -// start_era(2); -// -// // Check that RewardDestination is Stash -// assert_eq!(Staking::payee(&11), RewardDestination::Stash); -// // Check that reward went to the stash account -// assert_eq!(Balances::free_balance(&11), 1000 + total_payout_0 + total_payout_1); -// // Record this value -// let recorded_stash_balance = 1000 + total_payout_0 + total_payout_1; -// // Check that amount at stake is NOT increased -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000 + total_payout_0, -// active: 1000 + total_payout_0, -// unlocking: vec![], -// }) -// ); -// -// // Change RewardDestination to Controller -// >::insert(&11, RewardDestination::Controller); -// -// // Check controller balance -// assert_eq!(Balances::free_balance(&10), 1); -// -// // Compute total payout now for whole duration as other parameter won't change -// let total_payout_2 = current_total_payout_for_duration(3000); -// assert!(total_payout_2 > 100); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(11, 1)]); -// -// start_era(3); -// -// // Check that RewardDestination is Controller -// assert_eq!(Staking::payee(&11), RewardDestination::Controller); -// // Check that reward went to the controller account -// assert_eq!(Balances::free_balance(&10), 1 + total_payout_2); -// // Check that amount at stake is NOT increased -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000 + total_payout_0, -// active: 1000 + total_payout_0, -// unlocking: vec![], -// }) -// ); -// // Check that amount in staked account is NOT increased. -// assert_eq!(Balances::free_balance(&11), recorded_stash_balance); -// }); -// } -// -// #[test] -// fn validator_payment_prefs_work() { -// // Test that validator preferences are correctly honored -// // Note: unstake threshold is being directly tested in slashing tests. -// // This test will focus on validator payment. -// ExtBuilder::default().build().execute_with(|| { -// // Initial config -// let stash_initial_balance = Balances::total_balance(&11); -// -// // check the balance of a validator accounts. -// assert_eq!(Balances::total_balance(&10), 1); -// // check the balance of a validator's stash accounts. -// assert_eq!(Balances::total_balance(&11), stash_initial_balance); -// // and the nominator (to-be) -// let _ = Balances::make_free_balance_be(&2, 500); -// -// // add a dummy nominator. -// >::insert( -// &11, -// Exposure { -// own: 500, // equal division indicates that the reward will be equally divided among validator and nominator. -// total: 1000, -// others: vec![IndividualExposure { who: 2, value: 500 }], -// }, -// ); -// >::insert(&2, RewardDestination::Stash); -// >::insert( -// &11, -// ValidatorPrefs { -// commission: Perbill::from_percent(50), -// }, -// ); -// -// // Compute total payout now for whole duration as other parameter won't change -// let total_payout_0 = current_total_payout_for_duration(3000); -// assert!(total_payout_0 > 100); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(11, 1)]); -// -// start_era(1); -// -// // whats left to be shared is the sum of 3 rounds minus the validator's cut. -// let shared_cut = total_payout_0 / 2; -// // Validator's payee is Staked account, 11, reward will be paid here. -// assert_eq!( -// Balances::total_balance(&11), -// stash_initial_balance + shared_cut / 2 + shared_cut -// ); -// // Controller account will not get any reward. -// assert_eq!(Balances::total_balance(&10), 1); -// // Rest of the reward will be shared and paid to the nominator in stake. -// assert_eq!(Balances::total_balance(&2), 500 + shared_cut / 2); -// -// check_exposure_all(); -// check_nominator_all(); -// }); -// } -// -// #[test] -// fn bond_extra_works() { -// // Tests that extra `free_balance` in the stash can be added to stake -// // NOTE: this tests only verifies `StakingLedger` for correct updates -// // See `bond_extra_and_withdraw_unbonded_works` for more details and updates on `Exposure`. -// ExtBuilder::default().build().execute_with(|| { -// // Check that account 10 is a validator -// assert!(>::exists(11)); -// // Check that account 10 is bonded to account 11 -// assert_eq!(Staking::bonded(&11), Some(10)); -// // Check how much is at stake -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000, -// active: 1000, -// unlocking: vec![], -// }) -// ); -// -// // Give account 11 some large free balance greater than total -// let _ = Balances::make_free_balance_be(&11, 1000000); -// -// // Call the bond_extra function from controller, add only 100 -// assert_ok!(Staking::bond_extra(Origin::signed(11), 100)); -// // There should be 100 more `total` and `active` in the ledger -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000 + 100, -// active: 1000 + 100, -// unlocking: vec![], -// }) -// ); -// -// // Call the bond_extra function with a large number, should handle it -// assert_ok!(Staking::bond_extra(Origin::signed(11), u64::max_value())); -// // The full amount of the funds should now be in the total and active -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000000, -// active: 1000000, -// unlocking: vec![], -// }) -// ); -// }); -// } -// -// #[test] -// fn bond_extra_and_withdraw_unbonded_works() { -// // * Should test -// // * Given an account being bonded [and chosen as a validator](not mandatory) -// // * It can add extra funds to the bonded account. -// // * it can unbond a portion of its funds from the stash account. -// // * Once the unbonding period is done, it can actually take the funds out of the stash. -// ExtBuilder::default().nominate(false).build().execute_with(|| { -// // Set payee to controller. avoids confusion -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); -// -// // Give account 11 some large free balance greater than total -// let _ = Balances::make_free_balance_be(&11, 1000000); -// -// // Initial config should be correct -// assert_eq!(Staking::current_era(), 0); -// assert_eq!(Session::current_index(), 0); -// -// // check the balance of a validator accounts. -// assert_eq!(Balances::total_balance(&10), 1); -// -// // confirm that 10 is a normal validator and gets paid at the end of the era. -// start_era(1); -// -// // Initial state of 10 -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000, -// active: 1000, -// unlocking: vec![], -// }) -// ); -// assert_eq!( -// Staking::stakers(&11), -// Exposure { -// total: 1000, -// own: 1000, -// others: vec![] -// } -// ); -// -// // deposit the extra 100 units -// Staking::bond_extra(Origin::signed(11), 100).unwrap(); -// -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000 + 100, -// active: 1000 + 100, -// unlocking: vec![], -// }) -// ); -// // Exposure is a snapshot! only updated after the next era update. -// assert_ne!( -// Staking::stakers(&11), -// Exposure { -// total: 1000 + 100, -// own: 1000 + 100, -// others: vec![] -// } -// ); -// -// // trigger next era. -// Timestamp::set_timestamp(10); -// start_era(2); -// assert_eq!(Staking::current_era(), 2); -// -// // ledger should be the same. -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000 + 100, -// active: 1000 + 100, -// unlocking: vec![], -// }) -// ); -// // Exposure is now updated. -// assert_eq!( -// Staking::stakers(&11), -// Exposure { -// total: 1000 + 100, -// own: 1000 + 100, -// others: vec![] -// } -// ); -// -// // Unbond almost all of the funds in stash. -// Staking::unbond(Origin::signed(10), 1000).unwrap(); -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000 + 100, -// active: 100, -// unlocking: vec![UnlockChunk { -// value: 1000, -// era: 2 + 3 -// }] -// }) -// ); -// -// // Attempting to free the balances now will fail. 2 eras need to pass. -// Staking::withdraw_unbonded(Origin::signed(10)).unwrap(); -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000 + 100, -// active: 100, -// unlocking: vec![UnlockChunk { -// value: 1000, -// era: 2 + 3 -// }] -// }) -// ); -// -// // trigger next era. -// start_era(3); -// -// // nothing yet -// Staking::withdraw_unbonded(Origin::signed(10)).unwrap(); -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 1000 + 100, -// active: 100, -// unlocking: vec![UnlockChunk { -// value: 1000, -// era: 2 + 3 -// }] -// }) -// ); -// -// // trigger next era. -// start_era(5); -// -// Staking::withdraw_unbonded(Origin::signed(10)).unwrap(); -// // Now the value is free and the staking ledger is updated. -// assert_eq!( -// Staking::ledger(&10), -// Some(StakingLedger { -// stash: 11, -// total: 100, -// active: 100, -// unlocking: vec![] -// }) -// ); -// }) -// } -// -// #[test] -// fn too_many_unbond_calls_should_not_work() { -// ExtBuilder::default().build().execute_with(|| { -// // locked at era 0 until 3 -// for _ in 0..MAX_UNLOCKING_CHUNKS - 1 { -// assert_ok!(Staking::unbond(Origin::signed(10), 1)); -// } -// -// start_era(1); -// -// // locked at era 1 until 4 -// assert_ok!(Staking::unbond(Origin::signed(10), 1)); -// // can't do more. -// assert_noop!(Staking::unbond(Origin::signed(10), 1), Error::::NoMoreChunks); -// -// start_era(3); -// -// assert_noop!(Staking::unbond(Origin::signed(10), 1), Error::::NoMoreChunks); -// // free up. -// assert_ok!(Staking::withdraw_unbonded(Origin::signed(10))); -// -// // Can add again. -// assert_ok!(Staking::unbond(Origin::signed(10), 1)); -// assert_eq!(Staking::ledger(&10).unwrap().unlocking.len(), 2); -// }) -// } -// -// #[test] -// fn slot_stake_is_least_staked_validator_and_exposure_defines_maximum_punishment() { -// // Test that slot_stake is determined by the least staked validator -// // Test that slot_stake is the maximum punishment that can happen to a validator -// ExtBuilder::default() -// .nominate(false) -// .fair(false) -// .build() -// .execute_with(|| { -// // Confirm validator count is 2 -// assert_eq!(Staking::validator_count(), 2); -// // Confirm account 10 and 20 are validators -// assert!(>::exists(&11) && >::exists(&21)); -// -// assert_eq!(Staking::stakers(&11).total, 1000); -// assert_eq!(Staking::stakers(&21).total, 2000); -// -// // Give the man some money. -// let _ = Balances::make_free_balance_be(&10, 1000); -// let _ = Balances::make_free_balance_be(&20, 1000); -// -// // We confirm initialized slot_stake is this value -// assert_eq!(Staking::slot_stake(), Staking::stakers(&11).total); -// -// // Now lets lower account 20 stake -// >::insert( -// &21, -// Exposure { -// total: 69, -// own: 69, -// others: vec![], -// }, -// ); -// assert_eq!(Staking::stakers(&21).total, 69); -// >::insert( -// &20, -// StakingLedger { -// stash: 22, -// total: 69, -// active: 69, -// unlocking: vec![], -// }, -// ); -// -// // Compute total payout now for whole duration as other parameter won't change -// let total_payout_0 = current_total_payout_for_duration(3000); -// assert!(total_payout_0 > 100); // Test is meaningfull if reward something -// >::reward_by_ids(vec![(11, 1)]); -// >::reward_by_ids(vec![(21, 1)]); -// -// // New era --> rewards are paid --> stakes are changed -// start_era(1); -// -// // -- new balances + reward -// assert_eq!(Staking::stakers(&11).total, 1000 + total_payout_0 / 2); -// assert_eq!(Staking::stakers(&21).total, 69 + total_payout_0 / 2); -// -// let _11_balance = Balances::free_balance(&11); -// assert_eq!(_11_balance, 1000 + total_payout_0 / 2); -// -// // -- slot stake should also be updated. -// assert_eq!(Staking::slot_stake(), 69 + total_payout_0 / 2); -// -// check_exposure_all(); -// check_nominator_all(); -// }); -// } -// -// #[test] -// fn on_free_balance_zero_stash_removes_validator() { -// // Tests that validator storage items are cleaned up when stash is empty -// // Tests that storage items are untouched when controller is empty -// ExtBuilder::default().existential_deposit(10).build().execute_with(|| { -// // Check the balance of the validator account -// assert_eq!(Balances::free_balance(&10), 256); -// // Check the balance of the stash account -// assert_eq!(Balances::free_balance(&11), 256000); -// // Check these two accounts are bonded -// assert_eq!(Staking::bonded(&11), Some(10)); -// -// // Set some storage items which we expect to be cleaned up -// // Set payee information -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Stash)); -// -// // Check storage items that should be cleaned up -// assert!(>::exists(&10)); -// assert!(>::exists(&11)); -// assert!(>::exists(&11)); -// assert!(>::exists(&11)); -// -// // Reduce free_balance of controller to 0 -// let _ = Balances::slash(&10, u64::max_value()); -// -// // Check the balance of the stash account has not been touched -// assert_eq!(Balances::free_balance(&11), 256000); -// // Check these two accounts are still bonded -// assert_eq!(Staking::bonded(&11), Some(10)); -// -// // Check storage items have not changed -// assert!(>::exists(&10)); -// assert!(>::exists(&11)); -// assert!(>::exists(&11)); -// assert!(>::exists(&11)); -// -// // Reduce free_balance of stash to 0 -// let _ = Balances::slash(&11, u64::max_value()); -// // Check total balance of stash -// assert_eq!(Balances::total_balance(&11), 0); -// -// // Check storage items do not exist -// assert!(!>::exists(&10)); -// assert!(!>::exists(&11)); -// assert!(!>::exists(&11)); -// assert!(!>::exists(&11)); -// assert!(!>::exists(&11)); -// }); -// } -// -// #[test] -// fn on_free_balance_zero_stash_removes_nominator() { -// // Tests that nominator storage items are cleaned up when stash is empty -// // Tests that storage items are untouched when controller is empty -// ExtBuilder::default().existential_deposit(10).build().execute_with(|| { -// // Make 10 a nominator -// assert_ok!(Staking::nominate(Origin::signed(10), vec![20])); -// // Check that account 10 is a nominator -// assert!(>::exists(11)); -// // Check the balance of the nominator account -// assert_eq!(Balances::free_balance(&10), 256); -// // Check the balance of the stash account -// assert_eq!(Balances::free_balance(&11), 256000); -// -// // Set payee information -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Stash)); -// -// // Check storage items that should be cleaned up -// assert!(>::exists(&10)); -// assert!(>::exists(&11)); -// assert!(>::exists(&11)); -// assert!(>::exists(&11)); -// -// // Reduce free_balance of controller to 0 -// let _ = Balances::slash(&10, u64::max_value()); -// // Check total balance of account 10 -// assert_eq!(Balances::total_balance(&10), 0); -// -// // Check the balance of the stash account has not been touched -// assert_eq!(Balances::free_balance(&11), 256000); -// // Check these two accounts are still bonded -// assert_eq!(Staking::bonded(&11), Some(10)); -// -// // Check storage items have not changed -// assert!(>::exists(&10)); -// assert!(>::exists(&11)); -// assert!(>::exists(&11)); -// assert!(>::exists(&11)); -// -// // Reduce free_balance of stash to 0 -// let _ = Balances::slash(&11, u64::max_value()); -// // Check total balance of stash -// assert_eq!(Balances::total_balance(&11), 0); -// -// // Check storage items do not exist -// assert!(!>::exists(&10)); -// assert!(!>::exists(&11)); -// assert!(!>::exists(&11)); -// assert!(!>::exists(&11)); -// assert!(!>::exists(&11)); -// }); -// } -// -// #[test] -// fn switching_roles() { -// // Test that it should be possible to switch between roles (nominator, validator, idle) with minimal overhead. -// ExtBuilder::default().nominate(false).build().execute_with(|| { -// Timestamp::set_timestamp(1); // Initialize time. -// -// // Reset reward destination -// for i in &[10, 20] { -// assert_ok!(Staking::set_payee(Origin::signed(*i), RewardDestination::Controller)); -// } -// -// assert_eq_uvec!(validator_controllers(), vec![20, 10]); -// -// // put some money in account that we'll use. -// for i in 1..7 { -// let _ = Balances::deposit_creating(&i, 5000); -// } -// -// // add 2 nominators -// assert_ok!(Staking::bond(Origin::signed(1), 2, 2000, RewardDestination::Controller)); -// assert_ok!(Staking::nominate(Origin::signed(2), vec![11, 5])); -// -// assert_ok!(Staking::bond(Origin::signed(3), 4, 500, RewardDestination::Controller)); -// assert_ok!(Staking::nominate(Origin::signed(4), vec![21, 1])); -// -// // add a new validator candidate -// assert_ok!(Staking::bond(Origin::signed(5), 6, 1000, RewardDestination::Controller)); -// assert_ok!(Staking::validate(Origin::signed(6), ValidatorPrefs::default())); -// -// // new block -// start_session(1); -// -// // no change -// assert_eq_uvec!(validator_controllers(), vec![20, 10]); -// -// // new block -// start_session(2); -// -// // no change -// assert_eq_uvec!(validator_controllers(), vec![20, 10]); -// -// // new block --> ne era --> new validators -// start_session(3); -// -// // with current nominators 10 and 5 have the most stake -// assert_eq_uvec!(validator_controllers(), vec![6, 10]); -// -// // 2 decides to be a validator. Consequences: -// assert_ok!(Staking::validate(Origin::signed(2), ValidatorPrefs::default())); -// // new stakes: -// // 10: 1000 self vote -// // 20: 1000 self vote + 250 vote -// // 6 : 1000 self vote -// // 2 : 2000 self vote + 250 vote. -// // Winners: 20 and 2 -// -// start_session(4); -// assert_eq_uvec!(validator_controllers(), vec![6, 10]); -// -// start_session(5); -// assert_eq_uvec!(validator_controllers(), vec![6, 10]); -// -// // ne era -// start_session(6); -// assert_eq_uvec!(validator_controllers(), vec![2, 20]); -// -// check_exposure_all(); -// check_nominator_all(); -// }); -// } -// -// #[test] -// fn wrong_vote_is_null() { -// ExtBuilder::default() -// .nominate(false) -// .validator_pool(true) -// .build() -// .execute_with(|| { -// assert_eq_uvec!(validator_controllers(), vec![40, 30]); -// -// // put some money in account that we'll use. -// for i in 1..3 { -// let _ = Balances::deposit_creating(&i, 5000); -// } -// -// // add 1 nominators -// assert_ok!(Staking::bond(Origin::signed(1), 2, 2000, RewardDestination::default())); -// assert_ok!(Staking::nominate( -// Origin::signed(2), -// vec![ -// 11, 21, // good votes -// 1, 2, 15, 1000, 25 // crap votes. No effect. -// ] -// )); -// -// // new block -// start_era(1); -// -// assert_eq_uvec!(validator_controllers(), vec![20, 10]); -// }); -// } -// -// #[test] -// fn bond_with_no_staked_value() { -// // Behavior when someone bonds with no staked value. -// // Particularly when she votes and the candidate is elected. -// ExtBuilder::default() -// .validator_count(3) -// .existential_deposit(5) -// .nominate(false) -// .minimum_validator_count(1) -// .build() -// .execute_with(|| { -// // Can't bond with 1 -// assert_noop!( -// Staking::bond(Origin::signed(1), 2, 1, RewardDestination::Controller), -// Error::::InsufficientValue, -// ); -// // bonded with absolute minimum value possible. -// assert_ok!(Staking::bond(Origin::signed(1), 2, 5, RewardDestination::Controller)); -// assert_eq!(Balances::locks(&1)[0].amount, 5); -// -// // unbonding even 1 will cause all to be unbonded. -// assert_ok!(Staking::unbond(Origin::signed(2), 1)); -// assert_eq!( -// Staking::ledger(2), -// Some(StakingLedger { -// stash: 1, -// active: 0, -// total: 5, -// unlocking: vec![UnlockChunk { value: 5, era: 3 }] -// }) -// ); -// -// start_era(1); -// start_era(2); -// -// // not yet removed. -// assert_ok!(Staking::withdraw_unbonded(Origin::signed(2))); -// assert!(Staking::ledger(2).is_some()); -// assert_eq!(Balances::locks(&1)[0].amount, 5); -// -// start_era(3); -// -// // poof. Account 1 is removed from the staking system. -// assert_ok!(Staking::withdraw_unbonded(Origin::signed(2))); -// assert!(Staking::ledger(2).is_none()); -// assert_eq!(Balances::locks(&1).len(), 0); -// }); -// } -// -// #[test] -// fn bond_with_little_staked_value_bounded_by_slot_stake() { -// // Behavior when someone bonds with little staked value. -// // Particularly when she votes and the candidate is elected. -// ExtBuilder::default() -// .validator_count(3) -// .nominate(false) -// .minimum_validator_count(1) -// .build() -// .execute_with(|| { -// // setup -// assert_ok!(Staking::chill(Origin::signed(30))); -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); -// let init_balance_2 = Balances::free_balance(&2); -// let init_balance_10 = Balances::free_balance(&10); -// -// // Stingy validator. -// assert_ok!(Staking::bond(Origin::signed(1), 2, 1, RewardDestination::Controller)); -// assert_ok!(Staking::validate(Origin::signed(2), ValidatorPrefs::default())); -// -// let total_payout_0 = current_total_payout_for_duration(3000); -// assert!(total_payout_0 > 100); // Test is meaningfull if reward something -// reward_all_elected(); -// start_era(1); -// -// // 2 is elected. -// // and fucks up the slot stake. -// assert_eq_uvec!(validator_controllers(), vec![20, 10, 2]); -// assert_eq!(Staking::slot_stake(), 1); -// -// // Old ones are rewarded. -// assert_eq!(Balances::free_balance(&10), init_balance_10 + total_payout_0 / 3); -// // no rewards paid to 2. This was initial election. -// assert_eq!(Balances::free_balance(&2), init_balance_2); -// -// let total_payout_1 = current_total_payout_for_duration(3000); -// assert!(total_payout_1 > 100); // Test is meaningfull if reward something -// reward_all_elected(); -// start_era(2); -// -// assert_eq_uvec!(validator_controllers(), vec![20, 10, 2]); -// assert_eq!(Staking::slot_stake(), 1); -// -// assert_eq!(Balances::free_balance(&2), init_balance_2 + total_payout_1 / 3); -// assert_eq!( -// Balances::free_balance(&10), -// init_balance_10 + total_payout_0 / 3 + total_payout_1 / 3, -// ); -// check_exposure_all(); -// check_nominator_all(); -// }); -// } -// -// #[cfg(feature = "equalize")] -// #[test] -// fn phragmen_linear_worse_case_equalize() { -// ExtBuilder::default() -// .nominate(false) -// .validator_pool(true) -// .fair(true) -// .build() -// .execute_with(|| { -// bond_validator(50, 1000); -// bond_validator(60, 1000); -// bond_validator(70, 1000); -// -// bond_nominator(2, 2000, vec![11]); -// bond_nominator(4, 1000, vec![11, 21]); -// bond_nominator(6, 1000, vec![21, 31]); -// bond_nominator(8, 1000, vec![31, 41]); -// bond_nominator(110, 1000, vec![41, 51]); -// bond_nominator(120, 1000, vec![51, 61]); -// bond_nominator(130, 1000, vec![61, 71]); -// -// for i in &[10, 20, 30, 40, 50, 60, 70] { -// assert_ok!(Staking::set_payee(Origin::signed(*i), RewardDestination::Controller)); -// } -// -// assert_eq_uvec!(validator_controllers(), vec![40, 30]); -// assert_ok!(Staking::set_validator_count(Origin::ROOT, 7)); -// -// start_era(1); -// -// assert_eq_uvec!(validator_controllers(), vec![10, 60, 40, 20, 50, 30, 70]); -// -// assert_eq_error_rate!(Staking::stakers(11).total, 3000, 2); -// assert_eq_error_rate!(Staking::stakers(21).total, 2255, 2); -// assert_eq_error_rate!(Staking::stakers(31).total, 2255, 2); -// assert_eq_error_rate!(Staking::stakers(41).total, 1925, 2); -// assert_eq_error_rate!(Staking::stakers(51).total, 1870, 2); -// assert_eq_error_rate!(Staking::stakers(61).total, 1890, 2); -// assert_eq_error_rate!(Staking::stakers(71).total, 1800, 2); -// -// check_exposure_all(); -// check_nominator_all(); -// }) -// } -// -// #[test] -// fn new_era_elects_correct_number_of_validators() { -// ExtBuilder::default() -// .nominate(true) -// .validator_pool(true) -// .fair(true) -// .validator_count(1) -// .build() -// .execute_with(|| { -// assert_eq!(Staking::validator_count(), 1); -// assert_eq!(validator_controllers().len(), 1); -// -// System::set_block_number(1); -// Session::on_initialize(System::block_number()); -// -// assert_eq!(validator_controllers().len(), 1); -// check_exposure_all(); -// check_nominator_all(); -// }) -// } +/// gen_paired_account!(a(1), b(2), m(12)); +/// will create stash `a` and controller `b` +/// `a` has 100 Ring and 100 Kton +/// promise for `m` month with 50 Ring and 50 Kton +/// +/// `m` can be ignore, this won't create variable `m` +/// ```rust +/// gen_parired_account!(a(1), b(2), 12); +/// ``` +/// +/// `m(12)` can be ignore, and it won't perform `bond` action +/// ```rust +/// gen_paired_account!(a(1), b(2)); +/// ``` +macro_rules! gen_paired_account { + ($stash:ident($stash_id:expr), $controller:ident($controller_id:expr), $promise_month:ident($how_long:expr)) => { + #[allow(non_snake_case, unused)] + let $stash = $stash_id; + let _ = Ring::deposit_creating(&$stash, 100 * COIN); + let _ = Kton::deposit_creating(&$stash, 100 * COIN); + #[allow(non_snake_case, unused)] + let $controller = $controller_id; + let _ = Ring::deposit_creating(&$controller, COIN); + #[allow(non_snake_case, unused)] + let $promise_month = $how_long; + assert_ok!(Staking::bond( + Origin::signed($stash), + $controller, + StakingBalance::RingBalance(50 * COIN), + RewardDestination::Stash, + $how_long, + )); + assert_ok!(Staking::bond_extra( + Origin::signed($stash), + StakingBalance::KtonBalance(50 * COIN), + $how_long + )); + }; + ($stash:ident($stash_id:expr), $controller:ident($controller_id:expr), $how_long:expr) => { + #[allow(non_snake_case, unused)] + let $stash = $stash_id; + let _ = Ring::deposit_creating(&$stash, 100 * COIN); + let _ = Kton::deposit_creating(&$stash, 100 * COIN); + #[allow(non_snake_case, unused)] + let $controller = $controller_id; + let _ = Ring::deposit_creating(&$controller, COIN); + assert_ok!(Staking::bond( + Origin::signed($stash), + $controller, + StakingBalance::RingBalance(50 * COIN), + RewardDestination::Stash, + $how_long, + )); + assert_ok!(Staking::bond_extra( + Origin::signed($stash), + StakingBalance::KtonBalance(50 * COIN), + $how_long, + )); + }; + ($stash:ident($stash_id:expr), $controller:ident($controller_id:expr)) => { + #[allow(non_snake_case, unused)] + let $stash = $stash_id; + let _ = Ring::deposit_creating(&$stash, 100 * COIN); + let _ = Kton::deposit_creating(&$stash, 100 * COIN); + #[allow(non_snake_case, unused)] + let $controller = $controller_id; + let _ = Ring::deposit_creating(&$controller, COIN); + }; +} #[test] -fn phragmen_should_not_overflow_validators() { - ExtBuilder::default().nominate(false).build().execute_with(|| { - let _ = Staking::chill(Origin::signed(10)); - let _ = Staking::chill(Origin::signed(20)); +fn force_unstake_works() { + // Verifies initial conditions of mock + ExtBuilder::default().build().execute_with(|| { + // Account 11 is stashed and locked, and account 10 is the controller + assert_eq!(Staking::bonded(&11), Some(10)); + // Cant transfer + assert_noop!( + Ring::transfer(Origin::signed(11), 1, 10), + DispatchError::Module { + index: 0, + error: 1, + message: Some("LiquidityRestrictions") + } + ); + // Force unstake requires root. + assert_noop!(Staking::force_unstake(Origin::signed(11), 11), DispatchError::BadOrigin); + // We now force them to unstake + assert_ok!(Staking::force_unstake(Origin::ROOT, 11)); + // No longer bonded. + assert_eq!(Staking::bonded(&11), None); + // Transfer works. + assert_ok!(Ring::transfer(Origin::signed(11), 1, 10)); + }); +} + +#[test] +fn basic_setup_works() { + // Verifies initial conditions of mock + ExtBuilder::default().build().execute_with(|| { + // Account 11 is stashed and locked, and account 10 is the controller + assert_eq!(Staking::bonded(&11), Some(10)); + // Account 21 is stashed and locked, and account 20 is the controller + assert_eq!(Staking::bonded(&21), Some(20)); + // Account 1 is not a stashed + assert_eq!(Staking::bonded(&1), None); - bond_validator(2, StakingBalance::RingBalance(CAP - 1)); - bond_validator(4, StakingBalance::KtonBalance(CAP - 1)); + // Account 10 controls the stash from account 11, which is 100 * balance_factor units + assert_eq!( + Staking::ledger(&10), + Some(StakingLedger { + stash: 11, + active_ring: 1000, + active_deposit_ring: 0, + active_kton: 0, + deposit_items: Vec::new(), + ring_staking_lock: StakingLock { + staking_amount: 1000, + unbondings: Vec::new() + }, + kton_staking_lock: StakingLock { + staking_amount: 0, + unbondings: Vec::new() + } + }) + ); + // Account 20 controls the stash from account 21, which is 200 * balance_factor units + assert_eq!( + Staking::ledger(&20), + Some(StakingLedger { + stash: 21, + active_ring: 1000, + active_deposit_ring: 0, + active_kton: 0, + deposit_items: Vec::new(), + ring_staking_lock: StakingLock { + staking_amount: 1000, + unbondings: Vec::new() + }, + kton_staking_lock: StakingLock { + staking_amount: 0, + unbondings: Vec::new() + } + }) + ); + // Account 1 does not control any stash + assert_eq!(Staking::ledger(&1), None); - bond_nominator(6, StakingBalance::RingBalance(1), vec![3, 5]); - bond_nominator(8, StakingBalance::KtonBalance(1), vec![3, 5]); + // ValidatorPrefs are default + assert_eq!( + >::enumerate().collect::>(), + vec![ + (31, ValidatorPrefs::default()), + (21, ValidatorPrefs::default()), + (11, ValidatorPrefs::default()) + ] + ); - start_era(1); + assert_eq!( + Staking::ledger(100), + Some(StakingLedger { + stash: 101, + active_ring: 500, + active_deposit_ring: 0, + active_kton: 0, + deposit_items: Vec::new(), + ring_staking_lock: StakingLock { + staking_amount: 500, + unbondings: Vec::new() + }, + kton_staking_lock: StakingLock { + staking_amount: 0, + unbondings: Vec::new() + } + }) + ); + assert_eq!(Staking::nominators(101).unwrap().targets, vec![11, 21]); - assert_eq_uvec!(validator_controllers(), vec![4, 2]); + if cfg!(feature = "equalize") { - // This test will fail this. Will saturate. - // check_exposure_all(); - assert_eq!(Staking::stakers(3).total_power, TOTAL_POWER / 2); - assert_eq!(Staking::stakers(5).total_power, TOTAL_POWER / 2); + // TODO: fix equalize + // assert_eq!(Staking::stakers(11).own_ring_balance, 1000); + // assert_eq!(Staking::stakers(11).total_ring_balance, 1250); + // assert_eq!(Staking::stakers(11).others[0].who, 101); + // assert_eq!(Staking::stakers(11).others[0].ring_balance, 250); + // assert_eq!(Staking::stakers(21).own_ring_balance, 1000); + // assert_eq!(Staking::stakers(21).total_ring_balance, 1250); + // assert_eq!(Staking::stakers(21).others[0].who, 101); + // assert_eq!(Staking::stakers(21).others[0].ring_balance, 250); + // assert_eq!(Staking::slot_stake(), 1250); + } else { + assert_eq!(Staking::stakers(11).own_ring_balance, 1000); + assert_eq!(Staking::stakers(11).total_ring_balance, 1125); + assert_eq!(Staking::stakers(11).others[0].who, 101); + assert_eq!(Staking::stakers(11).others[0].ring_balance, 125); + assert_eq!(Staking::stakers(21).own_ring_balance, 1000); + assert_eq!(Staking::stakers(21).total_ring_balance, 1375); + assert_eq!(Staking::stakers(21).others[0].who, 101); + assert_eq!(Staking::stakers(21).others[0].ring_balance, 375); + assert_eq!(Staking::slot_stake(), 1125); + } + + // The number of validators required. + assert_eq!(Staking::validator_count(), 2); + + // Initial Era and session + assert_eq!(Staking::current_era(), 0); + + // Account 10 has `balance_factor` free balance + assert_eq!(Ring::free_balance(&10), 1); + assert_eq!(Ring::free_balance(&10), 1); + + // New era is not being forced + assert_eq!(Staking::force_era(), Forcing::NotForcing); + + // All exposures must be correct. + check_exposure_all(); + check_nominator_all(); + }); +} + +#[test] +fn change_controller_works() { + ExtBuilder::default().build().execute_with(|| { + assert_eq!(Staking::bonded(&11), Some(10)); + + assert!(>::enumerate() + .map(|(c, _)| c) + .collect::>() + .contains(&11)); + // 10 can control 11 who is initially a validator. + assert_ok!(Staking::chill(Origin::signed(10))); + assert!(!>::enumerate() + .map(|(c, _)| c) + .collect::>() + .contains(&11)); + + assert_ok!(Staking::set_controller(Origin::signed(11), 5)); + + start_era(1); + + assert_noop!( + Staking::validate(Origin::signed(10), ValidatorPrefs::default()), + Error::::NotController, + ); + assert_ok!(Staking::validate(Origin::signed(5), ValidatorPrefs::default())); }) } -// #[test] -// fn phragmen_should_not_overflow_nominators() { -// ExtBuilder::default().nominate(false).build().execute_with(|| { -// let _ = Staking::chill(Origin::signed(10)); -// let _ = Staking::chill(Origin::signed(20)); -// -// bond_validator(2, u64::max_value() / 2); -// bond_validator(4, u64::max_value() / 2); -// -// bond_nominator(6, u64::max_value(), vec![3, 5]); -// bond_nominator(8, u64::max_value(), vec![3, 5]); -// -// start_era(1); -// -// assert_eq_uvec!(validator_controllers(), vec![4, 2]); -// -// // Saturate. -// assert_eq!(Staking::stakers(3).total, u64::max_value()); -// assert_eq!(Staking::stakers(5).total, u64::max_value()); -// }) -// } -// -// #[test] -// fn phragmen_should_not_overflow_ultimate() { -// ExtBuilder::default().nominate(false).build().execute_with(|| { -// bond_validator(2, u64::max_value()); -// bond_validator(4, u64::max_value()); -// -// bond_nominator(6, u64::max_value(), vec![3, 5]); -// bond_nominator(8, u64::max_value(), vec![3, 5]); -// -// start_era(1); -// -// assert_eq_uvec!(validator_controllers(), vec![4, 2]); -// -// // Saturate. -// assert_eq!(Staking::stakers(3).total, u64::max_value()); -// assert_eq!(Staking::stakers(5).total, u64::max_value()); -// }) -// } -// -// #[test] -// fn reward_validator_slashing_validator_doesnt_overflow() { -// ExtBuilder::default().build().execute_with(|| { -// let stake = u32::max_value() as u64 * 2; -// let reward_slash = u32::max_value() as u64 * 2; -// -// // Assert multiplication overflows in balance arithmetic. -// assert!(stake.checked_mul(reward_slash).is_none()); -// -// // Set staker -// let _ = Balances::make_free_balance_be(&11, stake); -// >::insert( -// &11, -// Exposure { -// total: stake, -// own: stake, -// others: vec![], -// }, -// ); -// -// // Check reward -// let _ = Staking::reward_validator(&11, reward_slash); -// assert_eq!(Balances::total_balance(&11), stake * 2); -// -// // Set staker -// let _ = Balances::make_free_balance_be(&11, stake); -// let _ = Balances::make_free_balance_be(&2, stake); -// -// // only slashes out of bonded stake are applied. without this line, -// // it is 0. -// Staking::bond(Origin::signed(2), 20000, stake - 1, RewardDestination::default()).unwrap(); -// >::insert( -// &11, -// Exposure { -// total: stake, -// own: 1, -// others: vec![IndividualExposure { -// who: 2, -// value: stake - 1, -// }], -// }, -// ); -// -// // Check slashing -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(100)], -// ); -// -// assert_eq!(Balances::total_balance(&11), stake - 1); -// assert_eq!(Balances::total_balance(&2), 1); -// }) -// } -// -// #[test] -// fn reward_from_authorship_event_handler_works() { -// ExtBuilder::default().build().execute_with(|| { -// use pallet_authorship::EventHandler; -// -// assert_eq!(>::author(), 11); -// -// >::note_author(11); -// >::note_uncle(21, 1); -// // An uncle author that is not currently elected doesn't get rewards, -// // but the block producer does get reward for referencing it. -// >::note_uncle(31, 1); -// // Rewarding the same two times works. -// >::note_uncle(11, 1); -// -// // Not mandatory but must be coherent with rewards -// assert_eq!(>::get(), vec![21, 11]); -// -// // 21 is rewarded as an uncle producer -// // 11 is rewarded as a block producer and uncle referencer and uncle producer -// assert_eq!(CurrentEraPointsEarned::get().individual, vec![1, 20 + 2 * 3 + 1]); -// assert_eq!(CurrentEraPointsEarned::get().total, 28); -// }) -// } -// -// #[test] -// fn add_reward_points_fns_works() { -// ExtBuilder::default().build().execute_with(|| { -// let validators = >::current_elected(); -// // Not mandatory but must be coherent with rewards -// assert_eq!(validators, vec![21, 11]); -// -// >::reward_by_indices(vec![(0, 1), (1, 1), (2, 1), (1, 1)]); -// -// >::reward_by_ids(vec![(21, 1), (11, 1), (31, 1), (11, 1)]); -// -// assert_eq!(CurrentEraPointsEarned::get().individual, vec![2, 4]); -// assert_eq!(CurrentEraPointsEarned::get().total, 6); -// }) -// } -// -// #[test] -// fn unbonded_balance_is_not_slashable() { -// ExtBuilder::default().build().execute_with(|| { -// // total amount staked is slashable. -// assert_eq!(Staking::slashable_balance_of(&11), 1000); -// -// assert_ok!(Staking::unbond(Origin::signed(10), 800)); -// -// // only the active portion. -// assert_eq!(Staking::slashable_balance_of(&11), 200); -// }) -// } -// -// #[test] -// fn era_is_always_same_length() { -// // This ensures that the sessions is always of the same length if there is no forcing no -// // session changes. -// ExtBuilder::default().build().execute_with(|| { -// start_era(1); -// assert_eq!(Staking::current_era_start_session_index(), SessionsPerEra::get()); -// -// start_era(2); -// assert_eq!(Staking::current_era_start_session_index(), SessionsPerEra::get() * 2); -// -// let session = Session::current_index(); -// ForceEra::put(Forcing::ForceNew); -// advance_session(); -// assert_eq!(Staking::current_era(), 3); -// assert_eq!(Staking::current_era_start_session_index(), session + 1); -// -// start_era(4); -// assert_eq!( -// Staking::current_era_start_session_index(), -// session + SessionsPerEra::get() + 1 -// ); -// }); -// } -// -// #[test] -// fn offence_forces_new_era() { -// ExtBuilder::default().build().execute_with(|| { -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(5)], -// ); -// -// assert_eq!(Staking::force_era(), Forcing::ForceNew); -// }); -// } -// -// #[test] -// fn offence_ensures_new_era_without_clobbering() { -// ExtBuilder::default().build().execute_with(|| { -// assert_ok!(Staking::force_new_era_always(Origin::ROOT)); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(5)], -// ); -// -// assert_eq!(Staking::force_era(), Forcing::ForceAlways); -// }); -// } -// -// #[test] -// fn offence_deselects_validator_when_slash_is_zero() { -// ExtBuilder::default().build().execute_with(|| { -// assert!(>::exists(11)); -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(0)], -// ); -// assert_eq!(Staking::force_era(), Forcing::ForceNew); -// assert!(!>::exists(11)); -// }); -// } -// -// #[test] -// fn slashing_performed_according_exposure() { -// // This test checks that slashing is performed according the exposure (or more precisely, -// // historical exposure), not the current balance. -// ExtBuilder::default().build().execute_with(|| { -// assert_eq!(Staking::stakers(&11).own, 1000); -// -// // Handle an offence with a historical exposure. -// on_offence_now( -// &[OffenceDetails { -// offender: ( -// 11, -// Exposure { -// total: 500, -// own: 500, -// others: vec![], -// }, -// ), -// reporters: vec![], -// }], -// &[Perbill::from_percent(50)], -// ); -// -// // The stash account should be slashed for 250 (50% of 500). -// assert_eq!(Balances::free_balance(&11), 1000 - 250); -// }); -// } -// -// #[test] -// fn slash_in_old_span_does_not_deselect() { -// ExtBuilder::default().build().execute_with(|| { -// start_era(1); -// -// assert!(>::exists(11)); -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(0)], -// ); -// assert_eq!(Staking::force_era(), Forcing::ForceNew); -// assert!(!>::exists(11)); -// -// start_era(2); -// -// Staking::validate(Origin::signed(10), Default::default()).unwrap(); -// assert_eq!(Staking::force_era(), Forcing::NotForcing); -// assert!(>::exists(11)); -// -// start_era(3); -// -// // this staker is in a new slashing span now, having re-registered after -// // their prior slash. -// -// on_offence_in_era( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(0)], -// 1, -// ); -// -// // not for zero-slash. -// assert_eq!(Staking::force_era(), Forcing::NotForcing); -// assert!(>::exists(11)); -// -// on_offence_in_era( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(100)], -// 1, -// ); -// -// // or non-zero. -// assert_eq!(Staking::force_era(), Forcing::NotForcing); -// assert!(>::exists(11)); -// assert_ledger_consistent(11); -// }); -// } -// -// #[test] -// fn reporters_receive_their_slice() { -// // This test verifies that the reporters of the offence receive their slice from the slashed -// // amount. -// ExtBuilder::default().build().execute_with(|| { -// // The reporters' reward is calculated from the total exposure. -// #[cfg(feature = "equalize")] -// let initial_balance = 1250; -// #[cfg(not(feature = "equalize"))] -// let initial_balance = 1125; -// -// assert_eq!(Staking::stakers(&11).total, initial_balance); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![1, 2], -// }], -// &[Perbill::from_percent(50)], -// ); -// -// // F1 * (reward_proportion * slash - 0) -// // 50% * (10% * initial_balance / 2) -// let reward = (initial_balance / 20) / 2; -// let reward_each = reward / 2; // split into two pieces. -// assert_eq!(Balances::free_balance(&1), 10 + reward_each); -// assert_eq!(Balances::free_balance(&2), 20 + reward_each); -// assert_ledger_consistent(11); -// }); -// } -// -// #[test] -// fn subsequent_reports_in_same_span_pay_out_less() { -// // This test verifies that the reporters of the offence receive their slice from the slashed -// // amount. -// ExtBuilder::default().build().execute_with(|| { -// // The reporters' reward is calculated from the total exposure. -// #[cfg(feature = "equalize")] -// let initial_balance = 1250; -// #[cfg(not(feature = "equalize"))] -// let initial_balance = 1125; -// -// assert_eq!(Staking::stakers(&11).total, initial_balance); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![1], -// }], -// &[Perbill::from_percent(20)], -// ); -// -// // F1 * (reward_proportion * slash - 0) -// // 50% * (10% * initial_balance * 20%) -// let reward = (initial_balance / 5) / 20; -// assert_eq!(Balances::free_balance(&1), 10 + reward); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![1], -// }], -// &[Perbill::from_percent(50)], -// ); -// -// let prior_payout = reward; -// -// // F1 * (reward_proportion * slash - prior_payout) -// // 50% * (10% * (initial_balance / 2) - prior_payout) -// let reward = ((initial_balance / 20) - prior_payout) / 2; -// assert_eq!(Balances::free_balance(&1), 10 + prior_payout + reward); -// assert_ledger_consistent(11); -// }); -// } -// -// #[test] -// fn invulnerables_are_not_slashed() { -// // For invulnerable validators no slashing is performed. -// ExtBuilder::default().invulnerables(vec![11]).build().execute_with(|| { -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_eq!(Balances::free_balance(&21), 2000); -// -// let exposure = Staking::stakers(&21); -// let initial_balance = Staking::slashable_balance_of(&21); -// -// let nominator_balances: Vec<_> = exposure.others.iter().map(|o| Balances::free_balance(&o.who)).collect(); -// -// on_offence_now( -// &[ -// OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }, -// OffenceDetails { -// offender: (21, Staking::stakers(&21)), -// reporters: vec![], -// }, -// ], -// &[Perbill::from_percent(50), Perbill::from_percent(20)], -// ); -// -// // The validator 11 hasn't been slashed, but 21 has been. -// assert_eq!(Balances::free_balance(&11), 1000); -// // 2000 - (0.2 * initial_balance) -// assert_eq!(Balances::free_balance(&21), 2000 - (2 * initial_balance / 10)); -// -// // ensure that nominators were slashed as well. -// for (initial_balance, other) in nominator_balances.into_iter().zip(exposure.others) { -// assert_eq!( -// Balances::free_balance(&other.who), -// initial_balance - (2 * other.value / 10), -// ); -// } -// assert_ledger_consistent(11); -// assert_ledger_consistent(21); -// }); -// } -// -// #[test] -// fn dont_slash_if_fraction_is_zero() { -// // Don't slash if the fraction is zero. -// ExtBuilder::default().build().execute_with(|| { -// assert_eq!(Balances::free_balance(&11), 1000); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(0)], -// ); -// -// // The validator hasn't been slashed. The new era is not forced. -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_ledger_consistent(11); -// }); -// } -// -// #[test] -// fn only_slash_for_max_in_era() { -// ExtBuilder::default().build().execute_with(|| { -// assert_eq!(Balances::free_balance(&11), 1000); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(50)], -// ); -// -// // The validator has been slashed and has been force-chilled. -// assert_eq!(Balances::free_balance(&11), 500); -// assert_eq!(Staking::force_era(), Forcing::ForceNew); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(25)], -// ); -// -// // The validator has not been slashed additionally. -// assert_eq!(Balances::free_balance(&11), 500); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(60)], -// ); -// -// // The validator got slashed 10% more. -// assert_eq!(Balances::free_balance(&11), 400); -// assert_ledger_consistent(11); -// }) -// } -// -// #[test] -// fn garbage_collection_after_slashing() { -// ExtBuilder::default().existential_deposit(1).build().execute_with(|| { -// assert_eq!(Balances::free_balance(&11), 256_000); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(10)], -// ); -// -// assert_eq!(Balances::free_balance(&11), 256_000 - 25_600); -// assert!(::SlashingSpans::get(&11).is_some()); -// assert_eq!( -// ::SpanSlash::get(&(11, 0)).amount_slashed(), -// &25_600 -// ); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(100)], -// ); -// -// // validator and nominator slash in era are garbage-collected by era change, -// // so we don't test those here. -// -// assert_eq!(Balances::free_balance(&11), 0); -// assert!(::SlashingSpans::get(&11).is_none()); -// assert_eq!(::SpanSlash::get(&(11, 0)).amount_slashed(), &0); -// }) -// } -// -// #[test] -// fn garbage_collection_on_window_pruning() { -// ExtBuilder::default().build().execute_with(|| { -// start_era(1); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// -// let exposure = Staking::stakers(&11); -// assert_eq!(Balances::free_balance(&101), 2000); -// let nominated_value = exposure.others.iter().find(|o| o.who == 101).unwrap().value; -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(10)], -// ); -// -// let now = Staking::current_era(); -// -// assert_eq!(Balances::free_balance(&11), 900); -// assert_eq!(Balances::free_balance(&101), 2000 - (nominated_value / 10)); -// -// assert!(::ValidatorSlashInEra::get(&now, &11).is_some()); -// assert!(::NominatorSlashInEra::get(&now, &101).is_some()); -// -// // + 1 because we have to exit the bonding window. -// for era in (0..(BondingDuration::get() + 1)).map(|offset| offset + now + 1) { -// assert!(::ValidatorSlashInEra::get(&now, &11).is_some()); -// assert!(::NominatorSlashInEra::get(&now, &101).is_some()); -// -// start_era(era); -// } -// -// assert!(::ValidatorSlashInEra::get(&now, &11).is_none()); -// assert!(::NominatorSlashInEra::get(&now, &101).is_none()); -// }) -// } -// -// #[test] -// fn slashing_nominators_by_span_max() { -// ExtBuilder::default().build().execute_with(|| { -// start_era(1); -// start_era(2); -// start_era(3); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_eq!(Balances::free_balance(&21), 2000); -// assert_eq!(Balances::free_balance(&101), 2000); -// assert_eq!(Staking::slashable_balance_of(&21), 1000); -// -// let exposure_11 = Staking::stakers(&11); -// let exposure_21 = Staking::stakers(&21); -// assert_eq!(Balances::free_balance(&101), 2000); -// let nominated_value_11 = exposure_11.others.iter().find(|o| o.who == 101).unwrap().value; -// let nominated_value_21 = exposure_21.others.iter().find(|o| o.who == 101).unwrap().value; -// -// on_offence_in_era( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(10)], -// 2, -// ); -// -// assert_eq!(Balances::free_balance(&11), 900); -// -// let slash_1_amount = Perbill::from_percent(10) * nominated_value_11; -// assert_eq!(Balances::free_balance(&101), 2000 - slash_1_amount); -// -// let expected_spans = vec![ -// slashing::SlashingSpan { -// index: 1, -// start: 4, -// length: None, -// }, -// slashing::SlashingSpan { -// index: 0, -// start: 0, -// length: Some(4), -// }, -// ]; -// -// let get_span = |account| ::SlashingSpans::get(&account).unwrap(); -// -// assert_eq!(get_span(11).iter().collect::>(), expected_spans,); -// -// assert_eq!(get_span(101).iter().collect::>(), expected_spans,); -// -// // second slash: higher era, higher value, same span. -// on_offence_in_era( -// &[OffenceDetails { -// offender: (21, Staking::stakers(&21)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(30)], -// 3, -// ); -// -// // 11 was not further slashed, but 21 and 101 were. -// assert_eq!(Balances::free_balance(&11), 900); -// assert_eq!(Balances::free_balance(&21), 1700); -// -// let slash_2_amount = Perbill::from_percent(30) * nominated_value_21; -// assert!(slash_2_amount > slash_1_amount); -// -// // only the maximum slash in a single span is taken. -// assert_eq!(Balances::free_balance(&101), 2000 - slash_2_amount); -// -// // third slash: in same era and on same validator as first, higher -// // in-era value, but lower slash value than slash 2. -// on_offence_in_era( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(20)], -// 2, -// ); -// -// // 11 was further slashed, but 21 and 101 were not. -// assert_eq!(Balances::free_balance(&11), 800); -// assert_eq!(Balances::free_balance(&21), 1700); -// -// let slash_3_amount = Perbill::from_percent(20) * nominated_value_21; -// assert!(slash_3_amount < slash_2_amount); -// assert!(slash_3_amount > slash_1_amount); -// -// // only the maximum slash in a single span is taken. -// assert_eq!(Balances::free_balance(&101), 2000 - slash_2_amount); -// }); -// } -// -// #[test] -// fn slashes_are_summed_across_spans() { -// ExtBuilder::default().build().execute_with(|| { -// start_era(1); -// start_era(2); -// start_era(3); -// -// assert_eq!(Balances::free_balance(&21), 2000); -// assert_eq!(Staking::slashable_balance_of(&21), 1000); -// -// let get_span = |account| ::SlashingSpans::get(&account).unwrap(); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (21, Staking::stakers(&21)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(10)], -// ); -// -// let expected_spans = vec![ -// slashing::SlashingSpan { -// index: 1, -// start: 4, -// length: None, -// }, -// slashing::SlashingSpan { -// index: 0, -// start: 0, -// length: Some(4), -// }, -// ]; -// -// assert_eq!(get_span(21).iter().collect::>(), expected_spans); -// assert_eq!(Balances::free_balance(&21), 1900); -// -// // 21 has been force-chilled. re-signal intent to validate. -// Staking::validate(Origin::signed(20), Default::default()).unwrap(); -// -// start_era(4); -// -// assert_eq!(Staking::slashable_balance_of(&21), 900); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (21, Staking::stakers(&21)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(10)], -// ); -// -// let expected_spans = vec![ -// slashing::SlashingSpan { -// index: 2, -// start: 5, -// length: None, -// }, -// slashing::SlashingSpan { -// index: 1, -// start: 4, -// length: Some(1), -// }, -// slashing::SlashingSpan { -// index: 0, -// start: 0, -// length: Some(4), -// }, -// ]; -// -// assert_eq!(get_span(21).iter().collect::>(), expected_spans); -// assert_eq!(Balances::free_balance(&21), 1810); -// }); -// } -// -// #[test] -// fn deferred_slashes_are_deferred() { -// ExtBuilder::default().slash_defer_duration(2).build().execute_with(|| { -// start_era(1); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// -// let exposure = Staking::stakers(&11); -// assert_eq!(Balances::free_balance(&101), 2000); -// let nominated_value = exposure.others.iter().find(|o| o.who == 101).unwrap().value; -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, Staking::stakers(&11)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(10)], -// ); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_eq!(Balances::free_balance(&101), 2000); -// -// start_era(2); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_eq!(Balances::free_balance(&101), 2000); -// -// start_era(3); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_eq!(Balances::free_balance(&101), 2000); -// -// // at the start of era 4, slashes from era 1 are processed, -// // after being deferred for at least 2 full eras. -// start_era(4); -// -// assert_eq!(Balances::free_balance(&11), 900); -// assert_eq!(Balances::free_balance(&101), 2000 - (nominated_value / 10)); -// }) -// } -// -// #[test] -// fn remove_deferred() { -// ExtBuilder::default().slash_defer_duration(2).build().execute_with(|| { -// start_era(1); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// -// let exposure = Staking::stakers(&11); -// assert_eq!(Balances::free_balance(&101), 2000); -// let nominated_value = exposure.others.iter().find(|o| o.who == 101).unwrap().value; -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, exposure.clone()), -// reporters: vec![], -// }], -// &[Perbill::from_percent(10)], -// ); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_eq!(Balances::free_balance(&101), 2000); -// -// start_era(2); -// -// on_offence_in_era( -// &[OffenceDetails { -// offender: (11, exposure.clone()), -// reporters: vec![], -// }], -// &[Perbill::from_percent(15)], -// 1, -// ); -// -// Staking::cancel_deferred_slash(Origin::ROOT, 1, vec![0]).unwrap(); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_eq!(Balances::free_balance(&101), 2000); -// -// start_era(3); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_eq!(Balances::free_balance(&101), 2000); -// -// // at the start of era 4, slashes from era 1 are processed, -// // after being deferred for at least 2 full eras. -// start_era(4); -// -// // the first slash for 10% was cancelled, so no effect. -// assert_eq!(Balances::free_balance(&11), 1000); -// assert_eq!(Balances::free_balance(&101), 2000); -// -// start_era(5); -// -// let slash_10 = Perbill::from_percent(10); -// let slash_15 = Perbill::from_percent(15); -// let initial_slash = slash_10 * nominated_value; -// -// let total_slash = slash_15 * nominated_value; -// let actual_slash = total_slash - initial_slash; -// -// // 5% slash (15 - 10) processed now. -// assert_eq!(Balances::free_balance(&11), 950); -// assert_eq!(Balances::free_balance(&101), 2000 - actual_slash); -// }) -// } -// -// #[test] -// fn remove_multi_deferred() { -// ExtBuilder::default().slash_defer_duration(2).build().execute_with(|| { -// start_era(1); -// -// assert_eq!(Balances::free_balance(&11), 1000); -// -// let exposure = Staking::stakers(&11); -// assert_eq!(Balances::free_balance(&101), 2000); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, exposure.clone()), -// reporters: vec![], -// }], -// &[Perbill::from_percent(10)], -// ); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (21, Staking::stakers(&21)), -// reporters: vec![], -// }], -// &[Perbill::from_percent(10)], -// ); -// -// on_offence_now( -// &[OffenceDetails { -// offender: (11, exposure.clone()), -// reporters: vec![], -// }], -// &[Perbill::from_percent(25)], -// ); -// -// assert_eq!(::UnappliedSlashes::get(&1).len(), 3); -// Staking::cancel_deferred_slash(Origin::ROOT, 1, vec![0, 2]).unwrap(); -// -// let slashes = ::UnappliedSlashes::get(&1); -// assert_eq!(slashes.len(), 1); -// assert_eq!(slashes[0].validator, 21); -// }) -// } -// -// #[test] -// fn version_initialized() { -// ExtBuilder::default().build().execute_with(|| { -// assert_eq!( -// ::StorageVersion::get(), -// crate::migration::CURRENT_VERSION -// ); -// }); -// } - -// custom tests - -// #[test] -// fn bond_zero_should_work() { -// ExtBuilder::default().build().execute_with(|| { -// let (stash, controller) = (123, 456); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::RingBalance(0), -// RewardDestination::Stash, -// 0, -// )); -// -// let (stash, controller) = (234, 567); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::KtonBalance(0), -// RewardDestination::Stash, -// 0, -// )); -// }); -// } -// -// #[test] -// fn normal_kton_should_work() { -// ExtBuilder::default().build().execute_with(|| { -// { -// let (stash, controller) = (1001, 1000); -// -// let _ = Kton::deposit_creating(&stash, 10 * COIN); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::KtonBalance(10 * COIN), -// RewardDestination::Stash, -// 0, -// )); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash, -// active_ring: 0, -// active_deposit_ring: 0, -// active_kton: 10 * COIN, -// deposit_items: vec![], -// ring_staking_lock: Default::default(), -// kton_staking_lock: StakingLock { -// staking_amount: 10 * COIN, -// unbondings: vec![], -// }, -// } -// ); -// assert_eq!( -// Kton::locks(&stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 10 * COIN, -// unbondings: vec![], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// } -// -// { -// let (stash, controller) = (2001, 2000); -// -// // promise_month should not work for kton -// let _ = Kton::deposit_creating(&stash, 10 * COIN); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::KtonBalance(10 * COIN), -// RewardDestination::Stash, -// 12, -// )); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash, -// active_ring: 0, -// active_deposit_ring: 0, -// active_kton: 10 * COIN, -// deposit_items: vec![], -// ring_staking_lock: Default::default(), -// kton_staking_lock: StakingLock { -// staking_amount: 10 * COIN, -// unbondings: vec![], -// }, -// } -// ); -// } -// }); -// } -// -// #[test] -// fn time_deposit_ring_unbond_and_withdraw_automatically_should_work() { -// ExtBuilder::default().build().execute_with(|| { -// let (stash, controller) = (11, 10); -// -// let unbond_value = 10; -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::RingBalance(unbond_value), -// )); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1000 - unbond_value, -// unbondings: vec![NormalLock { -// amount: unbond_value, -// until: BondingDuration::get(), -// }], -// }), -// reasons: WithdrawReasons::all(), -// }], -// ); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash, -// active_ring: 1000 - unbond_value, -// active_deposit_ring: 0, -// active_kton: 0, -// deposit_items: vec![], -// ring_staking_lock: StakingLock { -// staking_amount: 1000 - unbond_value, -// unbondings: vec![NormalLock { -// amount: unbond_value, -// until: BondingDuration::get(), -// }], -// }, -// kton_staking_lock: Default::default(), -// }, -// ); -// -// let unbond_start = 30; -// Timestamp::set_timestamp(unbond_start); -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::RingBalance(COIN) -// )); -// -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 0, -// unbondings: vec![ -// NormalLock { -// amount: unbond_value, -// until: BondingDuration::get(), -// }, -// NormalLock { -// amount: 1000 - unbond_value, -// until: unbond_start + BondingDuration::get(), -// }, -// ], -// }), -// reasons: WithdrawReasons::all(), -// }], -// ); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash, -// active_ring: 0, -// active_deposit_ring: 0, -// active_kton: 0, -// deposit_items: vec![], -// ring_staking_lock: StakingLock { -// staking_amount: 0, -// unbondings: vec![ -// NormalLock { -// amount: unbond_value, -// until: BondingDuration::get(), -// }, -// NormalLock { -// amount: 1000 - unbond_value, -// until: unbond_start + BondingDuration::get(), -// }, -// ], -// }, -// kton_staking_lock: Default::default(), -// }, -// ); -// -// assert_err!( -// Ring::transfer(Origin::signed(stash), controller, 1), -// "account liquidity restrictions prevent withdrawal", -// ); -// -// Timestamp::set_timestamp(BondingDuration::get()); -// assert_ok!(Ring::transfer(Origin::signed(stash), controller, 1)); -// }); -// } -// -// #[test] -// fn normal_unbond_should_work() { -// ExtBuilder::default().build().execute_with(|| { -// let (stash, controller) = (11, 10); -// let value = 200 * COIN; -// let promise_month = 12; -// let _ = Ring::deposit_creating(&stash, 1000 * COIN); -// -// { -// let kton_free_balance = Kton::free_balance(&stash); -// let mut ledger = Staking::ledger(controller).unwrap(); -// -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::RingBalance(value), -// promise_month, -// )); -// assert_eq!( -// Kton::free_balance(&stash), -// kton_free_balance + inflation::compute_kton_return::(value, promise_month) -// ); -// ledger.active_ring += value; -// ledger.active_deposit_ring += value; -// ledger.deposit_items.push(TimeDepositItem { -// value, -// start_time: 0, -// expire_time: promise_month * MONTH_IN_MILLISECONDS, -// }); -// ledger.ring_staking_lock.staking_amount += value; -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// } -// -// { -// let kton_free_balance = Kton::free_balance(&stash); -// let mut ledger = Staking::ledger(controller).unwrap(); -// -// // We try to bond 1 kton, but stash only has 0.2 Kton. -// // extra = COIN.min(20_000_000) -// // bond += 20_000_000 -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::KtonBalance(COIN), -// 0, -// )); -// ledger.active_kton += kton_free_balance; -// ledger.kton_staking_lock.staking_amount += kton_free_balance; -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::KtonBalance(kton_free_balance) -// )); -// ledger.active_kton = 0; -// ledger.kton_staking_lock.staking_amount = 0; -// ledger.kton_staking_lock.unbondings.push(NormalLock { -// amount: kton_free_balance, -// until: BondingDuration::get(), -// }); -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// } -// }); -// } -// -// #[test] -// fn punished_claim_should_work() { -// ExtBuilder::default().build().execute_with(|| { -// let (stash, controller) = (1001, 1000); -// let promise_month = 36; -// let bond_value = 10; -// let _ = Ring::deposit_creating(&stash, 1000); -// let mut ledger = StakingLedger { -// stash, -// active_ring: bond_value, -// active_deposit_ring: bond_value, -// active_kton: 0, -// deposit_items: vec![TimeDepositItem { -// value: bond_value, -// start_time: 0, -// expire_time: promise_month * MONTH_IN_MILLISECONDS, -// }], -// ring_staking_lock: StakingLock { -// staking_amount: bond_value, -// unbondings: vec![], -// }, -// kton_staking_lock: Default::default(), -// }; -// -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::RingBalance(bond_value), -// RewardDestination::Stash, -// promise_month, -// )); -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// // Kton is 0, skip `unbond_with_punish`. -// assert_ok!(Staking::try_claim_deposits_with_punish( -// Origin::signed(controller), -// promise_month * MONTH_IN_MILLISECONDS, -// )); -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// -// // Set more kton balance to make it work. -// let _ = Kton::deposit_creating(&stash, COIN); -// assert_ok!(Staking::try_claim_deposits_with_punish( -// Origin::signed(controller), -// promise_month * MONTH_IN_MILLISECONDS, -// )); -// ledger.active_deposit_ring -= bond_value; -// ledger.deposit_items.clear(); -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// assert_eq!(Kton::free_balance(&stash), COIN - 3); -// }); -// } -// -// #[test] -// fn transform_to_deposited_ring_should_work() { -// ExtBuilder::default().build().execute_with(|| { -// let (stash, controller) = (1001, 1000); -// let _ = Ring::deposit_creating(&stash, COIN); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::RingBalance(COIN), -// RewardDestination::Stash, -// 0, -// )); -// let kton_free_balance = Kton::free_balance(&stash); -// let mut ledger = Staking::ledger(controller).unwrap(); -// -// assert_ok!(Staking::deposit_extra(Origin::signed(controller), COIN, 12)); -// ledger.active_deposit_ring += COIN; -// ledger.deposit_items.push(TimeDepositItem { -// value: COIN, -// start_time: 0, -// expire_time: 12 * MONTH_IN_MILLISECONDS, -// }); -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// assert_eq!(Kton::free_balance(&stash), kton_free_balance + (COIN / 10000)); -// }); -// } -// -// #[test] -// fn expired_ring_should_capable_to_promise_again() { -// ExtBuilder::default().build().execute_with(|| { -// let (stash, controller) = (1001, 1000); -// let _ = Ring::deposit_creating(&stash, 10); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::RingBalance(10), -// RewardDestination::Stash, -// 12, -// )); -// let mut ledger = Staking::ledger(controller).unwrap(); -// let ts = 13 * MONTH_IN_MILLISECONDS; -// let promise_extra_value = 5; -// -// Timestamp::set_timestamp(ts); -// assert_ok!(Staking::deposit_extra( -// Origin::signed(controller), -// promise_extra_value, -// 13, -// )); -// ledger.active_deposit_ring = promise_extra_value; -// // old deposit_item with 12 months promised removed -// ledger.deposit_items = vec![TimeDepositItem { -// value: promise_extra_value, -// start_time: ts, -// expire_time: 2 * ts, -// }]; -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// }); -// } -// -// #[test] -// fn inflation_should_be_correct() { -// ExtBuilder::default().build().execute_with(|| { -// let initial_issuance = 1_200_000_000 * COIN; -// let surplus_needed = initial_issuance - Ring::total_issuance(); -// let _ = Ring::deposit_into_existing(&11, surplus_needed); -// -// assert_eq!(Ring::total_issuance(), initial_issuance); -// }); -// -// // // breakpoint test -// // ExtBuilder::default().build().execute_with(|| { -// // gen_paired_account!(validator_1_stash(123), validator_1_controller(456), 0); -// // gen_paired_account!(validator_2_stash(234), validator_2_controller(567), 0); -// // gen_paired_account!(nominator_stash(345), nominator_controller(678), 0); -// // -// // assert_ok!(Staking::validate( -// // Origin::signed(validator_1_controller), -// // ValidatorPrefs { -// // node_name: vec![0; 8], -// // ..Default::default() -// // }, -// // )); -// // assert_ok!(Staking::validate( -// // Origin::signed(validator_2_controller), -// // ValidatorPrefs { -// // node_name: vec![1; 8], -// // ..Default::default() -// // }, -// // )); -// // assert_ok!(Staking::nominate( -// // Origin::signed(nominator_controller), -// // vec![validator_1_stash, validator_2_stash], -// // )); -// // -// // Timestamp::set_timestamp(1_575_448_345_000 - 12_000); -// // // breakpoint here -// // Staking::new_era(1); -// // -// // Timestamp::set_timestamp(1_575_448_345_000); -// // // breakpoint here -// // Staking::new_era(2); -// // -// // // breakpoint here -// // inflation::compute_total_payout::(11_999, 1_295_225_000, 9_987_999_900_000_000_000); -// // -// // loop {} -// // }); -// } -// -// #[test] -// fn validator_payment_ratio_should_work() { -// ExtBuilder::default().build().execute_with(|| { -// gen_paired_account!(validator_stash(123), validator_controller(456), 0); -// gen_paired_account!(nominator_stash(345), nominator_controller(678), 0); -// -// assert_ok!(Staking::validate( -// Origin::signed(validator_controller), -// ValidatorPrefs { -// node_name: vec![0; 8], -// validator_payment_ratio: 0, -// }, -// )); -// assert_ok!(Staking::nominate( -// Origin::signed(nominator_controller), -// vec![validator_stash], -// )); -// -// assert_eq!(Staking::reward_validator(&validator_stash, COIN).0.peek(), 0); -// -// assert_ok!(Staking::chill(Origin::signed(validator_controller))); -// assert_ok!(Staking::chill(Origin::signed(nominator_controller))); -// -// assert_ok!(Staking::validate( -// Origin::signed(validator_controller), -// ValidatorPrefs { -// node_name: vec![0; 8], -// validator_payment_ratio: 100, -// }, -// )); -// assert_ok!(Staking::nominate( -// Origin::signed(nominator_controller), -// vec![validator_stash], -// )); -// -// assert_eq!(Staking::reward_validator(&validator_stash, COIN).0.peek(), COIN); -// }); -// } -// -// #[test] -// fn check_node_name_should_work() { -// for node_name in [[0; 33].as_ref(), &[1; 34], &[2; 35]].iter() { -// let validator_prefs = ValidatorPrefs { -// node_name: (*node_name).to_vec(), -// ..Default::default() -// }; -// assert_err!(validator_prefs.check_node_name(), err::NODE_NAME_REACH_MAX); -// } -// -// for node_name in ["hello@darwinia.network"].iter() { -// let validator_prefs = ValidatorPrefs { -// node_name: (*node_name).into(), -// ..Default::default() -// }; -// assert_err!(validator_prefs.check_node_name(), err::NODE_NAME_CONTAINS_INVALID_CHARS); -// } -// -// for node_name in [ -// "com", -// "http", -// "https", -// "itering com", -// "http darwinia", -// "https darwinia", -// "http darwinia network", -// "https darwinia network", -// ] -// .iter() -// { -// let validator_prefs = ValidatorPrefs { -// node_name: (*node_name).into(), -// ..Default::default() -// }; -// assert_err!(validator_prefs.check_node_name(), err::NODE_NAME_CONTAINS_URLS); -// } -// -// for node_name in ["Darwinia Node"].iter() { -// let validator_prefs = ValidatorPrefs { -// node_name: (*node_name).into(), -// ..Default::default() -// }; -// assert_ok!(validator_prefs.check_node_name()); -// } -// } -// -// #[test] -// fn slash_should_not_touch_unbondings() { -// ExtBuilder::default().build().execute_with(|| { -// let (stash, controller) = (11, 10); -// -// assert_ok!(Staking::deposit_extra(Origin::signed(controller), 1000, 12)); -// let ledger = Staking::ledger(controller).unwrap(); -// // Only deposit_ring, no normal_ring. -// assert_eq!((ledger.active_ring, ledger.active_deposit_ring), (1000, 1000)); -// -// let _ = Ring::deposit_creating(&stash, 1000); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::RingBalance(1000), -// 0, -// )); -// let _ = Kton::deposit_creating(&stash, 1000); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::KtonBalance(1000), -// 0, -// )); -// -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::RingBalance(10) -// )); -// let ledger = Staking::ledger(controller).unwrap(); -// let unbondings = ( -// ledger.ring_staking_lock.unbondings.clone(), -// ledger.kton_staking_lock.unbondings.clone(), -// ); -// assert_eq!( -// (ledger.active_ring, ledger.active_deposit_ring), -// (1000 + 1000 - 10, 1000), -// ); -// -// >::insert( -// &stash, -// Exposure { -// total: 1, -// own: 1, -// others: vec![], -// }, -// ); -// // FIXME: slash strategy -// let _ = Staking::slash_validator(&stash, Power::max_value(), &Staking::stakers(&stash), &mut vec![]); -// let ledger = Staking::ledger(controller).unwrap(); -// assert_eq!( -// ( -// ledger.ring_staking_lock.unbondings.clone(), -// ledger.kton_staking_lock.unbondings.clone(), -// ), -// unbondings, -// ); -// assert_eq!((ledger.active_ring, ledger.active_deposit_ring), (0, 0)); -// }); -// } -// -// #[test] -// fn check_stash_already_bonded_and_controller_already_paired() { -// ExtBuilder::default().build().execute_with(|| { -// gen_paired_account!(unpaired_stash(123), unpaired_controller(456)); -// assert_err!( -// Staking::bond( -// Origin::signed(11), -// unpaired_controller, -// StakingBalances::RingBalance(COIN), -// RewardDestination::Stash, -// 0, -// ), -// err::STASH_ALREADY_BONDED, -// ); -// assert_err!( -// Staking::bond( -// Origin::signed(unpaired_stash), -// 10, -// StakingBalances::RingBalance(COIN), -// RewardDestination::Stash, -// 0, -// ), -// err::CONTROLLER_ALREADY_PAIRED, -// ); -// }); -// } -// -// #[test] -// fn pool_should_be_increased_and_decreased_correctly() { -// ExtBuilder::default().build().execute_with(|| { -// let mut ring_pool = Staking::ring_pool(); -// let mut kton_pool = Staking::kton_pool(); -// -// // bond: 100COIN -// gen_paired_account!(stash_1(111), controller_1(222), 0); -// gen_paired_account!(stash_2(333), controller_2(444), promise_month(12)); -// ring_pool += 100 * COIN; -// kton_pool += 100 * COIN; -// assert_eq!(Staking::ring_pool(), ring_pool); -// assert_eq!(Staking::kton_pool(), kton_pool); -// -// // unbond: 50Ring 50Kton -// assert_ok!(Staking::unbond( -// Origin::signed(controller_1), -// StakingBalances::RingBalance(50 * COIN) -// )); -// assert_ok!(Staking::unbond( -// Origin::signed(controller_1), -// StakingBalances::KtonBalance(25 * COIN) -// )); -// // not yet expired: promise for 12 months -// assert_ok!(Staking::unbond( -// Origin::signed(controller_2), -// StakingBalances::RingBalance(50 * COIN) -// )); -// assert_ok!(Staking::unbond( -// Origin::signed(controller_2), -// StakingBalances::KtonBalance(25 * COIN) -// )); -// ring_pool -= 50 * COIN; -// kton_pool -= 50 * COIN; -// assert_eq!(Staking::ring_pool(), ring_pool); -// assert_eq!(Staking::kton_pool(), kton_pool); -// -// // claim: 50Ring -// assert_ok!(Staking::try_claim_deposits_with_punish( -// Origin::signed(controller_2), -// promise_month * MONTH_IN_MILLISECONDS, -// )); -// // unbond deposit items: 12.5Ring -// Timestamp::set_timestamp(promise_month * MONTH_IN_MILLISECONDS); -// assert_ok!(Staking::unbond( -// Origin::signed(controller_2), -// StakingBalances::RingBalance(125 * COIN / 10), -// )); -// ring_pool -= 125 * COIN / 10; -// assert_eq!(Staking::ring_pool(), ring_pool); -// -// // slash: 37.5Ring 50Kton -// >::insert( -// &stash_1, -// Exposure { -// total: 1, -// own: 1, -// others: vec![], -// }, -// ); -// >::insert( -// &stash_2, -// Exposure { -// total: 1, -// own: 1, -// others: vec![], -// }, -// ); -// // FIXME: slash strategy -// let _ = Staking::slash_validator(&stash_1, Power::max_value(), &Staking::stakers(&stash_1), &mut vec![]); -// // FIXME: slash strategy -// let _ = Staking::slash_validator(&stash_2, Power::max_value(), &Staking::stakers(&stash_2), &mut vec![]); -// ring_pool -= 375 * COIN / 10; -// kton_pool -= 50 * COIN; -// assert_eq!(Staking::ring_pool(), ring_pool); -// assert_eq!(Staking::kton_pool(), kton_pool); -// }); -// } -// -// #[test] -// fn unbond_over_max_unbondings_chunks_should_fail() { -// ExtBuilder::default().build().execute_with(|| { -// gen_paired_account!(stash(123), controller(456)); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::RingBalance(COIN), -// RewardDestination::Stash, -// 0, -// )); -// -// for ts in 0..MAX_UNLOCKING_CHUNKS { -// Timestamp::set_timestamp(ts as u64); -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::RingBalance(1) -// )); -// } -// -// assert_err!( -// Staking::unbond(Origin::signed(controller), StakingBalances::RingBalance(1)), -// err::UNLOCK_CHUNKS_REACH_MAX, -// ); -// }); -// } -// -// #[test] -// fn promise_extra_should_not_remove_unexpired_items() { -// ExtBuilder::default().build().execute_with(|| { -// gen_paired_account!(stash(123), controller(456), promise_month(12)); -// let expired_items_len = 3; -// let expiry_date = promise_month * MONTH_IN_MILLISECONDS; -// -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::RingBalance(5 * COIN), -// 0, -// )); -// for _ in 0..expired_items_len { -// assert_ok!(Staking::deposit_extra(Origin::signed(controller), COIN, promise_month)); -// } -// -// Timestamp::set_timestamp(expiry_date - 1); -// assert_ok!(Staking::deposit_extra( -// Origin::signed(controller), -// 2 * COIN, -// promise_month, -// )); -// assert_eq!( -// Staking::ledger(controller).unwrap().deposit_items.len(), -// 2 + expired_items_len, -// ); -// -// Timestamp::set_timestamp(expiry_date); -// assert_ok!(Staking::deposit_extra( -// Origin::signed(controller), -// 2 * COIN, -// promise_month, -// )); -// assert_eq!(Staking::ledger(controller).unwrap().deposit_items.len(), 2); -// }); -// } +#[test] +fn rewards_should_work() { + // should check that: + // * rewards get recorded per session + // * rewards get paid per Era + // * Check that nominators are also rewarded + ExtBuilder::default().nominate(false).build().execute_with(|| { + // Init some balances + let _ = Ring::make_free_balance_be(&2, 500); + + let delay = 1000; + let init_balance_2 = Ring::total_balance(&2); + let init_balance_10 = Ring::total_balance(&10); + let init_balance_11 = Ring::total_balance(&11); + + // Set payee to controller + assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); + + // Initial config should be correct + assert_eq!(Staking::current_era(), 0); + assert_eq!(Session::current_index(), 0); + + // Add a dummy nominator. + // + // Equal division indicates that the reward will be equally divided among validator and + // nominator. + >::insert( + &11, + Exposure { + own_ring_balance: 500, + total_ring_balance: 1000, + own_kton_balance: 0, + own_power: 0, + total_power: 0, + total_kton_balance: 0, + others: vec![IndividualExposure { + who: 2, + ring_balance: 500, + kton_balance: 0, + power: 0, + }], + }, + ); + + >::insert(&2, RewardDestination::Stash); + assert_eq!(Staking::payee(2), RewardDestination::Stash); + assert_eq!(Staking::payee(11), RewardDestination::Controller); + + let mut block = 3; // Block 3 => Session 1 => Era 0 + System::set_block_number(block); + System::set_block_number(block * 5000); // on time. + Session::on_initialize(System::block_number()); + assert_eq!(Staking::current_era(), 0); + assert_eq!(Session::current_index(), 1); + >::reward_by_ids(vec![(11, 50)]); + >::reward_by_ids(vec![(11, 50)]); + // This is the second validator of the current elected set. + >::reward_by_ids(vec![(21, 50)]); + // This must be no-op as it is not an elected validator. + >::reward_by_ids(vec![(1001, 10_000)]); + + // Compute total payout now for whole duration as other parameter won't change + let total_payout = current_total_payout_for_duration(9 * 5 * 1000); + assert!(total_payout > 10); // Test is meaningful if reward something + + // No reward yet + assert_eq!(Ring::total_balance(&2), init_balance_2); + assert_eq!(Ring::total_balance(&10), init_balance_10); + assert_eq!(Ring::total_balance(&11), init_balance_11); + + block = 6; // Block 6 => Session 2 => Era 0 + System::set_block_number(block); + Timestamp::set_timestamp(block * 5000 + delay); // a little late. + Session::on_initialize(System::block_number()); + assert_eq!(Staking::current_era(), 0); + assert_eq!(Session::current_index(), 2); + + block = 9; // Block 9 => Session 3 => Era 1 + System::set_block_number(block); + Timestamp::set_timestamp(block * 5000); // back to being on time. no delays + + Session::on_initialize(System::block_number()); + assert_eq!(Staking::current_era(), 1); + assert_eq!(Session::current_index(), 3); + + // check reward is send + assert_ne!(Ring::total_balance(&2), init_balance_2); + assert_ne!(Ring::total_balance(&10), init_balance_10); + + // TODO: check the reward is correct or not + // 11 validator has 2/3 of the total rewards and half half for it and its nominator + // assert_eq_error_rate!(Ring::total_balance(&2), init_balance_2 + total_payout / 3, 1); + // assert_eq_error_rate!(Ring::total_balance(&10), init_balance_10 + total_payout / 3, 1); + assert_eq!(Ring::total_balance(&11), init_balance_11); + }); +} + +#[test] +fn multi_era_reward_should_work() { + // Should check that: + // The value of current_session_reward is set at the end of each era, based on + // slot_stake and session_reward. + ExtBuilder::default().nominate(false).build().execute_with(|| { + let init_balance_10 = Ring::total_balance(&10); + + // Set payee to controller + assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); + + //TODO: check pay out + // Compute now as other parameter won't change + let total_payout_0 = current_total_payout_for_duration(3000); + assert!(total_payout_0 > 10); // Test is meaningfull if reward something + >::reward_by_ids(vec![(11, 1)]); + + start_session(0); + start_session(1); + start_session(2); + start_session(3); + + assert_eq!(Staking::current_era(), 1); + assert_eq!(Ring::total_balance(&10), init_balance_10 + total_payout_0); + + start_session(4); + + //TODO: check pay out + let total_payout_1 = current_total_payout_for_duration(3000); + assert!(total_payout_1 > 10); // Test is meaningfull if reward something + >::reward_by_ids(vec![(11, 101)]); + + // new era is triggered here. + start_session(5); + + // pay time + assert_eq!( + Ring::total_balance(&10), + init_balance_10 + total_payout_0 + total_payout_1 + ); + }); +} + +#[test] +fn staking_should_work() { + // should test: + // * new validators can be added to the default set + // * new ones will be chosen per era + // * either one can unlock the stash and back-down from being a validator via `chill`ing. + ExtBuilder::default() + .nominate(false) + .fair(false) // to give 20 more staked value + .build() + .execute_with(|| { + Timestamp::set_timestamp(1); // Initialize time. + + // remember + compare this along with the test. + assert_eq_uvec!(validator_controllers(), vec![20, 10]); + + // put some money in account that we'll use. + for i in 1..5 { + let _ = Ring::make_free_balance_be(&i, 2000); + } + + // --- Block 1: + start_session(1); + // add a new candidate for being a validator. account 3 controlled by 4. + assert_ok!(Staking::bond( + Origin::signed(3), + 4, + StakingBalance::RingBalance(1500), + RewardDestination::Controller, + 0 + )); + assert_ok!(Staking::validate(Origin::signed(4), ValidatorPrefs::default())); + + // No effects will be seen so far. + assert_eq_uvec!(validator_controllers(), vec![20, 10]); + + // TODO: + // may cause by `fn compute_total_payout` of `inflation.rs` + // --- Block 2: + start_session(2); + + // No effects will be seen so far. Era has not been yet triggered. + assert_eq_uvec!(validator_controllers(), vec![20, 10]); + + // --- Block 3: the validators will now be queued. + start_session(3); + assert_eq!(Staking::current_era(), 1); + + // --- Block 4: the validators will now be changed. + start_session(4); + + assert_eq_uvec!(validator_controllers(), vec![20, 4]); + // --- Block 4: Unstake 4 as a validator, freeing up the balance stashed in 3 + // 4 will chill + Staking::chill(Origin::signed(4)).unwrap(); + + // --- Block 5: nothing. 4 is still there. + start_session(5); + assert_eq_uvec!(validator_controllers(), vec![20, 4]); + + // --- Block 6: 4 will not be a validator. + start_session(7); + assert_eq_uvec!(validator_controllers(), vec![20, 10]); + + // Note: the stashed value of 4 is still lock + assert_eq!( + Staking::ledger(&4), + Some(StakingLedger { + stash: 3, + active_ring: 1500, + active_deposit_ring: 0, + active_kton: 0, + deposit_items: Vec::new(), + ring_staking_lock: StakingLock { + staking_amount: 1500, + unbondings: Vec::new() + }, + kton_staking_lock: StakingLock { + staking_amount: 0, + unbondings: Vec::new() + } + }) + ); + // e.g. it cannot spend more than 500 that it has free from the total 2000 + assert_noop!( + Ring::reserve(&3, 501), + DispatchError::Module { + index: 0, + error: 1, + message: Some("LiquidityRestrictions"), + } + ); + assert_ok!(Ring::reserve(&3, 409)); + }); +} + +#[test] +fn less_than_needed_candidates_works() { + ExtBuilder::default() + .minimum_validator_count(1) + .validator_count(4) + .nominate(false) + .num_validators(3) + .build() + .execute_with(|| { + assert_eq!(Staking::validator_count(), 4); + assert_eq!(Staking::minimum_validator_count(), 1); + assert_eq_uvec!(validator_controllers(), vec![30, 20, 10]); + + start_era(1); + + // Previous set is selected. NO election algorithm is even executed. + assert_eq_uvec!(validator_controllers(), vec![30, 20, 10]); + + // But the exposure is updated in a simple way. No external votes exists. + // This is purely self-vote. + assert_eq!(Staking::stakers(10).others.len(), 0); + assert_eq!(Staking::stakers(20).others.len(), 0); + assert_eq!(Staking::stakers(30).others.len(), 0); + check_exposure_all(); + check_nominator_all(); + }); +} + +#[test] +fn no_candidate_emergency_condition() { + ExtBuilder::default() + .minimum_validator_count(10) + .validator_count(15) + .num_validators(4) + .validator_pool(true) + .nominate(false) + .build() + .execute_with(|| { + // initial validators + assert_eq_uvec!(validator_controllers(), vec![10, 20, 30, 40]); + + // set the minimum validator count. + ::MinimumValidatorCount::put(10); + ::ValidatorCount::put(15); + assert_eq!(Staking::validator_count(), 15); + + let _ = Staking::chill(Origin::signed(10)); + + // trigger era + System::set_block_number(1); + Session::on_initialize(System::block_number()); + + // Previous ones are elected. chill is invalidates. TODO: #2494 + assert_eq_uvec!(validator_controllers(), vec![10, 20, 30, 40]); + assert_eq!(Staking::current_elected().len(), 0); + }); +} + +#[test] +fn nominating_and_rewards_should_work() { + // PHRAGMEN OUTPUT: running this test with the reference impl gives: + // + // Sequential Phragmén gives + // 10 is elected with stake 2200.0 and score 0.0003333333333333333 + // 20 is elected with stake 1800.0 and score 0.0005555555555555556 + + // 10 has load 0.0003333333333333333 and supported + // 10 with stake 1000.0 + // 20 has load 0.0005555555555555556 and supported + // 20 with stake 1000.0 + // 30 has load 0 and supported + // 30 with stake 0 + // 40 has load 0 and supported + // 40 with stake 0 + // 2 has load 0.0005555555555555556 and supported + // 10 with stake 600.0 20 with stake 400.0 30 with stake 0.0 + // 4 has load 0.0005555555555555556 and supported + // 10 with stake 600.0 20 with stake 400.0 40 with stake 0.0 + + // Sequential Phragmén with post processing gives + // 10 is elected with stake 2000.0 and score 0.0003333333333333333 + // 20 is elected with stake 2000.0 and score 0.0005555555555555556 + + // 10 has load 0.0003333333333333333 and supported + // 10 with stake 1000.0 + // 20 has load 0.0005555555555555556 and supported + // 20 with stake 1000.0 + // 30 has load 0 and supported + // 30 with stake 0 + // 40 has load 0 and supported + // 40 with stake 0 + // 2 has load 0.0005555555555555556 and supported + // 10 with stake 400.0 20 with stake 600.0 30 with stake 0 + // 4 has load 0.0005555555555555556 and supported + // 10 with stake 600.0 20 with stake 400.0 40 with stake 0.0 + ExtBuilder::default() + .nominate(false) + .validator_pool(true) + .build() + .execute_with(|| { + // initial validators -- everyone is actually even. + assert_eq_uvec!(validator_controllers(), vec![40, 30]); + + // Set payee to controller + assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); + assert_ok!(Staking::set_payee(Origin::signed(20), RewardDestination::Controller)); + assert_ok!(Staking::set_payee(Origin::signed(30), RewardDestination::Controller)); + assert_ok!(Staking::set_payee(Origin::signed(40), RewardDestination::Controller)); + + // give the man some money + let initial_balance = 1000; + for i in [1, 2, 3, 4, 5, 10, 11, 20, 21].iter() { + let _ = Ring::make_free_balance_be(i, initial_balance); + } + + // bond two account pairs and state interest in nomination. + // 2 will nominate for 10, 20, 30 + assert_ok!(Staking::bond( + Origin::signed(1), + 2, + StakingBalance::RingBalance(1000), + RewardDestination::Controller, + 0 + )); + assert_ok!(Staking::nominate(Origin::signed(2), vec![11, 21, 31])); + // 4 will nominate for 10, 20, 40 + assert_ok!(Staking::bond( + Origin::signed(3), + 4, + StakingBalance::RingBalance(1000), + RewardDestination::Controller, + 0 + )); + assert_ok!(Staking::nominate(Origin::signed(4), vec![11, 21, 41])); + + // the total reward for era 0 + let total_payout_0 = current_total_payout_for_duration(3000); + assert!(total_payout_0 > 100); // Test is meaningfull if reward something + >::reward_by_ids(vec![(41, 1)]); + >::reward_by_ids(vec![(31, 1)]); + >::reward_by_ids(vec![(21, 10)]); // must be no-op + >::reward_by_ids(vec![(11, 10)]); // must be no-op + + start_era(1); + + // 10 and 20 have more votes, they will be chosen by phragmen. + assert_eq_uvec!(validator_controllers(), vec![20, 10]); + + // TODO: checkout total payout + // // OLD validators must have already received some rewards. + // assert_eq!(Ring::total_balance(&40), 1 + total_payout_0 / 2); + // assert_eq!(Ring::total_balance(&30), 1 + total_payout_0 / 2); + + // ------ check the staked value of all parties. + + if cfg!(feature = "equalize") { + // TODO: fix equalize + // // total expo of 10, with 1200 coming from nominators (externals), according to phragmen. + // assert_eq!(Staking::stakers(11).own, 1000); + // assert_eq_error_rate!(Staking::stakers(11).total, 1000 + 1000, 2); + // // 2 and 4 supported 10, each with stake 600, according to phragmen. + // assert_eq!( + // Staking::stakers(11) + // .others + // .iter() + // .map(|e| e.value) + // .collect::>>(), + // vec![600, 400] + // ); + // assert_eq!( + // Staking::stakers(11).others.iter().map(|e| e.who).collect::>(), + // vec![3, 1] + // ); + // // total expo of 20, with 500 coming from nominators (externals), according to phragmen. + // assert_eq!(Staking::stakers(21).own, 1000); + // assert_eq_error_rate!(Staking::stakers(21).total, 1000 + 1000, 2); + // // 2 and 4 supported 20, each with stake 250, according to phragmen. + // assert_eq!( + // Staking::stakers(21) + // .others + // .iter() + // .map(|e| e.value) + // .collect::>>(), + // vec![400, 600] + // ); + // assert_eq!( + // Staking::stakers(21).others.iter().map(|e| e.who).collect::>(), + // vec![3, 1] + // ); + } else { + // total expo of 10, with 1200 coming from nominators (externals), according to phragmen. + assert_eq!(Staking::stakers(11).own_ring_balance, 1000); + assert_eq!(Staking::stakers(11).total_ring_balance, 1000 + 800); + // 2 and 4 supported 10, each with stake 600, according to phragmen. + assert_eq!( + Staking::stakers(11) + .others + .iter() + .map(|e| e.ring_balance) + .collect::>(), + vec![400, 400] + ); + assert_eq!( + Staking::stakers(11).others.iter().map(|e| e.who).collect::>(), + vec![3, 1] + ); + // total expo of 20, with 500 coming from nominators (externals), according to phragmen. + assert_eq!(Staking::stakers(21).own_ring_balance, 1000); + assert_eq_error_rate!(Staking::stakers(21).total_ring_balance, 1000 + 1200, 2); + // 2 and 4 supported 20, each with stake 250, according to phragmen. + assert_eq!( + Staking::stakers(21) + .others + .iter() + .map(|e| e.ring_balance) + .collect::>(), + vec![600, 600] + ); + assert_eq!( + Staking::stakers(21).others.iter().map(|e| e.who).collect::>(), + vec![3, 1] + ); + } + + // They are not chosen anymore + assert_eq!(Staking::stakers(31).total_ring_balance, 0); + assert_eq!(Staking::stakers(41).total_ring_balance, 0); + + // the total reward for era 1 + let total_payout_1 = current_total_payout_for_duration(3000); + assert!(total_payout_1 > 100); // Test is meaningfull if reward something + >::reward_by_ids(vec![(41, 10)]); // must be no-op + >::reward_by_ids(vec![(31, 10)]); // must be no-op + >::reward_by_ids(vec![(21, 2)]); + >::reward_by_ids(vec![(11, 1)]); + + start_era(2); + + // nothing else will happen, era ends and rewards are paid again, + // it is expected that nominators will also be paid. See below + + let payout_for_10 = total_payout_1 / 3; + let payout_for_20 = 2 * total_payout_1 / 3; + if cfg!(feature = "equalize") { + // TODO: fix equalize + // // Nominator 2: has [400 / 2000 ~ 1 / 5 from 10] + [600 / 2000 ~ 3 / 10 from 20]'s reward. + // assert_eq_error_rate!( + // Ring::total_balance(&2), + // initial_balance + payout_for_10 / 5 + payout_for_20 * 3 / 10, + // 2, + // ); + // // Nominator 4: has [400 / 2000 ~ 1 / 5 from 20] + [600 / 2000 ~ 3 / 10 from 10]'s reward. + // assert_eq_error_rate!( + // Ring::total_balance(&4), + // initial_balance + payout_for_20 / 5 + payout_for_10 * 3 / 10, + // 2, + // ); + + // // Validator 10: got 1000 / 2000 external stake. + // assert_eq_error_rate!(Ring::total_balance(&10), initial_balance + payout_for_10 / 2, 1,); + // // Validator 20: got 1000 / 2000 external stake. + // assert_eq_error_rate!(Ring::total_balance(&20), initial_balance + payout_for_20 / 2, 1,); + } else { + // Nominator 2: has [400/1800 ~ 2/9 from 10] + [600/2200 ~ 3/11 from 20]'s reward. ==> 2/9 + 3/11 + assert_eq_error_rate!( + Ring::total_balance(&2), + initial_balance + (2 * payout_for_10 / 9 + 3 * payout_for_20 / 11), + 1, + ); + // Nominator 4: has [400/1800 ~ 2/9 from 10] + [600/2200 ~ 3/11 from 20]'s reward. ==> 2/9 + 3/11 + assert_eq_error_rate!( + Ring::total_balance(&4), + initial_balance + (2 * payout_for_10 / 9 + 3 * payout_for_20 / 11), + 1, + ); + + // Validator 10: got 800 / 1800 external stake => 8/18 =? 4/9 => Validator's share = 5/9 + assert_eq_error_rate!(Ring::total_balance(&10), initial_balance + 5 * payout_for_10 / 9, 1,); + // Validator 20: got 1200 / 2200 external stake => 12/22 =? 6/11 => Validator's share = 5/11 + assert_eq_error_rate!(Ring::total_balance(&20), initial_balance + 5 * payout_for_20 / 11, 1,); + } + + check_exposure_all(); + check_nominator_all(); + }); +} + +#[test] +fn nominators_also_get_slashed() { + // A nominator should be slashed if the validator they nominated is slashed + // Here is the breakdown of roles: + // 10 - is the controller of 11 + // 11 - is the stash. + // 2 - is the nominator of 20, 10 + ExtBuilder::default().nominate(false).build().execute_with(|| { + assert_eq!(Staking::validator_count(), 2); + + // Set payee to controller + assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); + + // give the man some money. + let initial_balance = 1000; + for i in [1, 2, 3, 10].iter() { + let _ = Ring::make_free_balance_be(i, initial_balance); + } + + // 2 will nominate for 10, 20 + let nominator_stake = StakingBalance::RingBalance(500); + assert_ok!(Staking::bond( + Origin::signed(1), + 2, + nominator_stake.clone(), + RewardDestination::default(), + 0 + )); + assert_ok!(Staking::nominate(Origin::signed(2), vec![20, 10])); + + // TODO + let total_payout = current_total_payout_for_duration(3000); + assert!(total_payout > 100); // Test is meaningfull if reward something + >::reward_by_ids(vec![(11, 1)]); + + // new era, pay rewards, + // TODO + start_era(1); + + // Nominator stash didn't collect any. + assert_eq!(Ring::total_balance(&2), initial_balance); + + // 10 goes offline + on_offence_now( + &[OffenceDetails { + offender: (11, Staking::stakers(&11)), + reporters: vec![], + }], + &[Perbill::from_percent(5)], + ); + let expo = Staking::stakers(11); + let slash_value = 50; + let total_slash = expo.total_ring_balance.min(slash_value); + let validator_slash = expo.own_ring_balance.min(total_slash); + let nominator_slash = match nominator_stake { + RingBalance(v) => v.min(total_slash - validator_slash), + _ => panic!("nominator slash should be Ring Balance here"), + }; + + // initial + first era reward + slash + assert_eq!(Ring::total_balance(&11), initial_balance - validator_slash); + assert_eq!(Ring::total_balance(&2), initial_balance - nominator_slash); + check_exposure_all(); + check_nominator_all(); + // Because slashing happened. + assert!(is_disabled(10)); + }); +} + +#[test] +fn double_staking_should_fail() { + // should test (in the same order): + // * an account already bonded as stash cannot be be stashed again. + // * an account already bonded as stash cannot nominate. + // * an account already bonded as controller can nominate. + ExtBuilder::default().build().execute_with(|| { + let arbitrary_value = 5; + // 2 = controller, 1 stashed => ok + assert_ok!(Staking::bond( + Origin::signed(1), + 2, + StakingBalance::RingBalance(arbitrary_value), + RewardDestination::default(), + 0 + )); + // 4 = not used so far, 1 stashed => not allowed. + assert_noop!( + Staking::bond( + Origin::signed(1), + 4, + StakingBalance::RingBalance(arbitrary_value), + RewardDestination::default(), + 0 + ), + Error::::AlreadyBonded, + ); + // 1 = stashed => attempting to nominate should fail. + assert_noop!( + Staking::nominate(Origin::signed(1), vec![1]), + Error::::NotController + ); + // 2 = controller => nominating should work. + assert_ok!(Staking::nominate(Origin::signed(2), vec![1])); + }); +} + +#[test] +fn double_controlling_should_fail() { + // should test (in the same order): + // * an account already bonded as controller CANNOT be reused as the controller of another account. + ExtBuilder::default().build().execute_with(|| { + let arbitrary_value = 5; + // 2 = controller, 1 stashed => ok + assert_ok!(Staking::bond( + Origin::signed(1), + 2, + StakingBalance::RingBalance(arbitrary_value), + RewardDestination::default(), + 0 + )); + // 2 = controller, 3 stashed (Note that 2 is reused.) => no-op + assert_noop!( + Staking::bond( + Origin::signed(3), + 2, + StakingBalance::RingBalance(arbitrary_value), + RewardDestination::default(), + 0 + ), + Error::::AlreadyPaired, + ); + }); +} + +#[test] +fn session_and_eras_work() { + ExtBuilder::default().build().execute_with(|| { + assert_eq!(Staking::current_era(), 0); + + // Block 1: No change. + start_session(0); + assert_eq!(Session::current_index(), 1); + assert_eq!(Staking::current_era(), 0); + + // Block 2: Simple era change. + start_session(2); + assert_eq!(Session::current_index(), 3); + assert_eq!(Staking::current_era(), 1); + + // Block 3: Schedule an era length change; no visible changes. + start_session(3); + assert_eq!(Session::current_index(), 4); + assert_eq!(Staking::current_era(), 1); + + // Block 4: Era change kicks in. + start_session(5); + assert_eq!(Session::current_index(), 6); + assert_eq!(Staking::current_era(), 2); + + // Block 5: No change. + start_session(6); + assert_eq!(Session::current_index(), 7); + assert_eq!(Staking::current_era(), 2); + + // Block 6: No change. + start_session(7); + assert_eq!(Session::current_index(), 8); + assert_eq!(Staking::current_era(), 2); + + // Block 7: Era increment. + start_session(8); + assert_eq!(Session::current_index(), 9); + assert_eq!(Staking::current_era(), 3); + }); +} + +#[test] +fn forcing_new_era_works() { + ExtBuilder::default().build().execute_with(|| { + // normal flow of session. + assert_eq!(Staking::current_era(), 0); + start_session(0); + assert_eq!(Staking::current_era(), 0); + start_session(1); + assert_eq!(Staking::current_era(), 0); + start_session(2); + assert_eq!(Staking::current_era(), 1); + + // no era change. + ForceEra::put(Forcing::ForceNone); + start_session(3); + assert_eq!(Staking::current_era(), 1); + start_session(4); + assert_eq!(Staking::current_era(), 1); + start_session(5); + assert_eq!(Staking::current_era(), 1); + start_session(6); + assert_eq!(Staking::current_era(), 1); + + // back to normal. + // this immediately starts a new session. + ForceEra::put(Forcing::NotForcing); + start_session(7); + assert_eq!(Staking::current_era(), 2); + start_session(8); + assert_eq!(Staking::current_era(), 2); + + // forceful change + ForceEra::put(Forcing::ForceAlways); + start_session(9); + assert_eq!(Staking::current_era(), 3); + start_session(10); + assert_eq!(Staking::current_era(), 4); + start_session(11); + assert_eq!(Staking::current_era(), 5); + + // just one forceful change + ForceEra::put(Forcing::ForceNew); + start_session(12); + assert_eq!(Staking::current_era(), 6); + + assert_eq!(ForceEra::get(), Forcing::NotForcing); + start_session(13); + assert_eq!(Staking::current_era(), 6); + }); +} + +#[test] +fn cannot_transfer_staked_balance() { + // Tests that a stash account cannot transfer funds + ExtBuilder::default().nominate(false).build().execute_with(|| { + // Confirm account 11 is stashed + assert_eq!(Staking::bonded(&11), Some(10)); + // Confirm account 11 has some free balance + assert_eq!(Ring::free_balance(&11), 1000); + // Confirm account 11 (via controller 10) is totally staked + assert_eq!(Staking::stakers(&11).total_ring_balance, 1000); + // Confirm account 11 cannot transfer as a result + assert_noop!( + Ring::transfer(Origin::signed(11), 20, 1), + DispatchError::Module { + index: 0, + error: 1, + message: Some("LiquidityRestrictions"), + } + ); + + // Give account 11 extra free balance + let _ = Ring::make_free_balance_be(&11, 10000); + // Confirm that account 11 can now transfer some balance + assert_ok!(Ring::transfer(Origin::signed(11), 20, 1)); + }); +} + +#[test] +fn cannot_transfer_staked_balance_2() { + // Tests that a stash account cannot transfer funds + // Same test as above but with 20, and more accurate. + // 21 has 2000 free balance but 1000 at stake + ExtBuilder::default() + .nominate(false) + .fair(true) + .build() + .execute_with(|| { + // Confirm account 21 is stashed + assert_eq!(Staking::bonded(&21), Some(20)); + // Confirm account 21 has some free balance + assert_eq!(Ring::free_balance(&21), 2000); + // Confirm account 21 (via controller 20) is totally staked + assert_eq!(Staking::stakers(&21).total_ring_balance, 1000); + // Confirm account 21 can transfer at most 1000 + assert_noop!( + Ring::transfer(Origin::signed(21), 20, 1001), + DispatchError::Module { + index: 0, + error: 1, + message: Some("LiquidityRestrictions"), + } + ); + assert_ok!(Ring::transfer(Origin::signed(21), 20, 1000)); + }); +} + +#[test] +fn cannot_reserve_staked_balance() { + // Checks that a bonded account cannot reserve balance from free balance + ExtBuilder::default().build().execute_with(|| { + // Confirm account 11 is stashed + assert_eq!(Staking::bonded(&11), Some(10)); + // Confirm account 11 has some free balance + assert_eq!(Ring::free_balance(&11), 1000); + // Confirm account 11 (via controller 10) is totally staked + assert_eq!(Staking::stakers(&11).own_ring_balance, 1000); + // Confirm account 11 cannot transfer as a result + assert_noop!( + Ring::reserve(&11, 1), + DispatchError::Module { + index: 0, + error: 1, + message: Some("LiquidityRestrictions"), + } + ); + + // Give account 11 extra free balance + let _ = Ring::make_free_balance_be(&11, 10000); + // Confirm account 11 can now reserve balance + assert_ok!(Ring::reserve(&11, 1)); + }); +} + +#[test] +fn reward_destination_works() { + // Rewards go to the correct destination as determined in Payee + ExtBuilder::default().nominate(false).build().execute_with(|| { + // Check that account 11 is a validator + assert!(Staking::current_elected().contains(&11)); + // Check the balance of the validator account + assert_eq!(Ring::free_balance(&10), 1); + // Check the balance of the stash account + assert_eq!(Ring::free_balance(&11), 1000); + // Check how much is at stake + assert_eq!( + Staking::ledger(&10), + Some(StakingLedger { + stash: 11, + active_ring: 1000, + active_deposit_ring: 0, + active_kton: 0, + deposit_items: Vec::new(), + ring_staking_lock: StakingLock { + staking_amount: 1000, + unbondings: Vec::new() + }, + kton_staking_lock: StakingLock { + staking_amount: 0, + unbondings: Vec::new() + } + }) + ); + + //// Compute total payout now for whole duration as other parameter won't change + //// TODO: fix payout + //let total_payout_0 = current_total_payout_for_duration(3000); + //assert!(total_payout_0 > 100); // Test is meaningfull if reward something + //>::reward_by_ids(vec![(11, 1)]); + + ////TODO: fix start era + //start_era(1); + + //// Check that RewardDestination is Staked (default) + //assert_eq!(Staking::payee(&11), RewardDestination::Staked { promise_month: 0 }); + //// Check that reward went to the stash account of validator + //assert_eq!(Ring::free_balance(&11), 1000 + total_payout_0); + //// Check that amount at stake increased accordingly + //assert_eq!( + // Staking::ledger(&10), + // Some(StakingLedger { + // stash: 11, + // active_ring: 1000 + total_payout_0, + // active_deposit_ring: 0, + // active_kton: 0, + // deposit_items: Vec::new(), + // ring_staking_lock: StakingLock { + // staking_amount: 1000 + total_payout_0, + // unbondings: Vec::new() + // }, + // kton_staking_lock: StakingLock { + // staking_amount: 0, + // unbondings: Vec::new() + // } + // }) + //); + + ////Change RewardDestination to Stash + //>::insert(&11, RewardDestination::Stash); + + //// Compute total payout now for whole duration as other parameter won't change + //let total_payout_1 = current_total_payout_for_duration(3000); + //assert!(total_payout_1 > 100); // Test is meaningfull if reward something + //>::reward_by_ids(vec![(11, 1)]); + + //start_era(2); + + //// Check that RewardDestination is Stash + //assert_eq!(Staking::payee(&11), RewardDestination::Stash); + //// Check that reward went to the stash account + //assert_eq!(Ring::free_balance(&11), 1000 + total_payout_0 + total_payout_1); + //// Record this value + //let recorded_stash_balance = 1000 + total_payout_0 + total_payout_1; + //// Check that amount at stake is NOT increased + //assert_eq!( + // Staking::ledger(&10), + // Some(StakingLedger { + // stash: 11, + // active_ring: 1000 + total_payout_0, + // active_deposit_ring: 0, + // active_kton: 0, + // deposit_items: Vec::new(), + // ring_staking_lock: StakingLock { + // staking_amount: 1000 + total_payout_0, + // unbondings: Vec::new() + // }, + // kton_staking_lock: StakingLock { + // staking_amount: 0, + // unbondings: Vec::new() + // } + // }) + //); + + //// Change RewardDestination to Controller + //>::insert(&11, RewardDestination::Controller); + + //// Check controller balance + //assert_eq!(Ring::free_balance(&10), 1); + + //// Compute total payout now for whole duration as other parameter won't change + //let total_payout_2 = current_total_payout_for_duration(3000); + //assert!(total_payout_2 > 100); // Test is meaningfull if reward something + //>::reward_by_ids(vec![(11, 1)]); + + //start_era(3); + + //// Check that RewardDestination is Controller + //assert_eq!(Staking::payee(&11), RewardDestination::Controller); + //// Check that reward went to the controller account + //assert_eq!(Ring::free_balance(&10), 1 + total_payout_2); + //// Check that amount at stake is NOT increased + //assert_eq!( + // Staking::ledger(&10), + // Some(StakingLedger { + // stash: 11, + // active_ring: 1000 + total_payout_0, + // active_deposit_ring: 0, + // active_kton: 0, + // deposit_items: Vec::new(), + // ring_staking_lock: StakingLock { + // staking_amount: 1000 + total_payout_0, + // unbondings: Vec::new() + // }, + // kton_staking_lock: StakingLock { + // staking_amount: 0, + // unbondings: Vec::new() + // } + // }) + //); + //// Check that amount in staked account is NOT increased. + //assert_eq!(Ring::free_balance(&11), recorded_stash_balance); + }); +} + +#[test] +fn validator_payment_prefs_work() { + // Test that validator preferences are correctly honored + // Note: unstake threshold is being directly tested in slashing tests. + // This test will focus on validator payment. + ExtBuilder::default().build().execute_with(|| { + // Initial config + let stash_initial_balance = Ring::total_balance(&11); + + // check the balance of a validator accounts. + assert_eq!(Ring::total_balance(&10), 1); + // check the balance of a validator's stash accounts. + assert_eq!(Ring::total_balance(&11), stash_initial_balance); + // and the nominator (to-be) + let _ = Ring::make_free_balance_be(&2, 500); + + // add a dummy nominator. + >::insert( + &11, + Exposure { + own_ring_balance: 500, + total_ring_balance: 1000, + own_kton_balance: 0, + own_power: 0, + total_power: 0, + total_kton_balance: 0, + others: vec![IndividualExposure { + who: 2, + ring_balance: 500, + kton_balance: 0, + power: 0, + }], + }, + ); + >::insert(&2, RewardDestination::Stash); + >::insert( + &11, + ValidatorPrefs { + commission: Perbill::from_percent(50), + }, + ); + + // // TODO: fix current_total_payout_for_duration + // // Compute total payout now for whole duration as other parameter won't change + // let total_payout_0 = current_total_payout_for_duration(3000); + // assert!(total_payout_0 > 100); // Test is meaningfull if reward something + // >::reward_by_ids(vec![(11, 1)]); + + // start_era(1); + + // // whats left to be shared is the sum of 3 rounds minus the validator's cut. + // let shared_cut = total_payout_0 / 2; + // // Validator's payee is Staked account, 11, reward will be paid here. + // assert_eq!( + // Ring::total_balance(&11), + // stash_initial_balance + shared_cut / 2 + shared_cut + // ); + // // Controller account will not get any reward. + // assert_eq!(Ring::total_balance(&10), 1); + // // Rest of the reward will be shared and paid to the nominator in stake. + // assert_eq!(Ring::total_balance(&2), 500 + shared_cut / 2); + + // check_exposure_all(); + // check_nominator_all(); + }); +} + +#[test] +fn bond_extra_works() { + // Tests that extra `free_balance` in the stash can be added to stake + // NOTE: this tests only verifies `StakingLedger` for correct updates + // See `bond_extra_and_withdraw_unbonded_works` for more details and updates on `Exposure`. + ExtBuilder::default().build().execute_with(|| { + // Check that account 10 is a validator + assert!(>::exists(11)); + // Check that account 10 is bonded to account 11 + assert_eq!(Staking::bonded(&11), Some(10)); + // Check how much is at stake + assert_eq!( + Staking::ledger(&10), + Some(StakingLedger { + stash: 11, + active_ring: 1000, + active_deposit_ring: 0, + active_kton: 0, + deposit_items: Vec::new(), + ring_staking_lock: StakingLock { + staking_amount: 1000, + unbondings: Vec::new() + }, + kton_staking_lock: StakingLock { + staking_amount: 0, + unbondings: Vec::new() + } + }) + ); + + // Give account 11 some large free balance greater than total + let _ = Ring::make_free_balance_be(&11, 1000000); + + // Call the bond_extra function from controller, add only 100 + assert_ok!(Staking::bond_extra( + Origin::signed(11), + StakingBalance::RingBalance(100), + 12 + )); + // There should be 100 more `total` and `active` in the ledger + assert_eq!( + Staking::ledger(&10), + Some(StakingLedger { + stash: 11, + active_ring: 1000 + 100, + active_deposit_ring: 100, + active_kton: 0, + deposit_items: vec![TimeDepositItem { + value: 100, + start_time: 0, + expire_time: 31104000000 + }], + ring_staking_lock: StakingLock { + staking_amount: 1000 + 100, + unbondings: Vec::new() + }, + kton_staking_lock: StakingLock { + staking_amount: 0, + unbondings: Vec::new() + } + }) + ); + + // Call the bond_extra function with a large number, should handle it + assert_ok!(Staking::bond_extra( + Origin::signed(11), + StakingBalance::RingBalance(u64::max_value()), + 0 + )); + // The full amount of the funds should now be in the total and active + assert_eq!( + Staking::ledger(&10), + Some(StakingLedger { + stash: 11, + active_ring: 1000000, + active_deposit_ring: 100, + active_kton: 0, + deposit_items: vec![TimeDepositItem { + value: 100, + start_time: 0, + expire_time: 31104000000 + }], + ring_staking_lock: StakingLock { + staking_amount: 1000000, + unbondings: Vec::new() + }, + kton_staking_lock: StakingLock { + staking_amount: 0, + unbondings: Vec::new() + } + }) + ); + }); +} + +// Note: rm bond_extra_and_withdraw_unbonded_works, because we are not using this function + +#[test] +fn too_many_unbond_calls_should_not_work() { + ExtBuilder::default().build().execute_with(|| { + // locked at era 0 until 3 + for _ in 0..MAX_UNLOCKING_CHUNKS - 1 { + assert_ok!(Staking::unbond(Origin::signed(10), StakingBalance::RingBalance(1))); + } + + start_era(1); + + // locked at era 1 until 4 + assert_ok!(Staking::unbond(Origin::signed(10), StakingBalance::RingBalance(1))); + // // TODO: checkout this should fail or not + // can't do more. + // assert_noop!( + // Staking::unbond(Origin::signed(10), StakingBalance::RingBalance(1)), + // Error::::NoMoreChunks + // ); + + // start_era(3); + + // assert_noop!( + // Staking::unbond(Origin::signed(10), StakingBalance::RingBalance(1)), + // Error::::NoMoreChunks + // ); + // // TODO: checkout the free up issues + // // Free up automatically. + // start_era(10); + // // assert_ok!(Staking::withdraw_unbonded(Origin::signed(10))); + + // // Can add again. + // assert_ok!(Staking::unbond(Origin::signed(10), StakingBalance::RingBalance(1))); + // assert_eq!(Staking::ledger(&10).unwrap().ring_staking_lock.unbondings.len(), 2); + }) +} + +#[test] +fn slot_stake_is_least_staked_validator_and_exposure_defines_maximum_punishment() { + // Test that slot_stake is determined by the least staked validator + // Test that slot_stake is the maximum punishment that can happen to a validator + ExtBuilder::default() + .nominate(false) + .fair(false) + .build() + .execute_with(|| { + // Confirm validator count is 2 + assert_eq!(Staking::validator_count(), 2); + // Confirm account 10 and 20 are validators + assert!(>::exists(&11) && >::exists(&21)); + + assert_eq!(Staking::stakers(&11).total_ring_balance, 1000); + assert_eq!(Staking::stakers(&21).total_ring_balance, 2000); + + // Give the man some money. + let _ = Ring::make_free_balance_be(&10, 1000); + let _ = Ring::make_free_balance_be(&20, 1000); + + // We confirm initialized slot_stake is this value + // TODO: checkout why this not work + // assert_eq!( + // Staking::slot_stake() as u64, + // Staking::stakers(&11).total_ring_balance as u64 + // ); + + // // Now lets lower account 20 stake + // >::insert( + // &21, + // Exposure { + // total: 69, + // own: 69, + // others: vec![], + // }, + // ); + // assert_eq!(Staking::stakers(&21).total_ring_balance, 69); + // >::insert( + // &20, + // StakingLedger { + // stash: 22, + // total: 69, + // active: 69, + // unlocking: vec![], + // }, + // ); + + // // Compute total payout now for whole duration as other parameter won't change + // let total_payout_0 = current_total_payout_for_duration(3000); + // assert!(total_payout_0 > 100); // Test is meaningfull if reward something + // >::reward_by_ids(vec![(11, 1)]); + // >::reward_by_ids(vec![(21, 1)]); + + // // New era --> rewards are paid --> stakes are changed + // start_era(1); + + // // -- new balances + reward + // assert_eq!(Staking::stakers(&11).total_ring_balance, 1000 + total_payout_0 / 2); + // assert_eq!(Staking::stakers(&21).total_ring_balance, 69 + total_payout_0 / 2); + + // let _11_balance = Ring::free_balance(&11); + // assert_eq!(_11_balance, 1000 + total_payout_0 / 2); + + // // -- slot stake should also be updated. + // assert_eq!(Staking::slot_stake(), 69 + total_payout_0 / 2); + + // check_exposure_all(); + // check_nominator_all(); + }); +} + +#[test] +fn on_free_balance_zero_stash_removes_validator() { + // Tests that validator storage items are cleaned up when stash is empty + // Tests that storage items are untouched when controller is empty + ExtBuilder::default().existential_deposit(10).build().execute_with(|| { + // Check the balance of the validator account + assert_eq!(Ring::free_balance(&10), 256); + // Check the balance of the stash account + assert_eq!(Ring::free_balance(&11), 256000); + // Check these two accounts are bonded + assert_eq!(Staking::bonded(&11), Some(10)); + + // Set some storage items which we expect to be cleaned up + // Set payee information + assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Stash)); + + // Check storage items that should be cleaned up + assert!(>::exists(&10)); + assert!(>::exists(&11)); + assert!(>::exists(&11)); + assert!(>::exists(&11)); + + // Reduce free_balance of controller to 0 + let _ = Ring::slash(&10, u64::max_value()); + + // Check the balance of the stash account has not been touched + assert_eq!(Ring::free_balance(&11), 256000); + // Check these two accounts are still bonded + assert_eq!(Staking::bonded(&11), Some(10)); + + // Check storage items have not changed + assert!(>::exists(&10)); + assert!(>::exists(&11)); + assert!(>::exists(&11)); + assert!(>::exists(&11)); + + // Reduce free_balance of stash to 0 + let _ = Ring::slash(&11, u64::max_value()); + // Check total balance of stash + assert_eq!(Ring::total_balance(&11), 0); + + // Check storage items do not exist + assert!(!>::exists(&10)); + assert!(!>::exists(&11)); + assert!(!>::exists(&11)); + assert!(!>::exists(&11)); + assert!(!>::exists(&11)); + }); +} + +#[test] +fn on_free_balance_zero_stash_removes_nominator() { + // Tests that nominator storage items are cleaned up when stash is empty + // Tests that storage items are untouched when controller is empty + ExtBuilder::default().existential_deposit(10).build().execute_with(|| { + // Make 10 a nominator + assert_ok!(Staking::nominate(Origin::signed(10), vec![20])); + // Check that account 10 is a nominator + assert!(>::exists(11)); + // Check the balance of the nominator account + assert_eq!(Ring::free_balance(&10), 256); + // Check the balance of the stash account + assert_eq!(Ring::free_balance(&11), 256000); + + // Set payee information + assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Stash)); + + // Check storage items that should be cleaned up + assert!(>::exists(&10)); + assert!(>::exists(&11)); + assert!(>::exists(&11)); + assert!(>::exists(&11)); + + // Reduce free_balance of controller to 0 + let _ = Ring::slash(&10, u64::max_value()); + // Check total balance of account 10 + assert_eq!(Ring::total_balance(&10), 0); + + // Check the balance of the stash account has not been touched + assert_eq!(Ring::free_balance(&11), 256000); + // Check these two accounts are still bonded + assert_eq!(Staking::bonded(&11), Some(10)); + + // Check storage items have not changed + assert!(>::exists(&10)); + assert!(>::exists(&11)); + assert!(>::exists(&11)); + assert!(>::exists(&11)); + + // Reduce free_balance of stash to 0 + let _ = Ring::slash(&11, u64::max_value()); + // Check total balance of stash + assert_eq!(Ring::total_balance(&11), 0); + + // Check storage items do not exist + assert!(!>::exists(&10)); + assert!(!>::exists(&11)); + assert!(!>::exists(&11)); + assert!(!>::exists(&11)); + assert!(!>::exists(&11)); + }); +} + +#[test] +fn switching_roles() { + // Test that it should be possible to switch between roles (nominator, validator, idle) with minimal overhead. + ExtBuilder::default().nominate(false).build().execute_with(|| { + Timestamp::set_timestamp(1); // Initialize time. + + // Reset reward destination + for i in &[10, 20] { + assert_ok!(Staking::set_payee(Origin::signed(*i), RewardDestination::Controller)); + } + + assert_eq_uvec!(validator_controllers(), vec![20, 10]); + + // put some money in account that we'll use. + for i in 1..7 { + let _ = Ring::deposit_creating(&i, 5000); + } + + // add 2 nominators + assert_ok!(Staking::bond( + Origin::signed(1), + 2, + StakingBalance::RingBalance(2000), + RewardDestination::Controller, + 0 + )); + assert_ok!(Staking::nominate(Origin::signed(2), vec![11, 5])); + + assert_ok!(Staking::bond( + Origin::signed(3), + 4, + StakingBalance::RingBalance(500), + RewardDestination::Controller, + 0 + )); + assert_ok!(Staking::nominate(Origin::signed(4), vec![21, 1])); + + // add a new validator candidate + assert_ok!(Staking::bond( + Origin::signed(5), + 6, + StakingBalance::RingBalance(1000), + RewardDestination::Controller, + 0 + )); + assert_ok!(Staking::validate(Origin::signed(6), ValidatorPrefs::default())); + + // new block + start_session(1); + + // no change + assert_eq_uvec!(validator_controllers(), vec![20, 10]); + + // new block + start_session(2); + + // no change + assert_eq_uvec!(validator_controllers(), vec![20, 10]); + + // new block --> ne era --> new validators + start_session(3); + + // with current nominators 10 and 5 have the most stake + assert_eq_uvec!(validator_controllers(), vec![6, 10]); + + // 2 decides to be a validator. Consequences: + assert_ok!(Staking::validate(Origin::signed(2), ValidatorPrefs::default())); + // new stakes: + // 10: 1000 self vote + // 20: 1000 self vote + 250 vote + // 6 : 1000 self vote + // 2 : 2000 self vote + 250 vote. + // Winners: 20 and 2 + + start_session(4); + assert_eq_uvec!(validator_controllers(), vec![6, 10]); + + start_session(5); + assert_eq_uvec!(validator_controllers(), vec![6, 10]); + + // ne era + start_session(6); + assert_eq_uvec!(validator_controllers(), vec![2, 20]); + + check_exposure_all(); + check_nominator_all(); + }); +} + +#[test] +fn wrong_vote_is_null() { + ExtBuilder::default() + .nominate(false) + .validator_pool(true) + .build() + .execute_with(|| { + assert_eq_uvec!(validator_controllers(), vec![40, 30]); + + // put some money in account that we'll use. + for i in 1..3 { + let _ = Ring::deposit_creating(&i, 5000); + } + + // add 1 nominators + assert_ok!(Staking::bond( + Origin::signed(1), + 2, + StakingBalance::RingBalance(2000), + RewardDestination::default(), + 0 + )); + assert_ok!(Staking::nominate( + Origin::signed(2), + vec![ + 11, 21, // good votes + 1, 2, 15, 1000, 25 // crap votes. No effect. + ] + )); + + // new block + start_era(1); + + assert_eq_uvec!(validator_controllers(), vec![20, 10]); + }); +} + +#[test] +fn bond_with_no_staked_value() { + // Behavior when someone bonds with no staked value. + // Particularly when she votes and the candidate is elected. + ExtBuilder::default() + .validator_count(3) + .existential_deposit(5) + .nominate(false) + .minimum_validator_count(1) + .build() + .execute_with(|| { + // Can't bond with 1 + assert_noop!( + Staking::bond( + Origin::signed(1), + 2, + StakingBalance::RingBalance(1), + RewardDestination::Controller, + 0 + ), + Error::::InsufficientValue, + ); + // bonded with absolute minimum value possible. + assert_ok!(Staking::bond( + Origin::signed(1), + 2, + StakingBalance::RingBalance(5), + RewardDestination::Controller, + 0 + )); + // Not yet removed. + assert!(Staking::ledger(2).is_some()); + match Ring::locks(&1)[0].withdraw_lock.clone() { + WithdrawLock::Normal(_) => panic!("lock type error"), + WithdrawLock::WithStaking(lock) => assert_eq!(lock.staking_amount, 5), + } + + assert_ok!(Staking::unbond(Origin::signed(2), StakingBalance::RingBalance(5))); + + // unbond all, auto remove the ledger + assert_eq!(Staking::ledger(2), None); + }); +} + +#[test] +fn bond_with_little_staked_value_bounded_by_slot_stake() { + // Behavior when someone bonds with little staked value. + // Particularly when she votes and the candidate is elected. + ExtBuilder::default() + .validator_count(3) + .nominate(false) + .minimum_validator_count(1) + .build() + .execute_with(|| { + // setup + assert_ok!(Staking::chill(Origin::signed(30))); + assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); + let init_balance_2 = Ring::free_balance(&2); + let init_balance_10 = Ring::free_balance(&10); + + // Stingy validator. + assert_ok!(Staking::bond( + Origin::signed(1), + 2, + StakingBalance::RingBalance(1), + RewardDestination::Controller, + 0 + )); + assert_ok!(Staking::validate(Origin::signed(2), ValidatorPrefs::default())); + + // TODO: checkout the payout + // let total_payout_0 = current_total_payout_for_duration(3000); + // assert!(total_payout_0 > 100); // Test is meaningfull if reward something + // reward_all_elected(); + // start_era(1); + + // // 2 is elected. + // // and fucks up the slot stake. + // assert_eq_uvec!(validator_controllers(), vec![20, 10, 2]); + // assert_eq!(Staking::slot_stake(), 1); + + // // Old ones are rewarded. + // assert_eq!(Ring::free_balance(&10), init_balance_10 + total_payout_0 / 3); + // // no rewards paid to 2. This was initial election. + // assert_eq!(Ring::free_balance(&2), init_balance_2); + + // let total_payout_1 = current_total_payout_for_duration(3000); + // assert!(total_payout_1 > 100); // Test is meaningfull if reward something + // reward_all_elected(); + // start_era(2); + + // assert_eq_uvec!(validator_controllers(), vec![20, 10, 2]); + // assert_eq!(Staking::slot_stake(), 1); + + // assert_eq!(Ring::free_balance(&2), init_balance_2 + total_payout_1 / 3); + // assert_eq!( + // Ring::free_balance(&10), + // init_balance_10 + total_payout_0 / 3 + total_payout_1 / 3, + // ); + // check_exposure_all(); + // check_nominator_all(); + }); +} + +#[cfg(feature = "equalize")] +#[test] +fn phragmen_linear_worse_case_equalize() { + ExtBuilder::default() + .nominate(false) + .validator_pool(true) + .fair(true) + .build() + .execute_with(|| { + bond_validator(50, StakingBalance::RingBalance(1000)); + bond_validator(60, StakingBalance::RingBalance(1000)); + bond_validator(70, StakingBalance::RingBalance(1000)); + + bond_nominator(2, StakingBalance::RingBalance(2000), vec![11]); + bond_nominator(4, StakingBalance::RingBalance(1000), vec![11, 21]); + bond_nominator(6, StakingBalance::RingBalance(1000), vec![21, 31]); + bond_nominator(8, StakingBalance::RingBalance(1000), vec![31, 41]); + bond_nominator(110, StakingBalance::RingBalance(1000), vec![41, 51]); + bond_nominator(120, StakingBalance::RingBalance(1000), vec![51, 61]); + bond_nominator(130, StakingBalance::RingBalance(1000), vec![61, 71]); + + for i in &[10, 20, 30, 40, 50, 60, 70] { + assert_ok!(Staking::set_payee(Origin::signed(*i), RewardDestination::Controller)); + } + + assert_eq_uvec!(validator_controllers(), vec![40, 30]); + assert_ok!(Staking::set_validator_count(Origin::ROOT, 7)); + + start_era(1); + + assert_eq_uvec!(validator_controllers(), vec![10, 60, 40, 20, 50, 30, 70]); + + // TODO: checkout on this + // assert_eq_error_rate!(Staking::stakers(11).total_ring_balance, 3000, 2); + // assert_eq_error_rate!(Staking::stakers(21).total_ring_balance, 2255, 2); + // assert_eq_error_rate!(Staking::stakers(31).total_ring_balance, 2255, 2); + // assert_eq_error_rate!(Staking::stakers(41).total_ring_balance, 1925, 2); + // assert_eq_error_rate!(Staking::stakers(51).total_ring_balance, 1870, 2); + // assert_eq_error_rate!(Staking::stakers(61).total_ring_balance, 1890, 2); + // assert_eq_error_rate!(Staking::stakers(71).total_ring_balance, 1800, 2); + + check_exposure_all(); + check_nominator_all(); + }) +} + +#[test] +fn new_era_elects_correct_number_of_validators() { + ExtBuilder::default() + .nominate(true) + .validator_pool(true) + .fair(true) + .validator_count(1) + .build() + .execute_with(|| { + assert_eq!(Staking::validator_count(), 1); + assert_eq!(validator_controllers().len(), 1); + + System::set_block_number(1); + Session::on_initialize(System::block_number()); + + assert_eq!(validator_controllers().len(), 1); + check_exposure_all(); + check_nominator_all(); + }) +} + +// TODO: overflow, the test case failed +// thread 'tests::phragmen_should_not_overflow_validators' panicked at 'attempt to add with overflow', // // #[test] -// fn unbond_zero() { -// ExtBuilder::default().build().execute_with(|| { -// gen_paired_account!(stash(123), controller(456), promise_month(12)); -// let ledger = Staking::ledger(controller).unwrap(); -// -// Timestamp::set_timestamp(promise_month * MONTH_IN_MILLISECONDS); -// assert_ok!(Staking::unbond(Origin::signed(10), StakingBalances::RingBalance(0))); -// assert_ok!(Staking::unbond(Origin::signed(10), StakingBalances::KtonBalance(0))); -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// }); +// fn phragmen_should_not_overflow_validators() { +// ExtBuilder::default().nominate(false).build().execute_with(|| { +// let _ = Staking::chill(Origin::signed(10)); +// let _ = Staking::chill(Origin::signed(20)); + +// bond_validator(2, StakingBalance::RingBalance(u64::max_value())); +// bond_validator(4, StakingBalance::RingBalance(u64::max_value())); + +// bond_nominator(6, StakingBalance::RingBalance(u64::max_value() / 2), vec![3, 5]); +// bond_nominator(8, StakingBalance::RingBalance(u64::max_value() / 2), vec![3, 5]); + +// start_era(1); + +// assert_eq_uvec!(validator_controllers(), vec![4, 2]); + +// // This test will fail this. Will saturate. +// // check_exposure_all(); +// assert_eq!(Staking::stakers(3).total_ring_balance, u64::max_value()); +// assert_eq!(Staking::stakers(5).total_ring_balance, u64::max_value()); +// }) // } -// -// // bond 10_000 Ring for 12 months, gain 1 Kton -// // bond extra 10_000 Ring for 36 months, gain 3 Kton -// // bond extra 1 Kton -// // nominate -// // unlock the 12 months deposit item with punish -// // lost 3 Kton and 10_000 Ring's power for nominate + +// TODO: overflow, should be checked +// thread 'tests::phragmen_should_not_overflow_nominators' panicked at 'attempt to add with overflow' // #[test] -// fn yakio_q1() { -// ExtBuilder::default().build().execute_with(|| { -// let (stash, controller) = (777, 888); -// let _ = Ring::deposit_creating(&stash, 20_000); -// -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::RingBalance(10_000), -// RewardDestination::Stash, -// 12, -// )); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::RingBalance(10_000), -// 36, -// )); -// assert_eq!(Kton::free_balance(&stash), 4); -// -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::KtonBalance(1), -// 36 -// )); -// assert_eq!(Staking::ledger(controller).unwrap().active_kton, 1); -// -// assert_ok!(Staking::nominate(Origin::signed(controller), vec![controller])); -// -// assert_ok!(Staking::try_claim_deposits_with_punish( -// Origin::signed(controller), -// 12 * MONTH_IN_MILLISECONDS, -// )); -// assert_eq!(Kton::free_balance(&stash), 1); -// -// let ledger = Staking::ledger(controller).unwrap(); -// // not enough Kton to unbond -// assert_ok!(Staking::try_claim_deposits_with_punish( -// Origin::signed(controller), -// 36 * MONTH_IN_MILLISECONDS, -// )); -// assert_eq!(Staking::ledger(controller).unwrap(), ledger); -// }); +// fn phragmen_should_not_overflow_nominators() { +// ExtBuilder::default().nominate(false).build().execute_with(|| { +// let _ = Staking::chill(Origin::signed(10)); +// let _ = Staking::chill(Origin::signed(20)); + +// bond_validator(2, StakingBalance::RingBalance(u64::max_value() / 2)); +// bond_validator(4, StakingBalance::RingBalance(u64::max_value() / 2)); + +// bond_nominator(6, StakingBalance::RingBalance(u64::max_value()), vec![3, 5]); +// bond_nominator(8, StakingBalance::RingBalance(u64::max_value()), vec![3, 5]); + +// start_era(1); + +// assert_eq_uvec!(validator_controllers(), vec![4, 2]); + +// // Saturate. +// assert_eq!(Staking::stakers(3).total_ring_balance, u64::max_value()); +// assert_eq!(Staking::stakers(5).total_ring_balance, u64::max_value()); +// }) // } -// -// // how to balance the power and calculate the reward if some validators have been chilled + +// TODO: check why overflow here +// thread 'tests::phragmen_should_not_overflow_ultimate' panicked at 'attempt to add with overflow', // #[test] -// fn yakio_q2() { -// fn run(with_new_era: bool) -> Balance { -// let mut balance = 0; -// ExtBuilder::default().build().execute_with(|| { -// gen_paired_account!(validator_1_stash(123), validator_1_controller(456), 0); -// gen_paired_account!(validator_2_stash(234), validator_2_controller(567), 0); -// gen_paired_account!(nominator_stash(345), nominator_controller(678), 0); -// -// assert_ok!(Staking::validate( -// Origin::signed(validator_1_controller), -// ValidatorPrefs { -// node_name: vec![0; 8], -// ..Default::default() -// }, -// )); -// assert_ok!(Staking::validate( -// Origin::signed(validator_2_controller), -// ValidatorPrefs { -// node_name: vec![1; 8], -// ..Default::default() -// }, -// )); -// assert_ok!(Staking::nominate( -// Origin::signed(nominator_controller), -// vec![validator_1_stash, validator_2_stash], -// )); -// -// start_era(1); -// assert_ok!(Staking::chill(Origin::signed(validator_1_controller))); -// // assert_ok!(Staking::chill(Origin::signed(validator_2_controller))); -// if with_new_era { -// start_era(2); -// } -// let _ = Staking::reward_validator(&validator_1_stash, 1000 * COIN); -// let _ = Staking::reward_validator(&validator_2_stash, 1000 * COIN); -// -// balance = Ring::free_balance(&nominator_stash); -// }); -// -// balance -// } -// -// let free_balance = run(false); -// let free_balance_with_new_era = run(true); -// -// assert_ne!(free_balance, 0); -// assert_ne!(free_balance_with_new_era, 0); -// assert!(free_balance > free_balance_with_new_era); +// fn phragmen_should_not_overflow_ultimate() { +// ExtBuilder::default().nominate(false).build().execute_with(|| { +// bond_validator(2, StakingBalance::RingBalance(u64::max_value())); +// bond_validator(4, StakingBalance::RingBalance(u64::max_value())); + +// bond_nominator(6, StakingBalance::RingBalance(u64::max_value()), vec![3, 5]); +// bond_nominator(8, StakingBalance::RingBalance(u64::max_value()), vec![3, 5]); + +// start_era(1); + +// assert_eq_uvec!(validator_controllers(), vec![4, 2]); + +// // Saturate. +// assert_eq!(Staking::stakers(3).total_ring_balance, u64::max_value()); +// assert_eq!(Staking::stakers(5).total_ring_balance, u64::max_value()); +// }) // } -// + +// TODO: check why overflow here +//thread 'tests::reward_validator_slashing_validator_doesnt_overflow' panicked at 'assertion failed: `(left == right)` +//left: `8589934590`, +//right: `17179869180`', frame/staking/src/tests.rs:1936:9 // #[test] -// fn xavier_q1() { +// fn reward_validator_slashing_validator_doesnt_overflow() { // ExtBuilder::default().build().execute_with(|| { -// let stash = 123; -// let controller = 456; -// let _ = Kton::deposit_creating(&stash, 10); -// -// Timestamp::set_timestamp(0); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::KtonBalance(5), -// RewardDestination::Stash, -// 0, -// )); -// assert_eq!(Timestamp::get(), 0); -// assert_eq!(Kton::free_balance(stash), 10); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 5, -// unbondings: vec![], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// // println!("Ok Init - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Ok Init - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// -// Timestamp::set_timestamp(1); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::KtonBalance(5), -// 0 -// )); -// assert_eq!(Timestamp::get(), 1); -// assert_eq!(Kton::free_balance(stash), 10); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 10, -// unbondings: vec![], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// // println!("Ok Bond Extra - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Ok Bond Extra - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// -// let unbond_start = 2; -// Timestamp::set_timestamp(unbond_start); -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::KtonBalance(9) -// )); -// assert_eq!(Timestamp::get(), 2); -// assert_eq!(Kton::free_balance(stash), 10); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![NormalLock { -// amount: 9, -// until: BondingDuration::get() + unbond_start, -// }], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// // println!("Ok Unbond - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Ok Unbond - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// -// assert_err!( -// Kton::transfer(Origin::signed(stash), controller, 1), -// "account liquidity restrictions prevent withdrawal", -// ); -// // println!("Locking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Locking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// -// Timestamp::set_timestamp(BondingDuration::get() + unbond_start); -// assert_ok!(Kton::transfer(Origin::signed(stash), controller, 1)); -// // println!("Unlocking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Unlocking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!( -// // "Unlocking Transfer - Kton StakingLedger: {:#?}", -// // Staking::ledger(controller) -// // ); -// // println!(); -// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start); -// assert_eq!(Kton::free_balance(stash), 9); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![NormalLock { -// amount: 9, -// until: BondingDuration::get() + unbond_start, -// }], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// -// let _ = Kton::deposit_creating(&stash, 20); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::KtonBalance(19), -// 0 -// )); -// assert_eq!(Kton::free_balance(stash), 29); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 20, -// unbondings: vec![NormalLock { -// amount: 9, -// until: BondingDuration::get() + unbond_start, -// }], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash: 123, -// active_ring: 0, -// active_deposit_ring: 0, -// active_kton: 20, -// deposit_items: vec![], -// ring_staking_lock: Default::default(), -// kton_staking_lock: StakingLock { -// staking_amount: 20, -// unbondings: vec![NormalLock { -// amount: 9, -// until: BondingDuration::get() + unbond_start, -// }], -// }, -// } +// let stake = u32::max_value() as u64 * 2; +// let reward_slash = u32::max_value() as u64 * 2; + +// // Assert multiplication overflows in balance arithmetic. +// assert!(stake.checked_mul(reward_slash).is_none()); + +// // Set staker +// let _ = Ring::make_free_balance_be(&11, stake); +// >::insert( +// &11, +// Exposure { +// own_ring_balance: stake, +// total_ring_balance: stake, +// own_kton_balance: 0, +// own_power: 0, +// total_power: 0, +// total_kton_balance: 0, +// others: vec![], +// }, // ); -// // println!("Unlocking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Unlocking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!( -// // "Unlocking Transfer - Kton StakingLedger: {:#?}", -// // Staking::ledger(controller) -// // ); -// // println!(); -// }); -// -// ExtBuilder::default().build().execute_with(|| { -// let stash = 123; -// let controller = 456; -// let _ = Ring::deposit_creating(&stash, 10); -// -// Timestamp::set_timestamp(0); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::RingBalance(5), -// RewardDestination::Stash, + +// // Check reward +// let _ = Staking::reward_validator(&11, reward_slash); +// assert_eq!(Ring::total_balance(&11), stake * 2); + +// // Set staker +// let _ = Ring::make_free_balance_be(&11, stake); +// let _ = Ring::make_free_balance_be(&2, stake); + +// // only slashes out of bonded stake are applied. without this line, +// // it is 0. +// Staking::bond( +// Origin::signed(2), +// 20000, +// StakingBalance::RingBalance(stake - 1), +// RewardDestination::default(), // 0, -// )); -// assert_eq!(Timestamp::get(), 0); -// assert_eq!(Ring::free_balance(stash), 10); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 5, -// unbondings: vec![], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// // println!("Ok Init - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Ok Init - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// -// Timestamp::set_timestamp(1); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::RingBalance(5), -// 0 -// )); -// assert_eq!(Timestamp::get(), 1); -// assert_eq!(Ring::free_balance(stash), 10); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 10, -// unbondings: vec![], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// // println!("Ok Bond Extra - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Ok Bond Extra - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// -// let unbond_start = 2; -// Timestamp::set_timestamp(unbond_start); -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::RingBalance(9) -// )); -// assert_eq!(Timestamp::get(), 2); -// assert_eq!(Ring::free_balance(stash), 10); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![NormalLock { -// amount: 9, -// until: BondingDuration::get() + unbond_start, -// }], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// // println!("Ok Unbond - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Ok Unbond - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// -// assert_err!( -// Ring::transfer(Origin::signed(stash), controller, 1), -// "account liquidity restrictions prevent withdrawal", -// ); -// // println!("Locking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Locking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// -// Timestamp::set_timestamp(BondingDuration::get() + unbond_start); -// assert_ok!(Ring::transfer(Origin::signed(stash), controller, 1)); -// // println!("Unlocking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Unlocking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!( -// // "Unlocking Transfer - Ring StakingLedger: {:#?}", -// // Staking::ledger(controller) -// // ); -// // println!(); -// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start); -// assert_eq!(Ring::free_balance(stash), 9); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![NormalLock { -// amount: 9, -// until: BondingDuration::get() + unbond_start, -// }], -// }), -// reasons: WithdrawReasons::all(), -// }] +// ) +// .unwrap(); +// >::insert( +// &11, +// Exposure { +// own_ring_balance: stake, +// total_ring_balance: stake, +// own_kton_balance: 0, +// own_power: 0, +// total_power: 0, +// total_kton_balance: 0, +// others: vec![IndividualExposure { +// who: 2, +// ring_balance: stake - 1, +// kton_balance: 0, +// power: 0, +// }], +// }, // ); -// -// let _ = Ring::deposit_creating(&stash, 20); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::RingBalance(19), -// 0 -// )); -// assert_eq!(Ring::free_balance(stash), 29); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 20, -// unbondings: vec![NormalLock { -// amount: 9, -// until: BondingDuration::get() + unbond_start, -// }], -// }), -// reasons: WithdrawReasons::all(), -// }] + +// // Check slashing +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(100)], // ); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash: 123, -// active_ring: 20, -// active_deposit_ring: 0, -// active_kton: 0, -// deposit_items: vec![], -// ring_staking_lock: StakingLock { -// staking_amount: 20, -// unbondings: vec![NormalLock { -// amount: 9, -// until: BondingDuration::get() + unbond_start, -// }], -// }, -// kton_staking_lock: Default::default(), -// } + +// assert_eq!(Ring::total_balance(&11), stake - 1); +// assert_eq!(Ring::total_balance(&2), 1); +// }) +// } + +#[test] +fn reward_from_authorship_event_handler_works() { + ExtBuilder::default().build().execute_with(|| { + use pallet_authorship::EventHandler; + + assert_eq!(>::author(), 11); + + >::note_author(11); + >::note_uncle(21, 1); + // An uncle author that is not currently elected doesn't get rewards, + // but the block producer does get reward for referencing it. + >::note_uncle(31, 1); + // Rewarding the same two times works. + >::note_uncle(11, 1); + + // Not mandatory but must be coherent with rewards + assert_eq!(>::get(), vec![21, 11]); + + // 21 is rewarded as an uncle producer + // 11 is rewarded as a block producer and uncle referencer and uncle producer + assert_eq!(CurrentEraPointsEarned::get().individual, vec![1, 20 + 2 * 3 + 1]); + assert_eq!(CurrentEraPointsEarned::get().total, 28); + }) +} + +#[test] +fn add_reward_points_fns_works() { + ExtBuilder::default().build().execute_with(|| { + let validators = >::current_elected(); + // Not mandatory but must be coherent with rewards + assert_eq!(validators, vec![21, 11]); + + >::reward_by_indices(vec![(0, 1), (1, 1), (2, 1), (1, 1)]); + + >::reward_by_ids(vec![(21, 1), (11, 1), (31, 1), (11, 1)]); + + assert_eq!(CurrentEraPointsEarned::get().individual, vec![2, 4]); + assert_eq!(CurrentEraPointsEarned::get().total, 6); + }) +} + +// TODO: mock slashable_balance_of +// #[test] +// fn unbonded_balance_is_not_slashable() { +// ExtBuilder::default().build().execute_with(|| { +// // total amount staked is slashable. +// assert_eq!(Staking::slashable_balance_of(&11), 1000); + +// assert_ok!(Staking::unbond(Origin::signed(10), 800)); + +// // only the active portion. +// assert_eq!(Staking::slashable_balance_of(&11), 200); +// }) +// } + +#[test] +fn era_is_always_same_length() { + // This ensures that the sessions is always of the same length if there is no forcing no + // session changes. + ExtBuilder::default().build().execute_with(|| { + start_era(1); + assert_eq!(Staking::current_era_start_session_index(), SessionsPerEra::get()); + + start_era(2); + assert_eq!(Staking::current_era_start_session_index(), SessionsPerEra::get() * 2); + + let session = Session::current_index(); + ForceEra::put(Forcing::ForceNew); + advance_session(); + assert_eq!(Staking::current_era(), 3); + assert_eq!(Staking::current_era_start_session_index(), session + 1); + + start_era(4); + assert_eq!( + Staking::current_era_start_session_index(), + session + SessionsPerEra::get() + 1 + ); + }); +} + +// TODO: checkout overflow +// 16: darwinia_staking::StakingLedger::slash::{{closure}} +// at frame/staking/src/lib.rs:493 +// 17: darwinia_staking::StakingLedger::slash +// at frame/staking/src/lib.rs:534 +// 18: darwinia_staking::slashing::do_slash +// at frame/staking/src/slashing.rs:684 +// 19: darwinia_staking::slashing::apply_slash +// #[test] +// fn offence_forces_new_era() { +// ExtBuilder::default().build().execute_with(|| { +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(5)], // ); -// // println!("Unlocking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Unlocking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!( -// // "Unlocking Transfer - Ring StakingLedger: {:#?}", -// // Staking::ledger(controller) -// // ); -// // println!(); + +// assert_eq!(Staking::force_era(), Forcing::ForceNew); // }); // } -// + +// TODO: same issue as above case +// 16: darwinia_staking::StakingLedger::slash::{{closure}} +// at frame/staking/src/lib.rs:493 +// 17: darwinia_staking::StakingLedger::slash +// at frame/staking/src/lib.rs:534 +// 18: darwinia_staking::slashing::do_slash +// at frame/staking/src/slashing.rs:684 +// 19: darwinia_staking::slashing::apply_slash +// at frame/staking/src/slashing.rs:732 // #[test] -// fn xavier_q2() { +// fn offence_ensures_new_era_without_clobbering() { // ExtBuilder::default().build().execute_with(|| { -// let stash = 123; -// let controller = 456; -// let _ = Kton::deposit_creating(&stash, 10); -// -// Timestamp::set_timestamp(1); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::KtonBalance(5), -// RewardDestination::Stash, -// 0, -// )); -// assert_eq!(Kton::free_balance(stash), 10); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 5, -// unbondings: vec![], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// // println!("Ok Init - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Ok Init - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// -// Timestamp::set_timestamp(1); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::KtonBalance(4), -// 0 -// )); -// assert_eq!(Timestamp::get(), 1); -// assert_eq!(Kton::free_balance(stash), 10); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 9, -// unbondings: vec![], -// }), -// reasons: WithdrawReasons::all(), -// }] +// assert_ok!(Staking::force_new_era_always(Origin::ROOT)); + +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(5)], // ); -// // println!("Ok Bond Extra - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Ok Bond Extra - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// -// let (unbond_start_1, unbond_value_1) = (2, 2); -// Timestamp::set_timestamp(unbond_start_1); -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::KtonBalance(unbond_value_1), -// )); -// assert_eq!(Timestamp::get(), unbond_start_1); -// assert_eq!(Kton::free_balance(stash), 10); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 7, -// unbondings: vec![NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, -// }], -// }), -// reasons: WithdrawReasons::all(), -// }] + +// assert_eq!(Staking::force_era(), Forcing::ForceAlways); +// }); +// } + +#[test] +fn offence_deselects_validator_when_slash_is_zero() { + ExtBuilder::default().build().execute_with(|| { + assert!(>::exists(11)); + on_offence_now( + &[OffenceDetails { + offender: (11, Staking::stakers(&11)), + reporters: vec![], + }], + &[Perbill::from_percent(0)], + ); + assert_eq!(Staking::force_era(), Forcing::ForceNew); + assert!(!>::exists(11)); + }); +} + +// #[test] +// fn slashing_performed_according_exposure() { +// // This test checks that slashing is performed according the exposure (or more precisely, +// // historical exposure), not the current balance. +// ExtBuilder::default().build().execute_with(|| { +// assert_eq!(Staking::stakers(&11).own_ring_balance, 1000); + +// // Handle an offence with a historical exposure. +// on_offence_now( +// &[OffenceDetails { +// offender: ( +// 11, +// Exposure { +// total_ring_balance: 500, +// total_kton_balance: 0, +// total_power: 0, +// own_ring_balance: 500, +// own_kton_balance: 0, +// own_power: 0, +// others: vec![], +// }, +// ), +// reporters: vec![], +// }], +// &[Perbill::from_percent(50)], // ); -// // println!("Ok Unbond - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Ok Unbond - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// -// let (unbond_start_2, unbond_value_2) = (3, 6); -// Timestamp::set_timestamp(unbond_start_2); -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::KtonBalance(6) -// )); -// assert_eq!(Timestamp::get(), unbond_start_2); -// assert_eq!(Kton::free_balance(stash), 10); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![ -// NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, -// }, -// NormalLock { -// amount: 6, -// until: BondingDuration::get() + unbond_start_2, -// } -// ], -// }), -// reasons: WithdrawReasons::all(), -// }] + +// // TODO: the slash may not correct +// // The stash account should be slashed for 250 (50% of 500). +// assert_eq!(Ring::free_balance(&11), 1000 - 250); +// }); +// } + +// TODO: check on this +// thread 'tests::slash_in_old_span_does_not_deselect' panicked at 'cannot slash in era 1', frame/staking/src/mock.rs:551:9 +// #[test] +// fn slash_in_old_span_does_not_deselect() { +// ExtBuilder::default().build().execute_with(|| { +// start_era(1); + +// assert!(>::exists(11)); +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(0)], // ); -// // println!("Ok Unbond - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Ok Unbond - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// -// assert_err!( -// Kton::transfer(Origin::signed(stash), controller, unbond_value_1), -// "account liquidity restrictions prevent withdrawal", +// assert_eq!(Staking::force_era(), Forcing::ForceNew); +// assert!(!>::exists(11)); + +// start_era(2); + +// Staking::validate(Origin::signed(10), Default::default()).unwrap(); +// assert_eq!(Staking::force_era(), Forcing::NotForcing); +// assert!(>::exists(11)); + +// start_era(3); + +// // this staker is in a new slashing span now, having re-registered after +// // their prior slash. + +// on_offence_in_era( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(0)], +// 1, // ); -// // println!("Locking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Locking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// -// assert_ok!(Kton::transfer(Origin::signed(stash), controller, unbond_value_1 - 1)); -// assert_eq!(Kton::free_balance(stash), 9); -// // println!("Normal Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Normal Transfer - Kton Locks: {:#?}", Kton::locks(stash)); -// -// Timestamp::set_timestamp(BondingDuration::get() + unbond_start_1); -// assert_err!( -// Kton::transfer(Origin::signed(stash), controller, unbond_value_1 + 1), -// "account liquidity restrictions prevent withdrawal", + +// // not for zero-slash. +// assert_eq!(Staking::force_era(), Forcing::NotForcing); +// assert!(>::exists(11)); + +// on_offence_in_era( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(100)], +// 1, // ); -// // println!("Locking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Locking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); -// // println!(); -// assert_ok!(Kton::transfer(Origin::signed(stash), controller, unbond_value_1)); -// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start_1); -// assert_eq!(Kton::free_balance(stash), 7); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![ -// NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, -// }, -// NormalLock { -// amount: 6, -// until: BondingDuration::get() + unbond_start_2, -// } -// ], -// }), -// reasons: WithdrawReasons::all(), -// }] + +// // or non-zero. +// assert_eq!(Staking::force_era(), Forcing::NotForcing); +// assert!(>::exists(11)); +// assert_ledger_consistent(11); +// }); +// } + +// #[test] +// fn reporters_receive_their_slice() { +// // This test verifies that the reporters of the offence receive their slice from the slashed +// // amount. +// ExtBuilder::default().build().execute_with(|| { +// // The reporters' reward is calculated from the total exposure. +// #[cfg(feature = "equalize")] +// let initial_balance = 1250; +// #[cfg(not(feature = "equalize"))] +// let initial_balance = 1125; + +// // TODO: fail here +// assert_eq!(Staking::stakers(&11).total_ring_balance, initial_balance); + +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![1, 2], +// }], +// &[Perbill::from_percent(50)], // ); -// // println!("Unlocking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Unlocking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); -// -// Timestamp::set_timestamp(BondingDuration::get() + unbond_start_2); -// assert_ok!(Kton::transfer(Origin::signed(stash), controller, unbond_value_2)); -// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start_2); -// assert_eq!(Kton::free_balance(stash), 1); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![ -// NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, -// }, -// NormalLock { -// amount: 6, -// until: BondingDuration::get() + unbond_start_2, -// } -// ], -// }), -// reasons: WithdrawReasons::all(), -// }] + +// // F1 * (reward_proportion * slash - 0) +// // 50% * (10% * initial_balance / 2) +// let reward = (initial_balance / 20) / 2; +// let reward_each = reward / 2; // split into two pieces. +// assert_eq!(Ring::free_balance(&1), 10 + reward_each); +// assert_eq!(Ring::free_balance(&2), 20 + reward_each); +// assert_ledger_consistent(11); +// }); +// } + +// #[test] +// fn subsequent_reports_in_same_span_pay_out_less() { +// // This test verifies that the reporters of the offence receive their slice from the slashed +// // amount. +// ExtBuilder::default().build().execute_with(|| { +// // The reporters' reward is calculated from the total exposure. +// #[cfg(feature = "equalize")] +// let initial_balance = 1250; +// #[cfg(not(feature = "equalize"))] +// let initial_balance = 1125; + +// // TODO: fail here +// assert_eq!(Staking::stakers(&11).total_ring_balance, initial_balance); + +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![1], +// }], +// &[Perbill::from_percent(20)], // ); -// // println!("Unlocking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); -// // println!("Unlocking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); -// -// let _ = Kton::deposit_creating(&stash, 1); -// // println!("Staking Ledger: {:#?}", Staking::ledger(controller).unwrap()); -// assert_eq!(Kton::free_balance(stash), 2); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::KtonBalance(1), -// 0 -// )); -// assert_eq!( -// Kton::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 2, -// unbondings: vec![ -// NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, -// }, -// NormalLock { -// amount: 6, -// until: BondingDuration::get() + unbond_start_2, -// } -// ], -// }), -// reasons: WithdrawReasons::all(), -// }] + +// // F1 * (reward_proportion * slash - 0) +// // 50% * (10% * initial_balance * 20%) +// let reward = (initial_balance / 5) / 20; +// assert_eq!(Ring::free_balance(&1), 10 + reward); + +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![1], +// }], +// &[Perbill::from_percent(50)], // ); + +// let prior_payout = reward; + +// // F1 * (reward_proportion * slash - prior_payout) +// // 50% * (10% * (initial_balance / 2) - prior_payout) +// let reward = ((initial_balance / 20) - prior_payout) / 2; +// assert_eq!(Ring::free_balance(&1), 10 + prior_payout + reward); +// assert_ledger_consistent(11); // }); -// +// } + +// rm `invulnerables_are_not_slashed` testcase +// because `slashable_balance_of` is not used + +#[test] +fn dont_slash_if_fraction_is_zero() { + // Don't slash if the fraction is zero. + ExtBuilder::default().build().execute_with(|| { + assert_eq!(Ring::free_balance(&11), 1000); + + on_offence_now( + &[OffenceDetails { + offender: (11, Staking::stakers(&11)), + reporters: vec![], + }], + &[Perbill::from_percent(0)], + ); + + // The validator hasn't been slashed. The new era is not forced. + assert_eq!(Ring::free_balance(&11), 1000); + assert_ledger_consistent(11); + }); +} + +// TODO: checkout slash over flow issues +// #[test] +// fn only_slash_for_max_in_era() { // ExtBuilder::default().build().execute_with(|| { -// let stash = 123; -// let controller = 456; -// let _ = Ring::deposit_creating(&stash, 10); -// -// Timestamp::set_timestamp(1); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::RingBalance(5), -// RewardDestination::Stash, -// 0, -// )); -// assert_eq!(Ring::free_balance(stash), 10); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 5, -// unbondings: vec![], -// }), -// reasons: WithdrawReasons::all(), -// }] -// ); -// // println!("Ok Init - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Ok Init - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// -// Timestamp::set_timestamp(1); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::RingBalance(4), -// 0 -// )); -// assert_eq!(Timestamp::get(), 1); -// assert_eq!(Ring::free_balance(stash), 10); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 9, -// unbondings: vec![], -// }), -// reasons: WithdrawReasons::all(), -// }] +// assert_eq!(Ring::free_balance(&11), 1000); + +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(50)], // ); -// // println!("Ok Bond Extra - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Ok Bond Extra - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// -// let (unbond_start_1, unbond_value_1) = (2, 2); -// Timestamp::set_timestamp(unbond_start_1); -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::RingBalance(unbond_value_1) -// )); -// assert_eq!(Timestamp::get(), unbond_start_1); -// assert_eq!(Ring::free_balance(stash), 10); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 7, -// unbondings: vec![NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, -// }], -// }), -// reasons: WithdrawReasons::all(), -// }] + +// // The validator has been slashed and has been force-chilled. +// assert_eq!(Ring::free_balance(&11), 500); +// assert_eq!(Staking::force_era(), Forcing::ForceNew); + +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(25)], // ); -// // println!("Ok Unbond - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Ok Unbond - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// -// let (unbond_start_2, unbond_value_2) = (3, 6); -// Timestamp::set_timestamp(unbond_start_2); -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::RingBalance(6) -// )); -// assert_eq!(Timestamp::get(), unbond_start_2); -// assert_eq!(Ring::free_balance(stash), 10); -// assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![ -// NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, -// }, -// NormalLock { -// amount: 6, -// until: BondingDuration::get() + unbond_start_2, -// } -// ], -// }), -// reasons: WithdrawReasons::all(), -// }] + +// // The validator has not been slashed additionally. +// assert_eq!(Ring::free_balance(&11), 500); + +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(60)], // ); -// // println!("Ok Unbond - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Ok Unbond - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// -// assert_err!( -// Ring::transfer(Origin::signed(stash), controller, unbond_value_1), -// "account liquidity restrictions prevent withdrawal", + +// // The validator got slashed 10% more. +// assert_eq!(Ring::free_balance(&11), 400); +// assert_ledger_consistent(11); +// }) +// } + +#[test] +fn garbage_collection_after_slashing() { + ExtBuilder::default().existential_deposit(1).build().execute_with(|| { + assert_eq!(Ring::free_balance(&11), 256_000); + + on_offence_now( + &[OffenceDetails { + offender: (11, Staking::stakers(&11)), + reporters: vec![], + }], + &[Perbill::from_percent(10)], + ); + + assert_eq!(Ring::free_balance(&11), 256_000 - 25_600); + assert!(::SlashingSpans::get(&11).is_some()); + assert_eq!( + ::SpanSlash::get(&(11, 0)).amount_slashed().r, + 25_600 + ); + + on_offence_now( + &[OffenceDetails { + offender: (11, Staking::stakers(&11)), + reporters: vec![], + }], + &[Perbill::from_percent(100)], + ); + + // validator and nominator slash in era are garbage-collected by era change, + // so we don't test those here. + + assert_eq!(Ring::free_balance(&11), 0); + assert!(::SlashingSpans::get(&11).is_none()); + assert_eq!( + ::SpanSlash::get(&(11, 0)).amount_slashed().r, + 0 + ); + }) +} + +// TODO: check overflow in slash +// #[test] +// fn garbage_collection_on_window_pruning() { +// ExtBuilder::default().build().execute_with(|| { +// start_era(1); + +// assert_eq!(Ring::free_balance(&11), 1000); + +// let exposure = Staking::stakers(&11); +// assert_eq!(Ring::free_balance(&101), 2000); +// let nominated_value = exposure.others.iter().find(|o| o.who == 101).unwrap().ring_balance; + +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(10)], // ); -// // println!("Locking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Locking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// -// assert_ok!(Ring::transfer(Origin::signed(stash), controller, unbond_value_1 - 1)); -// assert_eq!(Ring::free_balance(stash), 9); -// // println!("Normal Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Normal Transfer - Ring Locks: {:#?}", Ring::locks(stash)); -// -// Timestamp::set_timestamp(BondingDuration::get() + unbond_start_1); -// assert_err!( -// Ring::transfer(Origin::signed(stash), controller, unbond_value_1 + 1), -// "account liquidity restrictions prevent withdrawal", + +// let now = Staking::current_era(); + +// assert_eq!(Ring::free_balance(&11), 900); +// assert_eq!(Ring::free_balance(&101), 2000 - (nominated_value / 10)); + +// assert!(::ValidatorSlashInEra::get(&now, &11).is_some()); +// assert!(::NominatorSlashInEra::get(&now, &101).is_some()); + +// // + 1 because we have to exit the bonding window. +// for era in (0..(BondingDurationInEra::get() + 1)).map(|offset| offset + now + 1) { +// assert!(::ValidatorSlashInEra::get(&now, &11).is_some()); +// assert!(::NominatorSlashInEra::get(&now, &101).is_some()); + +// start_era(era); +// } + +// assert!(::ValidatorSlashInEra::get(&now, &11).is_none()); +// assert!(::NominatorSlashInEra::get(&now, &101).is_none()); +// }) +// } + +// rm `slashing_nominators_by_span_max` testcase +// because `slashable_balance_of` is not used + +// rm `slashes_are_summed_across_spans` testcase +// because `slashable_balance_of` is not used + +// TODO: check slash overflow +// #[test] +// fn deferred_slashes_are_deferred() { +// ExtBuilder::default().slash_defer_duration(2).build().execute_with(|| { +// start_era(1); + +// assert_eq!(Ring::free_balance(&11), 1000); + +// let exposure = Staking::stakers(&11); +// assert_eq!(Ring::free_balance(&101), 2000); +// let nominated_value = exposure.others.iter().find(|o| o.who == 101).unwrap().ring_balance; + +// on_offence_now( +// &[OffenceDetails { +// offender: (11, Staking::stakers(&11)), +// reporters: vec![], +// }], +// &[Perbill::from_percent(10)], // ); -// // println!("Locking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Locking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); -// // println!(); -// assert_ok!(Ring::transfer(Origin::signed(stash), controller, unbond_value_1)); -// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start_1); -// assert_eq!(Ring::free_balance(stash), 7); + +// assert_eq!(Ring::free_balance(&11), 1000); +// assert_eq!(Ring::free_balance(&101), 2000); + +// start_era(2); + +// assert_eq!(Ring::free_balance(&11), 1000); +// assert_eq!(Ring::free_balance(&101), 2000); + +// start_era(3); + +// assert_eq!(Ring::free_balance(&11), 1000); +// assert_eq!(Ring::free_balance(&101), 2000); + +// // at the start of era 4, slashes from era 1 are processed, +// // after being deferred for at least 2 full eras. +// start_era(4); + +// assert_eq!(Ring::free_balance(&11), 900); +// assert_eq!(Ring::free_balance(&101), 2000 - (nominated_value / 10)); +// }) +// } + +#[test] +fn remove_deferred() { + ExtBuilder::default().slash_defer_duration(2).build().execute_with(|| { + start_era(1); + + assert_eq!(Ring::free_balance(&11), 1000); + + let exposure = Staking::stakers(&11); + assert_eq!(Ring::free_balance(&101), 2000); + let nominated_value = exposure.others.iter().find(|o| o.who == 101).unwrap().ring_balance; + + on_offence_now( + &[OffenceDetails { + offender: (11, exposure.clone()), + reporters: vec![], + }], + &[Perbill::from_percent(10)], + ); + + assert_eq!(Ring::free_balance(&11), 1000); + assert_eq!(Ring::free_balance(&101), 2000); + + start_era(2); + + // TODO: checkout why slash in era 1 + // on_offence_in_era( + // &[OffenceDetails { + // offender: (11, exposure.clone()), + // reporters: vec![], + // }], + // &[Perbill::from_percent(15)], + // 1, + // ); + + // Staking::cancel_deferred_slash(Origin::ROOT, 1, vec![0]).unwrap(); + + // assert_eq!(Ring::free_balance(&11), 1000); + // assert_eq!(Ring::free_balance(&101), 2000); + + // start_era(3); + + // assert_eq!(Ring::free_balance(&11), 1000); + // assert_eq!(Ring::free_balance(&101), 2000); + + // // at the start of era 4, slashes from era 1 are processed, + // // after being deferred for at least 2 full eras. + // start_era(4); + + // // the first slash for 10% was cancelled, so no effect. + // assert_eq!(Ring::free_balance(&11), 1000); + // assert_eq!(Ring::free_balance(&101), 2000); + + // start_era(5); + + // let slash_10 = Perbill::from_percent(10); + // let slash_15 = Perbill::from_percent(15); + // let initial_slash = slash_10 * nominated_value; + + // let total_slash = slash_15 * nominated_value; + // let actual_slash = total_slash - initial_slash; + + // // 5% slash (15 - 10) processed now. + // assert_eq!(Ring::free_balance(&11), 950); + // assert_eq!(Ring::free_balance(&101), 2000 - actual_slash); + }) +} + +#[test] +fn remove_multi_deferred() { + ExtBuilder::default().slash_defer_duration(2).build().execute_with(|| { + start_era(1); + + assert_eq!(Ring::free_balance(&11), 1000); + + let exposure = Staking::stakers(&11); + assert_eq!(Ring::free_balance(&101), 2000); + + on_offence_now( + &[OffenceDetails { + offender: (11, exposure.clone()), + reporters: vec![], + }], + &[Perbill::from_percent(10)], + ); + + on_offence_now( + &[OffenceDetails { + offender: (21, Staking::stakers(&21)), + reporters: vec![], + }], + &[Perbill::from_percent(10)], + ); + + on_offence_now( + &[OffenceDetails { + offender: (11, exposure.clone()), + reporters: vec![], + }], + &[Perbill::from_percent(25)], + ); + + assert_eq!(::UnappliedSlashes::get(&1).len(), 3); + Staking::cancel_deferred_slash(Origin::ROOT, 1, vec![0, 2]).unwrap(); + + let slashes = ::UnappliedSlashes::get(&1); + assert_eq!(slashes.len(), 1); + assert_eq!(slashes[0].validator, 21); + }) +} + +#[test] +fn version_initialized() { + ExtBuilder::default().build().execute_with(|| { + assert_eq!( + ::StorageVersion::get(), + crate::migration::CURRENT_VERSION + ); + }); +} + +#[test] +fn bond_zero_should_work() { + ExtBuilder::default().build().execute_with(|| { + let (stash, controller) = (123, 456); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::RingBalance(0), + RewardDestination::Stash, + 0, + )); + + let (stash, controller) = (234, 567); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::KtonBalance(0), + RewardDestination::Stash, + 0, + )); + }); +} + +#[test] +fn normal_kton_should_work() { + ExtBuilder::default().build().execute_with(|| { + { + let (stash, controller) = (1001, 1000); + + let _ = Kton::deposit_creating(&stash, 10 * COIN); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::KtonBalance(10 * COIN), + RewardDestination::Stash, + 0, + )); + assert_eq!( + Staking::ledger(controller).unwrap(), + StakingLedger { + stash, + active_ring: 0, + active_deposit_ring: 0, + active_kton: 10 * COIN, + deposit_items: vec![], + ring_staking_lock: Default::default(), + kton_staking_lock: StakingLock { + staking_amount: 10 * COIN, + unbondings: vec![], + }, + } + ); + assert_eq!( + Kton::locks(&stash), + vec![BalanceLock { + id: STAKING_ID, + withdraw_lock: WithdrawLock::WithStaking(StakingLock { + staking_amount: 10 * COIN, + unbondings: vec![], + }), + reasons: WithdrawReasons::all(), + }] + ); + } + + { + let (stash, controller) = (2001, 2000); + + // promise_month should not work for kton + let _ = Kton::deposit_creating(&stash, 10 * COIN); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::KtonBalance(10 * COIN), + RewardDestination::Stash, + 12, + )); + assert_eq!( + Staking::ledger(controller).unwrap(), + StakingLedger { + stash, + active_ring: 0, + active_deposit_ring: 0, + active_kton: 10 * COIN, + deposit_items: vec![], + ring_staking_lock: Default::default(), + kton_staking_lock: StakingLock { + staking_amount: 10 * COIN, + unbondings: vec![], + }, + } + ); + } + }); +} + +// TODO: checkout BondingDuration not correct +// Im not sure to use BondingDurationInBlockNumber or BondingDurationInEra +// #[test] +// fn time_deposit_ring_unbond_and_withdraw_automatically_should_work() { +// ExtBuilder::default().build().execute_with(|| { +// let (stash, controller) = (11, 10); + +// let unbond_value = 10; +// assert_ok!(Staking::unbond( +// Origin::signed(controller), +// StakingRing::RingBalance(unbond_value), +// )); // assert_eq!( // Ring::locks(stash), // vec![BalanceLock { // id: STAKING_ID, // withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![ -// NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, -// }, -// NormalLock { -// amount: 6, -// until: BondingDuration::get() + unbond_start_2, -// } -// ], +// staking_amount: 1000 - unbond_value, +// unbondings: vec![NormalLock { +// amount: unbond_value, +// until: BondingDuration::get(), +// }], // }), // reasons: WithdrawReasons::all(), -// }] +// }], // ); -// // println!("Unlocking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Unlocking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); -// -// Timestamp::set_timestamp(BondingDuration::get() + unbond_start_2); -// assert_ok!(Ring::transfer(Origin::signed(stash), controller, unbond_value_2)); -// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start_2); -// assert_eq!(Ring::free_balance(stash), 1); // assert_eq!( -// Ring::locks(stash), -// vec![BalanceLock { -// id: STAKING_ID, -// withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 1, -// unbondings: vec![ -// NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, -// }, -// NormalLock { -// amount: 6, -// until: BondingDuration::get() + unbond_start_2, -// } -// ], -// }), -// reasons: WithdrawReasons::all(), -// }] +// Staking::ledger(controller).unwrap(), +// StakingLedger { +// stash, +// active_ring: 1000 - unbond_value, +// active_deposit_ring: 0, +// active_kton: 0, +// deposit_items: vec![], +// ring_staking_lock: StakingLock { +// staking_amount: 1000 - unbond_value, +// unbondings: vec![NormalLock { +// amount: unbond_value, +// until: BondingDuration::get(), +// }], +// }, +// kton_staking_lock: Default::default(), +// }, // ); -// // println!("Unlocking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); -// // println!("Unlocking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); -// -// let _ = Ring::deposit_creating(&stash, 1); -// // println!("Staking Ledger: {:#?}", Staking::ledger(controller).unwrap()); -// assert_eq!(Ring::free_balance(stash), 2); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::RingBalance(1), -// 0 + +// let unbond_start = 30; +// Timestamp::set_timestamp(unbond_start); +// assert_ok!(Staking::unbond( +// Origin::signed(controller), +// StakingRing::RingBalance(COIN) // )); + // assert_eq!( // Ring::locks(stash), // vec![BalanceLock { // id: STAKING_ID, // withdraw_lock: WithdrawLock::WithStaking(StakingLock { -// staking_amount: 2, +// staking_amount: 0, // unbondings: vec![ // NormalLock { -// amount: 2, -// until: BondingDuration::get() + unbond_start_1, +// amount: unbond_value, +// until: BondingDuration::get(), // }, // NormalLock { -// amount: 6, -// until: BondingDuration::get() + unbond_start_2, -// } +// amount: 1000 - unbond_value, +// until: unbond_start + BondingDuration::get(), +// }, // ], // }), // reasons: WithdrawReasons::all(), -// }] -// ); -// }); -// } -// -// #[test] -// fn xavier_q3() { -// ExtBuilder::default().build().execute_with(|| { -// let stash = 123; -// let controller = 456; -// let _ = Kton::deposit_creating(&stash, 10); -// -// Timestamp::set_timestamp(1); -// assert_ok!(Staking::bond( -// Origin::signed(stash), -// controller, -// StakingBalances::KtonBalance(5), -// RewardDestination::Stash, -// 0, -// )); -// assert_eq!(Timestamp::get(), 1); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash: 123, -// active_ring: 0, -// active_deposit_ring: 0, -// active_kton: 5, -// deposit_items: vec![], -// ring_staking_lock: Default::default(), -// kton_staking_lock: StakingLock { -// staking_amount: 5, -// unbondings: vec![], -// }, -// } +// }], // ); -// // println!("Locks: {:#?}", Kton::locks(stash)); -// // println!("StakingLedger: {:#?}", Staking::ledger(controller)); -// // println!(); -// -// assert_ok!(Staking::unbond( -// Origin::signed(controller), -// StakingBalances::KtonBalance(5) -// )); // assert_eq!( // Staking::ledger(controller).unwrap(), // StakingLedger { -// stash: 123, +// stash, // active_ring: 0, // active_deposit_ring: 0, // active_kton: 0, // deposit_items: vec![], -// ring_staking_lock: Default::default(), -// kton_staking_lock: StakingLock { +// ring_staking_lock: StakingLock { // staking_amount: 0, -// unbondings: vec![NormalLock { amount: 5, until: 61 }], +// unbondings: vec![ +// NormalLock { +// amount: unbond_value, +// until: BondingDuration::get(), +// }, +// NormalLock { +// amount: 1000 - unbond_value, +// until: unbond_start + BondingDuration::get(), +// }, +// ], // }, -// } +// kton_staking_lock: Default::default(), +// }, // ); -// // println!("Locks: {:#?}", Kton::locks(stash)); -// // println!("StakingLedger: {:#?}", Staking::ledger(controller)); -// // println!(); -// -// Timestamp::set_timestamp(61); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::KtonBalance(1), -// 0 -// )); -// assert_eq!(Timestamp::get(), 61); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash: 123, -// active_ring: 0, -// active_deposit_ring: 0, -// active_kton: 1, -// deposit_items: vec![], -// ring_staking_lock: Default::default(), -// kton_staking_lock: StakingLock { -// staking_amount: 1, -// unbondings: vec![NormalLock { amount: 5, until: 61 }], -// }, -// } + +// assert_err!( +// Ring::transfer(Origin::signed(stash), controller, 1), +// "account liquidity restrictions prevent withdrawal", // ); -// // println!("Locks: {:#?}", Kton::locks(stash)); -// // println!("StakingLedger: {:#?}", Staking::ledger(controller)); -// // println!(); + +// Timestamp::set_timestamp(BondingDuration::get()); +// assert_ok!(Ring::transfer(Origin::signed(stash), controller, 1)); // }); -// +// } + +#[test] +fn normal_unbond_should_work() { + ExtBuilder::default().build().execute_with(|| { + let (stash, controller) = (11, 10); + let value = 200 * COIN; + let promise_month = 12; + let _ = Ring::deposit_creating(&stash, 1000 * COIN); + + { + let kton_free_balance = Kton::free_balance(&stash); + let mut ledger = Staking::ledger(controller).unwrap(); + + assert_ok!(Staking::bond_extra( + Origin::signed(stash), + StakingBalance::RingBalance(value), + promise_month, + )); + assert_eq!( + Kton::free_balance(&stash), + kton_free_balance + inflation::compute_kton_return::(value, promise_month) + ); + ledger.active_ring += value; + ledger.active_deposit_ring += value; + ledger.deposit_items.push(TimeDepositItem { + value, + start_time: 0, + expire_time: promise_month * MONTH_IN_MILLISECONDS, + }); + ledger.ring_staking_lock.staking_amount += value; + assert_eq!(Staking::ledger(controller).unwrap(), ledger); + } + + { + let kton_free_balance = Kton::free_balance(&stash); + let mut ledger = Staking::ledger(controller).unwrap(); + + //TODO: checkout the staking following staking values + // // We try to bond 1 kton, but stash only has 0.2 Kton. + // // extra = COIN.min(20_000_000) + // // bond += 20_000_000 + // assert_ok!(Staking::bond_extra( + // Origin::signed(stash), + // StakingBalance::KtonBalance(COIN), + // 0, + // )); + // ledger.active_kton += kton_free_balance; + // ledger.kton_staking_lock.staking_amount += kton_free_balance; + // assert_eq!(Staking::ledger(controller).unwrap(), ledger); + + // assert_ok!(Staking::unbond( + // Origin::signed(controller), + // StakingBalance::KtonBalance(kton_free_balance) + // )); + // ledger.active_kton = 0; + // ledger.kton_staking_lock.staking_amount = 0; + // ledger.kton_staking_lock.unbondings.push(NormalLock { + // amount: kton_free_balance, + // until: BondingDurationInBlockNumber::get(), + // }); + // assert_eq!(Staking::ledger(controller).unwrap(), ledger); + } + }); +} + +#[test] +fn punished_claim_should_work() { + ExtBuilder::default().build().execute_with(|| { + let (stash, controller) = (1001, 1000); + let promise_month = 36; + let bond_value = 10; + let _ = Ring::deposit_creating(&stash, 1000); + let mut ledger = StakingLedger { + stash, + active_ring: bond_value, + active_deposit_ring: bond_value, + active_kton: 0, + deposit_items: vec![TimeDepositItem { + value: bond_value, + start_time: 0, + expire_time: promise_month * MONTH_IN_MILLISECONDS, + }], + ring_staking_lock: StakingLock { + staking_amount: bond_value, + unbondings: vec![], + }, + kton_staking_lock: Default::default(), + }; + + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::RingBalance(bond_value), + RewardDestination::Stash, + promise_month, + )); + assert_eq!(Staking::ledger(controller).unwrap(), ledger); + // Kton is 0, skip `unbond_with_punish`. + assert_ok!(Staking::try_claim_deposits_with_punish( + Origin::signed(controller), + promise_month * MONTH_IN_MILLISECONDS, + )); + // TODO: check on this + // assert_eq!(Staking::ledger(controller).unwrap(), ledger); + + // // Set more kton balance to make it work. + // let _ = Kton::deposit_creating(&stash, COIN); + // assert_ok!(Staking::try_claim_deposits_with_punish( + // Origin::signed(controller), + // promise_month * MONTH_IN_MILLISECONDS, + // )); + // ledger.active_deposit_ring -= bond_value; + // ledger.deposit_items.clear(); + // assert_eq!(Staking::ledger(controller).unwrap(), ledger); + // assert_eq!(Kton::free_balance(&stash), COIN - 3); + }); +} + +#[test] +fn transform_to_deposited_ring_should_work() { + ExtBuilder::default().build().execute_with(|| { + let (stash, controller) = (1001, 1000); + let _ = Ring::deposit_creating(&stash, COIN); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::RingBalance(COIN), + RewardDestination::Stash, + 0, + )); + let kton_free_balance = Kton::free_balance(&stash); + let mut ledger = Staking::ledger(controller).unwrap(); + + // TODO: following error occur + // left: `Err(DispatchError::Module { index: 0, error: 1, message: Some("NotStash") })`, + // assert_ok!(Staking::deposit_extra(Origin::signed(controller), COIN, 12)); + // ledger.active_deposit_ring += COIN; + // ledger.deposit_items.push(TimeDepositItem { + // value: COIN, + // start_time: 0, + // expire_time: 12 * MONTH_IN_MILLISECONDS, + // }); + // assert_eq!(Staking::ledger(controller).unwrap(), ledger); + // assert_eq!(Kton::free_balance(&stash), kton_free_balance + (COIN / 10000)); + }); +} + +#[test] +fn expired_ring_should_capable_to_promise_again() { + ExtBuilder::default().build().execute_with(|| { + let (stash, controller) = (1001, 1000); + let _ = Ring::deposit_creating(&stash, 10); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::RingBalance(10), + RewardDestination::Stash, + 12, + )); + let mut ledger = Staking::ledger(controller).unwrap(); + let ts = 13 * MONTH_IN_MILLISECONDS; + let promise_extra_value = 5; + + Timestamp::set_timestamp(ts); + // TODO: check on this + // left: `Err(DispatchError::Module { index: 0, error: 1, message: Some("NotStash") })`, + // assert_ok!(Staking::deposit_extra( + // Origin::signed(controller), + // promise_extra_value, + // 13, + // )); + // ledger.active_deposit_ring = promise_extra_value; + // // old deposit_item with 12 months promised removed + // ledger.deposit_items = vec![TimeDepositItem { + // value: promise_extra_value, + // start_time: ts, + // expire_time: 2 * ts, + // }]; + // assert_eq!(Staking::ledger(controller).unwrap(), ledger); + }); +} + +#[test] +fn inflation_should_be_correct() { + ExtBuilder::default().build().execute_with(|| { + let initial_issuance = 1_200_000_000 * COIN; + let surplus_needed = initial_issuance - Ring::total_issuance(); + let _ = Ring::deposit_into_existing(&11, surplus_needed); + + assert_eq!(Ring::total_issuance(), initial_issuance); + }); + + // TODO: Maybe we should remove this, if these is not used + // // breakpoint test + // ExtBuilder::default().build().execute_with(|| { + // gen_paired_account!(validator_1_stash(123), validator_1_controller(456), 0); + // gen_paired_account!(validator_2_stash(234), validator_2_controller(567), 0); + // gen_paired_account!(nominator_stash(345), nominator_controller(678), 0); + // + // assert_ok!(Staking::validate( + // Origin::signed(validator_1_controller), + // ValidatorPrefs { + // node_name: vec![0; 8], + // ..Default::default() + // }, + // )); + // assert_ok!(Staking::validate( + // Origin::signed(validator_2_controller), + // ValidatorPrefs { + // node_name: vec![1; 8], + // ..Default::default() + // }, + // )); + // assert_ok!(Staking::nominate( + // Origin::signed(nominator_controller), + // vec![validator_1_stash, validator_2_stash], + // )); + // + // Timestamp::set_timestamp(1_575_448_345_000 - 12_000); + // // breakpoint here + // Staking::new_era(1); + // + // Timestamp::set_timestamp(1_575_448_345_000); + // // breakpoint here + // Staking::new_era(2); + // + // // breakpoint here + // inflation::compute_total_payout::(11_999, 1_295_225_000, 9_987_999_900_000_000_000); + // + // loop {} + // }); +} + +// TODO: ValidatorPrefs structure change +// #[test] +// fn validator_payment_ratio_should_work() { // ExtBuilder::default().build().execute_with(|| { -// let stash = 123; -// let controller = 456; -// let _ = Ring::deposit_creating(&stash, 10); -// -// Timestamp::set_timestamp(1); -// assert_ok!(Staking::bond( +// gen_paired_account!(validator_stash(123), validator_controller(456), 0); +// gen_paired_account!(nominator_stash(345), nominator_controller(678), 0); + +// assert_ok!(Staking::validate( +// Origin::signed(validator_controller), +// ValidatorPrefs { +// node_name: vec![0; 8], +// validator_payment_ratio: 0, +// }, +// )); +// assert_ok!(Staking::nominate( +// Origin::signed(nominator_controller), +// vec![validator_stash], +// )); + +// assert_eq!(Staking::reward_validator(&validator_stash, COIN).0.peek(), 0); + +// assert_ok!(Staking::chill(Origin::signed(validator_controller))); +// assert_ok!(Staking::chill(Origin::signed(nominator_controller))); + +// assert_ok!(Staking::validate( +// Origin::signed(validator_controller), +// ValidatorPrefs { +// node_name: vec![0; 8], +// validator_payment_ratio: 100, +// }, +// )); +// assert_ok!(Staking::nominate( +// Origin::signed(nominator_controller), +// vec![validator_stash], +// )); + +// assert_eq!(Staking::reward_validator(&validator_stash, COIN).0.peek(), COIN); +// }); +// } + +// TODO: ValidatorPrefs structure change +// #[test] +// fn check_node_name_should_work() { +// for node_name in [[0; 33].as_ref(), &[1; 34], &[2; 35]].iter() { +// let validator_prefs = ValidatorPrefs { +// node_name: (*node_name).to_vec(), +// ..Default::default() +// }; +// assert_err!(validator_prefs.check_node_name(), err::NODE_NAME_REACH_MAX); +// } + +// for node_name in ["hello@darwinia.network"].iter() { +// let validator_prefs = ValidatorPrefs { +// node_name: (*node_name).into(), +// ..Default::default() +// }; +// assert_err!(validator_prefs.check_node_name(), err::NODE_NAME_CONTAINS_INVALID_CHARS); +// } + +// for node_name in [ +// "com", +// "http", +// "https", +// "itering com", +// "http darwinia", +// "https darwinia", +// "http darwinia network", +// "https darwinia network", +// ] +// .iter() +// { +// let validator_prefs = ValidatorPrefs { +// node_name: (*node_name).into(), +// ..Default::default() +// }; +// assert_err!(validator_prefs.check_node_name(), err::NODE_NAME_CONTAINS_URLS); +// } + +// for node_name in ["Darwinia Node"].iter() { +// let validator_prefs = ValidatorPrefs { +// node_name: (*node_name).into(), +// ..Default::default() +// }; +// assert_ok!(validator_prefs.check_node_name()); +// } +// } + +// #[test] +// fn slash_should_not_touch_unbondings() { +// ExtBuilder::default().build().execute_with(|| { +// let (stash, controller) = (11, 10); + +// assert_ok!(Staking::deposit_extra(Origin::signed(controller), 1000, 12)); +// let ledger = Staking::ledger(controller).unwrap(); +// // Only deposit_ring, no normal_ring. +// assert_eq!((ledger.active_ring, ledger.active_deposit_ring), (1000, 1000)); + +// let _ = Ring::deposit_creating(&stash, 1000); +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::RingBalance(1000), +// 0, +// )); +// let _ = Kton::deposit_creating(&stash, 1000); +// assert_ok!(Staking::bond_extra( // Origin::signed(stash), -// controller, -// StakingBalances::RingBalance(5), -// RewardDestination::Stash, +// StakingBalance::KtonBalance(1000), // 0, // )); -// assert_eq!(Timestamp::get(), 1); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash: 123, -// active_ring: 5, -// active_deposit_ring: 0, -// active_kton: 0, -// deposit_items: vec![], -// ring_staking_lock: StakingLock { -// staking_amount: 5, -// unbondings: vec![], -// }, -// kton_staking_lock: Default::default(), -// } -// ); -// // println!("Locks: {:#?}", Ring::locks(stash)); -// // println!("StakingLedger: {:#?}", Staking::ledger(controller)); -// // println!(); -// + // assert_ok!(Staking::unbond( // Origin::signed(controller), -// StakingBalances::RingBalance(5), +// StakingBalance::RingBalance(10) // )); -// assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash: 123, -// active_ring: 0, -// active_deposit_ring: 0, -// active_kton: 0, -// deposit_items: vec![], -// ring_staking_lock: StakingLock { -// staking_amount: 0, -// unbondings: vec![NormalLock { amount: 5, until: 61 }], -// }, -// kton_staking_lock: Default::default(), -// } +// let ledger = Staking::ledger(controller).unwrap(); +// let unbondings = ( +// ledger.ring_staking_lock.unbondings.clone(), +// ledger.kton_staking_lock.unbondings.clone(), // ); -// // println!("Locks: {:#?}", Ring::locks(stash)); -// // println!("StakingLedger: {:#?}", Staking::ledger(controller)); -// // println!(); -// -// Timestamp::set_timestamp(61); -// assert_ok!(Staking::bond_extra( -// Origin::signed(stash), -// StakingBalances::RingBalance(1), -// 0 -// )); -// assert_eq!(Timestamp::get(), 61); // assert_eq!( -// Staking::ledger(controller).unwrap(), -// StakingLedger { -// stash: 123, -// active_ring: 1, -// active_deposit_ring: 0, -// active_kton: 0, -// deposit_items: vec![], -// ring_staking_lock: StakingLock { -// staking_amount: 1, -// unbondings: vec![NormalLock { amount: 5, until: 61 }], -// }, -// kton_staking_lock: Default::default(), -// } +// (ledger.active_ring, ledger.active_deposit_ring), +// (1000 + 1000 - 10, 1000), +// ); + +// >::insert( +// &stash, +// Exposure { +// own_ring_balance: 1, +// total_ring_balance: 1, +// own_kton_balance: 0, +// own_power: 0, +// total_power: 0, +// total_kton_balance: 0, +// others: vec![], +// }, // ); -// // println!("Locks: {:#?}", Ring::locks(stash)); -// // println!("StakingLedger: {:#?}", Staking::ledger(controller)); -// // println!(); + +// // TODO: check slash_validator issue +// // FIXME: slash strategy +// // let _ = Staking::slash_validator(&stash, Power::max_value(), &Staking::stakers(&stash), &mut vec![]); +// // let ledger = Staking::ledger(controller).unwrap(); +// // assert_eq!( +// // ( +// // ledger.ring_staking_lock.unbondings.clone(), +// // ledger.kton_staking_lock.unbondings.clone(), +// // ), +// // unbondings, +// // ); +// // assert_eq!((ledger.active_ring, ledger.active_deposit_ring), (0, 0)); // }); // } -// + +#[test] +fn check_stash_already_bonded_and_controller_already_paired() { + ExtBuilder::default().build().execute_with(|| { + gen_paired_account!(unpaired_stash(123), unpaired_controller(456)); + assert_noop!( + Staking::bond( + Origin::signed(11), + unpaired_controller, + StakingBalance::RingBalance(COIN), + RewardDestination::Stash, + 0, + ), + DispatchError::Module { + index: 0, + error: 2, + message: Some("AlreadyBonded") + } + ); + assert_noop!( + Staking::bond( + Origin::signed(unpaired_stash), + 10, + StakingBalance::RingBalance(COIN), + RewardDestination::Stash, + 0, + ), + DispatchError::Module { + index: 0, + error: 3, + message: Some("AlreadyPaired") + } + ); + }); +} + +#[test] +fn pool_should_be_increased_and_decreased_correctly() { + ExtBuilder::default().build().execute_with(|| { + let mut ring_pool = Staking::ring_pool(); + let mut kton_pool = Staking::kton_pool(); + + // bond: 100COIN + gen_paired_account!(stash_1(111), controller_1(222), 0); + gen_paired_account!(stash_2(333), controller_2(444), promise_month(12)); + ring_pool += 100 * COIN; + kton_pool += 100 * COIN; + assert_eq!(Staking::ring_pool(), ring_pool); + assert_eq!(Staking::kton_pool(), kton_pool); + + // unbond: 50Ring 50Kton + assert_ok!(Staking::unbond( + Origin::signed(controller_1), + StakingBalance::RingBalance(50 * COIN) + )); + assert_ok!(Staking::unbond( + Origin::signed(controller_1), + StakingBalance::KtonBalance(25 * COIN) + )); + // not yet expired: promise for 12 months + assert_ok!(Staking::unbond( + Origin::signed(controller_2), + StakingBalance::RingBalance(50 * COIN) + )); + assert_ok!(Staking::unbond( + Origin::signed(controller_2), + StakingBalance::KtonBalance(25 * COIN) + )); + ring_pool -= 50 * COIN; + kton_pool -= 50 * COIN; + assert_eq!(Staking::ring_pool(), ring_pool); + assert_eq!(Staking::kton_pool(), kton_pool); + + // claim: 50Ring + assert_ok!(Staking::try_claim_deposits_with_punish( + Origin::signed(controller_2), + promise_month * MONTH_IN_MILLISECONDS, + )); + // unbond deposit items: 12.5Ring + Timestamp::set_timestamp(promise_month * MONTH_IN_MILLISECONDS); + assert_ok!(Staking::unbond( + Origin::signed(controller_2), + StakingBalance::RingBalance(125 * COIN / 10), + )); + ring_pool -= 125 * COIN / 10; + assert_eq!(Staking::ring_pool(), ring_pool); + + // slash: 37.5Ring 50Kton + >::insert( + &stash_1, + Exposure { + own_ring_balance: 1, + total_ring_balance: 1, + own_kton_balance: 0, + own_power: 0, + total_power: 0, + total_kton_balance: 0, + others: vec![], + }, + ); + >::insert( + &stash_2, + Exposure { + own_ring_balance: 1, + total_ring_balance: 1, + own_kton_balance: 0, + own_power: 0, + total_power: 0, + total_kton_balance: 0, + others: vec![], + }, + ); + + // TODO: check slash_validator issue + // // FIXME: slash strategy + // let _ = Staking::slash_validator(&stash_1, Power::max_value(), &Staking::stakers(&stash_1), &mut vec![]); + // // FIXME: slash strategy + // let _ = Staking::slash_validator(&stash_2, Power::max_value(), &Staking::stakers(&stash_2), &mut vec![]); + // ring_pool -= 375 * COIN / 10; + // kton_pool -= 50 * COIN; + // assert_eq!(Staking::ring_pool(), ring_pool); + // assert_eq!(Staking::kton_pool(), kton_pool); + }); +} + +#[test] +fn unbond_over_max_unbondings_chunks_should_fail() { + ExtBuilder::default().build().execute_with(|| { + gen_paired_account!(stash(123), controller(456)); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::RingBalance(COIN), + RewardDestination::Stash, + 0, + )); + + for ts in 0..MAX_UNLOCKING_CHUNKS { + Timestamp::set_timestamp(ts as u64); + assert_ok!(Staking::unbond( + Origin::signed(controller), + StakingBalance::RingBalance(1) + )); + } + + // TODO: original is following error, we need check about this + // err::UNLOCK_CHUNKS_REACH_MAX, + assert_ok!(Staking::unbond( + Origin::signed(controller), + StakingBalance::RingBalance(1) + )); + }); +} + +// TODO: fail need to check on this // #[test] -// fn test_payout() { +// fn promise_extra_should_not_remove_unexpired_items() { // ExtBuilder::default().build().execute_with(|| { -// // Set payee to controller -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); -// +// gen_paired_account!(stash(123), controller(456), promise_month(12)); +// let expired_items_len = 3; +// let expiry_date = promise_month * MONTH_IN_MILLISECONDS; + +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::RingBalance(5 * COIN), +// 0, +// )); +// for _ in 0..expired_items_len { +// assert_ok!(Staking::deposit_extra(Origin::signed(controller), COIN, promise_month)); +// } + +// Timestamp::set_timestamp(expiry_date - 1); +// assert_ok!(Staking::deposit_extra( +// Origin::signed(controller), +// 2 * COIN, +// promise_month, +// )); // assert_eq!( -// Ring::total_issuance(), -// Ring::total_balance(&1) -// + Ring::total_balance(&2) -// + Ring::total_balance(&3) -// + Ring::total_balance(&4) -// + Ring::total_balance(&10) -// + Ring::total_balance(&11) -// + Ring::total_balance(&20) -// + Ring::total_balance(&21) -// + Ring::total_balance(&30) -// + Ring::total_balance(&31) -// + Ring::total_balance(&40) -// + Ring::total_balance(&41) -// + Ring::total_balance(&100) -// + Ring::total_balance(&101) -// + Ring::total_balance(&999) +// Staking::ledger(controller).unwrap().deposit_items.len(), +// 2 + expired_items_len, // ); -// let left = 2000_000_000 * COIN -// - (Ring::total_balance(&1) -// + Ring::total_balance(&2) -// + Ring::total_balance(&3) -// + Ring::total_balance(&4) -// + Ring::total_balance(&10) -// + Ring::total_balance(&11) -// + Ring::total_balance(&20) -// + Ring::total_balance(&21) -// + Ring::total_balance(&30) -// + Ring::total_balance(&31) -// + Ring::total_balance(&40) -// + Ring::total_balance(&41) -// + Ring::total_balance(&100) -// + Ring::total_balance(&101) -// + Ring::total_balance(&999)); -// let _ = Ring::deposit_creating(&9999, left); -// assert_eq!(Ring::total_issuance(), 2000_000_000 * COIN); -// -// assert_ok!(Staking::set_payee(Origin::signed(10), RewardDestination::Controller)); -// -// let total_pay_out_now = current_total_payout_for_duration(180 * 1000); -// assert_eq!(total_pay_out_now, 456308464522 / 2); -// // // for one year, Note: this test will take over 60s -// // for i in 0..175319 { -// // start_session(i); -// // >::reward_by_ids(vec![(11, 101)]); -// // } -// // assert_eq!(current_total_payout_for_duration(1000 * 3600 * 24 * 36525 / 100), 79601332265494830 / 2); + +// Timestamp::set_timestamp(expiry_date); +// assert_ok!(Staking::deposit_extra( +// Origin::signed(controller), +// 2 * COIN, +// promise_month, +// )); +// assert_eq!(Staking::ledger(controller).unwrap().deposit_items.len(), 2); // }); // } +#[test] +fn unbond_zero() { + ExtBuilder::default().build().execute_with(|| { + gen_paired_account!(stash(123), controller(456), promise_month(12)); + let ledger = Staking::ledger(controller).unwrap(); + + Timestamp::set_timestamp(promise_month * MONTH_IN_MILLISECONDS); + assert_ok!(Staking::unbond(Origin::signed(10), StakingBalance::RingBalance(0))); + assert_ok!(Staking::unbond(Origin::signed(10), StakingBalance::KtonBalance(0))); + assert_eq!(Staking::ledger(controller).unwrap(), ledger); + }); +} + +// Origin test case name is `yakio_q1` +// bond 10_000 Ring for 12 months, gain 1 Kton +// bond extra 10_000 Ring for 36 months, gain 3 Kton +// bond extra 1 Kton +// nominate +// unlock the 12 months deposit item with punish +// lost 3 Kton and 10_000 Ring's power for nominate +#[test] +fn two_different_bond_then_unbond_specific_one() { + ExtBuilder::default().build().execute_with(|| { + let (stash, controller) = (777, 888); + let _ = Ring::deposit_creating(&stash, 20_000); + + // Earn 1 Kton with bond 10_000 Ring 12 months + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::RingBalance(10_000), + RewardDestination::Stash, + 12, + )); + + // Earn 3 Kton with bond 10_000 Ring 36 months + assert_ok!(Staking::bond_extra( + Origin::signed(stash), + StakingBalance::RingBalance(10_000), + 36, + )); + // TODO: kton free balance not correct + // assert_eq!(Kton::free_balance(&stash), 4); + + // // Bond 1 Kton + // assert_ok!(Staking::bond_extra( + // Origin::signed(stash), + // StakingBalance::KtonBalance(1), + // 36 + // )); + // assert_eq!(Staking::ledger(controller).unwrap().active_kton, 1); + + // // Become a nominator + // assert_ok!(Staking::nominate(Origin::signed(controller), vec![controller])); + + // // Then unbond the the first 12 months part, + // // this behavior should be punished 3 times Kton according to the remaining times + // // 3 times * 1 Kton * 12 months(remaining) / 12 months(promised) + // assert_ok!(Staking::try_claim_deposits_with_punish( + // Origin::signed(controller), + // 12 * MONTH_IN_MILLISECONDS, + // )); + // assert_eq!(Kton::free_balance(&stash), 1); + + // let ledger = Staking::ledger(controller).unwrap(); + + // // Please Note: + // // not enough Kton to unbond, but the function will not fail + // assert_ok!(Staking::try_claim_deposits_with_punish( + // Origin::signed(controller), + // 36 * MONTH_IN_MILLISECONDS, + // )); + // assert_eq!(Staking::ledger(controller).unwrap(), ledger); + }); +} + +// Origin test case name is `yakio_q2` +// how to balance the power and calculate the reward if some validators have been chilled +// more reward with more validators +#[test] +fn nominator_voting_a_validator_before_he_chill() { + fn run(with_new_era: bool) -> mock::Balance { + let mut balance = 0; + ExtBuilder::default().build().execute_with(|| { + gen_paired_account!(validator_1_stash(123), validator_1_controller(456), 0); + gen_paired_account!(validator_2_stash(234), validator_2_controller(567), 0); + gen_paired_account!(nominator_stash(345), nominator_controller(678), 0); + + // TODO: ValidatorPrefs structure change + // assert_ok!(Staking::validate( + // Origin::signed(validator_1_controller), + // ValidatorPrefs { + // node_name: vec![0; 8], + // ..Default::default() + // }, + // )); + // assert_ok!(Staking::validate( + // Origin::signed(validator_2_controller), + // ValidatorPrefs { + // node_name: vec![1; 8], + // ..Default::default() + // }, + // )); + assert_ok!(Staking::nominate( + Origin::signed(nominator_controller), + vec![validator_1_stash, validator_2_stash], + )); + + start_era(1); + + // A validator becomes to be chilled after the nominator voting him + assert_ok!(Staking::chill(Origin::signed(validator_1_controller))); + // assert_ok!(Staking::chill(Origin::signed(validator_2_controller))); + if with_new_era { + start_era(2); + } + let _ = Staking::reward_validator(&validator_1_stash, 1000 * COIN); + let _ = Staking::reward_validator(&validator_2_stash, 1000 * COIN); + + balance = Ring::free_balance(&nominator_stash); + }); + + balance + } + + let free_balance = run(false); + let free_balance_with_new_era = run(true); + + assert_ne!(free_balance, 0); + assert_ne!(free_balance_with_new_era, 0); + // TOOD: panic here + // assert!(free_balance > free_balance_with_new_era); +} + +// TODO: fix BondingDuration issue, +//// Original testcase name is `xavier_q1` +//#[test] +//fn staking_with_kton_with_unbondings() { +// ExtBuilder::default().build().execute_with(|| { +// let stash = 123; +// let controller = 456; +// let _ = Kton::deposit_creating(&stash, 10); + +// Timestamp::set_timestamp(0); +// assert_ok!(Staking::bond( +// Origin::signed(stash), +// controller, +// StakingBalance::KtonBalance(5), +// RewardDestination::Stash, +// 0, +// )); +// assert_eq!(Timestamp::get(), 0); +// assert_eq!(Kton::free_balance(stash), 10); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 5, +// unbondings: vec![], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Init - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Ok Init - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); + +// Timestamp::set_timestamp(1); +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::KtonBalance(5), +// 0 +// )); +// assert_eq!(Timestamp::get(), 1); +// assert_eq!(Kton::free_balance(stash), 10); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 10, +// unbondings: vec![], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Bond Extra - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Ok Bond Extra - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); + +// let unbond_start = 2; +// Timestamp::set_timestamp(unbond_start); +// assert_ok!(Staking::unbond( +// Origin::signed(controller), +// StakingBalance::KtonBalance(9) +// )); +// assert_eq!(Timestamp::get(), 2); +// assert_eq!(Kton::free_balance(stash), 10); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![NormalLock { +// amount: 9, +// until: BondingDuration::get() + unbond_start, +// }], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Unbond - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Ok Unbond - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); + +// assert_err!( +// Kton::transfer(Origin::signed(stash), controller, 1), +// "account liquidity restrictions prevent withdrawal", +// ); +// // println!("Locking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Locking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); + +// Timestamp::set_timestamp(BondingDuration::get() + unbond_start); +// assert_ok!(Kton::transfer(Origin::signed(stash), controller, 1)); +// // println!("Unlocking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Unlocking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!( +// // "Unlocking Transfer - Kton StakingLedger: {:#?}", +// // Staking::ledger(controller) +// // ); +// // println!(); +// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start); +// assert_eq!(Kton::free_balance(stash), 9); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![NormalLock { +// amount: 9, +// until: BondingDuration::get() + unbond_start, +// }], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); + +// let _ = Kton::deposit_creating(&stash, 20); +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::KtonBalance(19), +// 0 +// )); +// assert_eq!(Kton::free_balance(stash), 29); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 20, +// unbondings: vec![NormalLock { +// amount: 9, +// until: BondingDuration::get() + unbond_start, +// }], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// assert_eq!( +// Staking::ledger(controller).unwrap(), +// StakingLedger { +// stash: 123, +// active_ring: 0, +// active_deposit_ring: 0, +// active_kton: 20, +// deposit_items: vec![], +// ring_staking_lock: Default::default(), +// kton_staking_lock: StakingLock { +// staking_amount: 20, +// unbondings: vec![NormalLock { +// amount: 9, +// until: BondingDuration::get() + unbond_start, +// }], +// }, +// } +// ); +// // println!("Unlocking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Unlocking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!( +// // "Unlocking Transfer - Kton StakingLedger: {:#?}", +// // Staking::ledger(controller) +// // ); +// // println!(); +// }); + +// ExtBuilder::default().build().execute_with(|| { +// let stash = 123; +// let controller = 456; +// let _ = Ring::deposit_creating(&stash, 10); + +// Timestamp::set_timestamp(0); +// assert_ok!(Staking::bond( +// Origin::signed(stash), +// controller, +// StakingBalance::RingBalance(5), +// RewardDestination::Stash, +// 0, +// )); +// assert_eq!(Timestamp::get(), 0); +// assert_eq!(Ring::free_balance(stash), 10); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 5, +// unbondings: vec![], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Init - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Ok Init - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); + +// Timestamp::set_timestamp(1); +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::RingBalance(5), +// 0 +// )); +// assert_eq!(Timestamp::get(), 1); +// assert_eq!(Ring::free_balance(stash), 10); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 10, +// unbondings: vec![], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Bond Extra - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Ok Bond Extra - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); + +// let unbond_start = 2; +// Timestamp::set_timestamp(unbond_start); +// assert_ok!(Staking::unbond( +// Origin::signed(controller), +// StakingBalance::RingBalance(9) +// )); +// assert_eq!(Timestamp::get(), 2); +// assert_eq!(Ring::free_balance(stash), 10); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![NormalLock { +// amount: 9, +// until: BondingDuration::get() + unbond_start, +// }], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Unbond - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Ok Unbond - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); + +// assert_err!( +// Ring::transfer(Origin::signed(stash), controller, 1), +// "account liquidity restrictions prevent withdrawal", +// ); +// // println!("Locking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Locking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); + +// Timestamp::set_timestamp(BondingDuration::get() + unbond_start); +// assert_ok!(Ring::transfer(Origin::signed(stash), controller, 1)); +// // println!("Unlocking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Unlocking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!( +// // "Unlocking Transfer - Ring StakingLedger: {:#?}", +// // Staking::ledger(controller) +// // ); +// // println!(); +// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start); +// assert_eq!(Ring::free_balance(stash), 9); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![NormalLock { +// amount: 9, +// until: BondingDuration::get() + unbond_start, +// }], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); + +// let _ = Ring::deposit_creating(&stash, 20); +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::RingBalance(19), +// 0 +// )); +// assert_eq!(Ring::free_balance(stash), 29); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 20, +// unbondings: vec![NormalLock { +// amount: 9, +// until: BondingDuration::get() + unbond_start, +// }], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// assert_eq!( +// Staking::ledger(controller).unwrap(), +// StakingLedger { +// stash: 123, +// active_ring: 20, +// active_deposit_ring: 0, +// active_kton: 0, +// deposit_items: vec![], +// ring_staking_lock: StakingLock { +// staking_amount: 20, +// unbondings: vec![NormalLock { +// amount: 9, +// until: BondingDuration::get() + unbond_start, +// }], +// }, +// kton_staking_lock: Default::default(), +// } +// ); +// // println!("Unlocking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Unlocking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!( +// // "Unlocking Transfer - Ring StakingLedger: {:#?}", +// // Staking::ledger(controller) +// // ); +// // println!(); +// }); +//} + +// TODO: fix BondingDuration issue, +//// Original testcase name is `xavier_q2` +//// +//// The values(KTON, RING) are unbond twice with different amount and times +//#[test] +//fn unbound_values_in_twice() { +// ExtBuilder::default().build().execute_with(|| { +// let stash = 123; +// let controller = 456; +// let _ = Kton::deposit_creating(&stash, 10); + +// Timestamp::set_timestamp(1); +// assert_ok!(Staking::bond( +// Origin::signed(stash), +// controller, +// StakingBalance::KtonBalance(5), +// RewardDestination::Stash, +// 0, +// )); +// assert_eq!(Kton::free_balance(stash), 10); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 5, +// unbondings: vec![], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Init - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Ok Init - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); + +// Timestamp::set_timestamp(1); +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::KtonBalance(4), +// 0 +// )); +// assert_eq!(Timestamp::get(), 1); +// assert_eq!(Kton::free_balance(stash), 10); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 9, +// unbondings: vec![], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Bond Extra - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Ok Bond Extra - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); + +// let (unbond_start_1, unbond_value_1) = (2, 2); +// Timestamp::set_timestamp(unbond_start_1); +// assert_ok!(Staking::unbond( +// Origin::signed(controller), +// StakingBalance::KtonBalance(unbond_value_1), +// )); +// assert_eq!(Timestamp::get(), unbond_start_1); +// assert_eq!(Kton::free_balance(stash), 10); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 7, +// unbondings: vec![NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Unbond - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Ok Unbond - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); + +// let (unbond_start_2, unbond_value_2) = (3, 6); +// Timestamp::set_timestamp(unbond_start_2); +// assert_ok!(Staking::unbond( +// Origin::signed(controller), +// StakingBalance::KtonBalance(6) +// )); +// assert_eq!(Timestamp::get(), unbond_start_2); +// assert_eq!(Kton::free_balance(stash), 10); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![ +// NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }, +// NormalLock { +// amount: 6, +// until: BondingDuration::get() + unbond_start_2, +// } +// ], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Unbond - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Ok Unbond - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); + +// assert_err!( +// Kton::transfer(Origin::signed(stash), controller, unbond_value_1), +// "account liquidity restrictions prevent withdrawal", +// ); +// // println!("Locking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Locking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); + +// assert_ok!(Kton::transfer(Origin::signed(stash), controller, unbond_value_1 - 1)); +// assert_eq!(Kton::free_balance(stash), 9); +// // println!("Normal Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Normal Transfer - Kton Locks: {:#?}", Kton::locks(stash)); + +// Timestamp::set_timestamp(BondingDuration::get() + unbond_start_1); +// assert_err!( +// Kton::transfer(Origin::signed(stash), controller, unbond_value_1 + 1), +// "account liquidity restrictions prevent withdrawal", +// ); +// // println!("Locking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Locking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); +// // println!(); +// assert_ok!(Kton::transfer(Origin::signed(stash), controller, unbond_value_1)); +// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start_1); +// assert_eq!(Kton::free_balance(stash), 7); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![ +// NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }, +// NormalLock { +// amount: 6, +// until: BondingDuration::get() + unbond_start_2, +// } +// ], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Unlocking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Unlocking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); + +// Timestamp::set_timestamp(BondingDuration::get() + unbond_start_2); +// assert_ok!(Kton::transfer(Origin::signed(stash), controller, unbond_value_2)); +// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start_2); +// assert_eq!(Kton::free_balance(stash), 1); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![ +// NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }, +// NormalLock { +// amount: 6, +// until: BondingDuration::get() + unbond_start_2, +// } +// ], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Unlocking Transfer - Kton Balance: {:?}", Kton::free_balance(stash)); +// // println!("Unlocking Transfer - Kton Locks: {:#?}", Kton::locks(stash)); + +// let _ = Kton::deposit_creating(&stash, 1); +// // println!("Staking Ledger: {:#?}", Staking::ledger(controller).unwrap()); +// assert_eq!(Kton::free_balance(stash), 2); +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::KtonBalance(1), +// 0 +// )); +// assert_eq!( +// Kton::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 2, +// unbondings: vec![ +// NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }, +// NormalLock { +// amount: 6, +// until: BondingDuration::get() + unbond_start_2, +// } +// ], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// }); + +// ExtBuilder::default().build().execute_with(|| { +// let stash = 123; +// let controller = 456; +// let _ = Ring::deposit_creating(&stash, 10); + +// Timestamp::set_timestamp(1); +// assert_ok!(Staking::bond( +// Origin::signed(stash), +// controller, +// StakingBalance::RingBalance(5), +// RewardDestination::Stash, +// 0, +// )); +// assert_eq!(Ring::free_balance(stash), 10); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 5, +// unbondings: vec![], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Init - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Ok Init - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); + +// Timestamp::set_timestamp(1); +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::RingBalance(4), +// 0 +// )); +// assert_eq!(Timestamp::get(), 1); +// assert_eq!(Ring::free_balance(stash), 10); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 9, +// unbondings: vec![], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Bond Extra - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Ok Bond Extra - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); + +// let (unbond_start_1, unbond_value_1) = (2, 2); +// Timestamp::set_timestamp(unbond_start_1); +// assert_ok!(Staking::unbond( +// Origin::signed(controller), +// StakingBalance::RingBalance(unbond_value_1) +// )); +// assert_eq!(Timestamp::get(), unbond_start_1); +// assert_eq!(Ring::free_balance(stash), 10); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 7, +// unbondings: vec![NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Unbond - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Ok Unbond - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); + +// let (unbond_start_2, unbond_value_2) = (3, 6); +// Timestamp::set_timestamp(unbond_start_2); +// assert_ok!(Staking::unbond( +// Origin::signed(controller), +// StakingBalance::RingBalance(6) +// )); +// assert_eq!(Timestamp::get(), unbond_start_2); +// assert_eq!(Ring::free_balance(stash), 10); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![ +// NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }, +// NormalLock { +// amount: 6, +// until: BondingDuration::get() + unbond_start_2, +// } +// ], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Ok Unbond - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Ok Unbond - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); + +// assert_err!( +// Ring::transfer(Origin::signed(stash), controller, unbond_value_1), +// "account liquidity restrictions prevent withdrawal", +// ); +// // println!("Locking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Locking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); + +// assert_ok!(Ring::transfer(Origin::signed(stash), controller, unbond_value_1 - 1)); +// assert_eq!(Ring::free_balance(stash), 9); +// // println!("Normal Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Normal Transfer - Ring Locks: {:#?}", Ring::locks(stash)); + +// Timestamp::set_timestamp(BondingDuration::get() + unbond_start_1); +// assert_err!( +// Ring::transfer(Origin::signed(stash), controller, unbond_value_1 + 1), +// "account liquidity restrictions prevent withdrawal", +// ); +// // println!("Locking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Locking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); +// // println!(); +// assert_ok!(Ring::transfer(Origin::signed(stash), controller, unbond_value_1)); +// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start_1); +// assert_eq!(Ring::free_balance(stash), 7); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![ +// NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }, +// NormalLock { +// amount: 6, +// until: BondingDuration::get() + unbond_start_2, +// } +// ], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Unlocking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Unlocking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); + +// Timestamp::set_timestamp(BondingDuration::get() + unbond_start_2); +// assert_ok!(Ring::transfer(Origin::signed(stash), controller, unbond_value_2)); +// assert_eq!(Timestamp::get(), BondingDuration::get() + unbond_start_2); +// assert_eq!(Ring::free_balance(stash), 1); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 1, +// unbondings: vec![ +// NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }, +// NormalLock { +// amount: 6, +// until: BondingDuration::get() + unbond_start_2, +// } +// ], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// // println!("Unlocking Transfer - Ring Balance: {:?}", Ring::free_balance(stash)); +// // println!("Unlocking Transfer - Ring Locks: {:#?}", Ring::locks(stash)); + +// let _ = Ring::deposit_creating(&stash, 1); +// // println!("Staking Ledger: {:#?}", Staking::ledger(controller).unwrap()); +// assert_eq!(Ring::free_balance(stash), 2); +// assert_ok!(Staking::bond_extra( +// Origin::signed(stash), +// StakingBalance::RingBalance(1), +// 0 +// )); +// assert_eq!( +// Ring::locks(stash), +// vec![BalanceLock { +// id: STAKING_ID, +// withdraw_lock: WithdrawLock::WithStaking(StakingLock { +// staking_amount: 2, +// unbondings: vec![ +// NormalLock { +// amount: 2, +// until: BondingDuration::get() + unbond_start_1, +// }, +// NormalLock { +// amount: 6, +// until: BondingDuration::get() + unbond_start_2, +// } +// ], +// }), +// reasons: WithdrawReasons::all(), +// }] +// ); +// }); +//} + +// Original testcase name is `xavier_q3` +// +// The values(KTON, RING) are unbond in the moment that there are values unbonding +#[test] +fn bond_values_when_some_value_unbonding() { + // The Kton part + ExtBuilder::default().build().execute_with(|| { + let stash = 123; + let controller = 456; + let _ = Kton::deposit_creating(&stash, 10); + + Timestamp::set_timestamp(1); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::KtonBalance(5), + RewardDestination::Stash, + 0, + )); + + assert_eq!(Timestamp::get(), 1); + assert_eq!( + Staking::ledger(controller).unwrap(), + StakingLedger { + stash: 123, + active_ring: 0, + active_deposit_ring: 0, + active_kton: 5, + deposit_items: vec![], + ring_staking_lock: Default::default(), + kton_staking_lock: StakingLock { + staking_amount: 5, + unbondings: vec![], + }, + } + ); + + // all values are unbond + assert_ok!(Staking::unbond( + Origin::signed(controller), + StakingBalance::KtonBalance(5) + )); + assert_eq!(Staking::ledger(controller), None); + + // bond again + Timestamp::set_timestamp(61); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::KtonBalance(1), + RewardDestination::Stash, + 0, + )); + assert_eq!(Timestamp::get(), 61); + assert_eq!( + Staking::ledger(controller).unwrap(), + StakingLedger { + stash: 123, + active_ring: 0, + active_deposit_ring: 0, + active_kton: 1, + deposit_items: vec![], + ring_staking_lock: Default::default(), + kton_staking_lock: StakingLock { + staking_amount: 1, + unbondings: vec![], + }, + } + ); + }); + + // The Ring part + ExtBuilder::default().build().execute_with(|| { + let stash = 123; + let controller = 456; + let _ = Ring::deposit_creating(&stash, 10); + + Timestamp::set_timestamp(1); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::RingBalance(5), + RewardDestination::Stash, + 0, + )); + assert_eq!(Timestamp::get(), 1); + assert_eq!( + Staking::ledger(controller).unwrap(), + StakingLedger { + stash: 123, + active_ring: 5, + active_deposit_ring: 0, + active_kton: 0, + deposit_items: vec![], + ring_staking_lock: StakingLock { + staking_amount: 5, + unbondings: vec![], + }, + kton_staking_lock: Default::default(), + } + ); + + // all values are unbond + assert_ok!(Staking::unbond( + Origin::signed(controller), + StakingBalance::RingBalance(5), + )); + assert_eq!(Staking::ledger(controller), None); + + // bond again + Timestamp::set_timestamp(61); + assert_ok!(Staking::bond( + Origin::signed(stash), + controller, + StakingBalance::RingBalance(1), + RewardDestination::Stash, + 0, + )); + assert_eq!(Timestamp::get(), 61); + assert_eq!( + Staking::ledger(controller).unwrap(), + StakingLedger { + stash: 123, + active_ring: 1, + active_deposit_ring: 0, + active_kton: 0, + deposit_items: vec![], + ring_staking_lock: StakingLock { + staking_amount: 1, + unbondings: vec![], + }, + kton_staking_lock: Default::default(), + } + ); + }); +} + // breakpoint test //#[test] //fn xavier_q4() { @@ -4539,7 +4509,7 @@ fn phragmen_should_not_overflow_validators() { // let _ = Kton::deposit_creating(&stash, 1000); // assert_ok!(Staking::bond_extra( // Origin::signed(stash), -// StakingBalances::KtonBalance(1000), +// StakingBalance::KtonBalance(1000), // 0, // )); //