Skip to content

Commit

Permalink
change staking target extrinsic, closes #1570 (#1623)
Browse files Browse the repository at this point in the history
  • Loading branch information
shannonwells committed Dec 11, 2023
1 parent a3fa8e3 commit 8023276
Show file tree
Hide file tree
Showing 12 changed files with 416 additions and 19 deletions.
54 changes: 54 additions & 0 deletions e2e/capacity/change_staking_target.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
import "@frequency-chain/api-augment";
import { KeyringPair } from "@polkadot/keyring/types";
import { u64, } from "@polkadot/types";
import assert from "assert";
import { ExtrinsicHelper, } from "../scaffolding/extrinsicHelpers";
import {
devAccounts, createKeys, createMsaAndProvider,
stakeToProvider, CHAIN_ENVIRONMENT,
TEST_EPOCH_LENGTH, setEpochLength,
CENTS, DOLLARS, createAndFundKeypair, createProviderKeysAndId
}
from "../scaffolding/helpers";
import { firstValueFrom } from "rxjs";
import { MessageSourceId} from "@frequency-chain/api-augment/interfaces";

describe.only("change_staking_target tests", () => {
const tokenMinStake: bigint = 1n * CENTS;
const capacityMin: bigint = tokenMinStake / 50n;

const unusedMsaId = async () => {
const maxMsaId = (await ExtrinsicHelper.getCurrentMsaIdentifierMaximum()).toNumber();
return maxMsaId + 99;
}

before(async () => {
if (process.env.CHAIN_ENVIRONMENT === CHAIN_ENVIRONMENT.DEVELOPMENT) {
await setEpochLength(devAccounts[0].keys, TEST_EPOCH_LENGTH);
}
});

it("happy path succeeds", async () => {
const providerBalance = 2n * DOLLARS;
const stakeKeys = createKeys("staker");
const oldProvider = await createMsaAndProvider(stakeKeys, "Provider1", providerBalance);
const [_unused, newProvider] = await createProviderKeysAndId();

await assert.doesNotReject(stakeToProvider(stakeKeys, oldProvider, tokenMinStake*3n));

const call = ExtrinsicHelper.changeStakingTarget(stakeKeys, oldProvider, newProvider, tokenMinStake);
const [events] = await call.signAndSend();
assert.notEqual(events, undefined);
});

// not intended to be exhaustive, just check one error case
it("fails if 'to' is not a Provider", async () => {
const providerBalance = 2n * DOLLARS;
const stakeKeys = createKeys("staker");
const notAProvider = await unusedMsaId();
const oldProvider = await createMsaAndProvider(stakeKeys, "Provider1", providerBalance);
await assert.doesNotReject(stakeToProvider(stakeKeys, oldProvider, tokenMinStake*3n));
const call = ExtrinsicHelper.changeStakingTarget(stakeKeys, oldProvider, notAProvider, tokenMinStake);
await assert.rejects(call.signAndSend(), {name: "InvalidTarget"})
});
});
317 changes: 317 additions & 0 deletions pallets/capacity/src/tests/change_staking_target_tests.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,317 @@
use super::{mock::*, testing_utils::*};
use crate::{
BalanceOf, CapacityDetails, Config, CurrentEraInfo, Error, Event, RewardEraInfo,
StakingAccountDetails, StakingAccountLedger, StakingTargetDetails,
};
use common_primitives::{
capacity::StakingType::{MaximumCapacity, ProviderBoost},
msa::MessageSourceId,
};
use frame_support::{assert_noop, assert_ok, traits::Get};

// staker is unused unless amount > 0
fn setup_provider(staker: u64, target: MessageSourceId, amount: u64) {
let provider_name = String::from("Cst-") + target.to_string().as_str();
register_provider(target, provider_name);
if amount > 0 {
assert_ok!(Capacity::stake(RuntimeOrigin::signed(staker), target, amount, ProviderBoost));
}
}

type TestCapacityDetails = CapacityDetails<BalanceOf<Test>, u32>;
type TestTargetDetails = StakingTargetDetails<Test>;

#[test]
fn do_retarget_happy_path() {
new_test_ext().execute_with(|| {
let staker = 200u64;
let from_msa: MessageSourceId = 1;
let from_amount = 20u64;
let to_amount = from_amount / 2;
let to_msa: MessageSourceId = 2;
setup_provider(staker, from_msa, from_amount);
setup_provider(staker, to_msa, to_amount);

// retarget half the stake to to_msa
assert_ok!(Capacity::do_retarget(&staker, &from_msa, &to_msa, &to_amount));

// expect from stake amounts to be halved
let expected_from_details: TestCapacityDetails = CapacityDetails {
remaining_capacity: 1,
total_tokens_staked: 10,
total_capacity_issued: 1,
last_replenished_epoch: 0,
};
let from_capacity_details: TestCapacityDetails =
Capacity::get_capacity_for(from_msa).unwrap();
assert_eq!(from_capacity_details, expected_from_details);

// expect to stake amounts to be increased by the retarget amount
let expected_to_details: TestCapacityDetails = CapacityDetails {
remaining_capacity: 2,
total_tokens_staked: 20,
total_capacity_issued: 2,
last_replenished_epoch: 0,
};
let to_capacity_details = Capacity::get_capacity_for(to_msa).unwrap();
assert_eq!(to_capacity_details, expected_to_details);

let expected_from_target_details: TestTargetDetails =
StakingTargetDetails { amount: 10, capacity: 1 };
let from_target_details = Capacity::get_target_for(staker, from_msa).unwrap();
assert_eq!(from_target_details, expected_from_target_details);

let expected_to_target_details: TestTargetDetails =
StakingTargetDetails { amount: 20, capacity: 2 };
let to_target_details = Capacity::get_target_for(staker, to_msa).unwrap();
assert_eq!(to_target_details, expected_to_target_details);
})
}

#[test]
fn do_retarget_deletes_staking_target_details_if_zero_balance() {
new_test_ext().execute_with(|| {
let staker = 200u64;
let from_msa: MessageSourceId = 1;
let to_msa: MessageSourceId = 2;
let amount = 10u64;
setup_provider(staker, from_msa, amount);
setup_provider(staker, to_msa, amount);

// stake additional to provider from another Msa, doesn't matter which type.
// total staked to from_msa is now 22u64.
assert_ok!(Capacity::stake(
RuntimeOrigin::signed(300u64),
from_msa,
12u64,
MaximumCapacity
));

assert_ok!(Capacity::do_retarget(&staker, &from_msa, &to_msa, &amount));

let expected_from_details: TestCapacityDetails = CapacityDetails {
remaining_capacity: 1,
total_tokens_staked: 12,
total_capacity_issued: 1,
last_replenished_epoch: 0,
};
let from_capacity_details: TestCapacityDetails =
Capacity::get_capacity_for(from_msa).unwrap();
assert_eq!(from_capacity_details, expected_from_details);

let expected_to_details: TestCapacityDetails = CapacityDetails {
remaining_capacity: 2,
total_tokens_staked: 2 * amount,
total_capacity_issued: 2,
last_replenished_epoch: 0,
};

let to_capacity_details = Capacity::get_capacity_for(to_msa).unwrap();
assert_eq!(to_capacity_details, expected_to_details);

assert!(Capacity::get_target_for(staker, from_msa).is_none());

let expected_to_target_details: TestTargetDetails =
StakingTargetDetails { amount: 2 * amount, capacity: 2 };
let to_target_details = Capacity::get_target_for(staker, to_msa).unwrap();
assert_eq!(to_target_details, expected_to_target_details);

assert!(Capacity::get_target_for(staker, from_msa).is_none());
})
}

#[test]
fn change_staking_starget_emits_event_on_success() {
new_test_ext().execute_with(|| {
let staker = 200u64;
let from_msa: MessageSourceId = 1;
let from_amount = 20u64;
let to_amount = from_amount / 2;
let to_msa: MessageSourceId = 2;
setup_provider(staker, from_msa, from_amount);
setup_provider(staker, to_msa, to_amount);

assert_ok!(Capacity::change_staking_target(
RuntimeOrigin::signed(staker),
from_msa,
to_msa,
to_amount
));
let events = staking_events();

assert_eq!(
events.last().unwrap(),
&Event::StakingTargetChanged { account: staker, from_msa, to_msa, amount: to_amount }
);
})
}

#[test]
fn change_staking_target_errors_if_too_many_changes_before_thaw() {
new_test_ext().execute_with(|| {
let staker = 200u64;
let from_msa: MessageSourceId = 1;
let to_msa: MessageSourceId = 2;

let max_chunks: u32 = <Test as Config>::MaxUnlockingChunks::get();
let staking_amount = ((max_chunks + 2u32) * 10u32) as u64;
setup_provider(staker, from_msa, staking_amount);
setup_provider(staker, to_msa, 10);

let retarget_amount = 10u64;
for _i in 0..(max_chunks) {
assert_ok!(Capacity::change_staking_target(
RuntimeOrigin::signed(staker),
from_msa,
to_msa,
retarget_amount
));
}

assert_noop!(
Capacity::change_staking_target(
RuntimeOrigin::signed(staker),
from_msa,
to_msa,
retarget_amount
),
Error::<Test>::MaxUnlockingChunksExceeded
);
});
}

#[test]
fn change_staking_target_garbage_collects_thawed_chunks() {
new_test_ext().execute_with(|| {
let staked_amount = 50u64;
let staking_account = 200u64;
let from_target: MessageSourceId = 3;
let to_target: MessageSourceId = 4;
setup_provider(staking_account, from_target, staked_amount);
setup_provider(staking_account, to_target, staked_amount);

CurrentEraInfo::<Test>::set(RewardEraInfo { era_index: 20, started_at: 100 });
let max_chunks = <Test as Config>::MaxUnlockingChunks::get();
for i in 0..max_chunks {
println!("{:?}", i);
assert_ok!(Capacity::change_staking_target(
RuntimeOrigin::signed(staking_account),
from_target,
to_target,
10u64,
));
}
CurrentEraInfo::<Test>::set(RewardEraInfo { era_index: 25, started_at: 100 });
assert_ok!(Capacity::change_staking_target(
RuntimeOrigin::signed(staking_account),
from_target,
to_target,
10u64,
));
})
}

#[test]
fn change_staking_target_test_parametric_validity() {
new_test_ext().execute_with(|| {
let staked_amount = 10u64;
let from_account = 200u64;

StakingAccountLedger::<Test>::insert(
from_account,
StakingAccountDetails {
active: 20,
total: 20,
unlocking: Default::default(),
staking_type: ProviderBoost,
last_rewards_claimed_at: None,
stake_change_unlocking: Default::default(),
},
);
let from_account_not_staking = 100u64;
let from_target_not_staked: MessageSourceId = 1;
let to_target_not_provider: MessageSourceId = 2;
let from_target: MessageSourceId = 3;
let to_target: MessageSourceId = 4;
setup_provider(from_account, from_target_not_staked, 0);
setup_provider(from_account, from_target, staked_amount);
setup_provider(from_account, to_target, staked_amount);

assert_ok!(Capacity::stake(
RuntimeOrigin::signed(from_account),
from_target,
staked_amount,
ProviderBoost
));

struct TestCase {
from_account: u64,
from_target: MessageSourceId,
to_target: MessageSourceId,
retarget_amount: u64,
expected_err: Error<Test>,
}
let test_cases: Vec<TestCase> = vec![
// from is a provider but account is not staking to it
TestCase {
from_account,
from_target: from_target_not_staked,
to_target,
retarget_amount: staked_amount,
expected_err: Error::<Test>::StakerTargetRelationshipNotFound,
},
// from_account is not staking at all.
TestCase {
from_account: from_account_not_staking,
from_target,
to_target,
retarget_amount: staked_amount,
expected_err: Error::<Test>::NotAStakingAccount,
},
// // from and to providers are valid, but zero amount too small
TestCase {
from_account,
from_target,
to_target,
retarget_amount: 0,
expected_err: Error::<Test>::StakingAmountBelowMinimum,
},
// // nonzero amount below minimum is still too small
TestCase {
from_account,
from_target,
to_target,
retarget_amount: 9,
expected_err: Error::<Test>::StakingAmountBelowMinimum,
},
// // account is staked with from-target, but to-target is not a provider
TestCase {
from_account,
from_target,
to_target: to_target_not_provider,
retarget_amount: staked_amount,
expected_err: Error::<Test>::InvalidTarget,
},
// account doesn't have enough staked to make the transfer
TestCase {
from_account,
from_target,
to_target,
retarget_amount: 999,
expected_err: Error::<Test>::InsufficientStakingBalance,
},
];

for tc in test_cases {
assert_noop!(
Capacity::change_staking_target(
RuntimeOrigin::signed(tc.from_account),
tc.from_target,
tc.to_target,
tc.retarget_amount,
),
tc.expected_err
);
}
});
}
5 changes: 3 additions & 2 deletions pallets/capacity/src/tests/mock.rs
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
use crate as pallet_capacity;

use crate::{BalanceOf, Config, StakingRewardClaim, StakingRewardsProvider};
use crate::{BalanceOf, StakingRewardClaim, StakingRewardsProvider};
use common_primitives::{
node::{AccountId, Balance, Hash, Header, ProposalProvider},
node::{AccountId, Hash, Header, ProposalProvider},
schema::{SchemaId, SchemaValidator},
};
use frame_support::{
Expand Down Expand Up @@ -189,6 +189,7 @@ impl pallet_capacity::Config for Test {
type EraLength = ConstU32<10>;
type StakingRewardsPastErasMax = ConstU32<5>;
type RewardsProvider = TestStakingRewardsProvider;
type ChangeStakingTargetThawEras = ConstU32<5>;
}

pub fn new_test_ext() -> sp_io::TestExternalities {
Expand Down
Loading

0 comments on commit 8023276

Please sign in to comment.