mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-26 14:37:57 +00:00
002d9260f9
**Update:** Pushed additional changes based on the review comments. **This pull request fixes various spelling mistakes in this repository.** Most of the changes are contained in the first **3** commits: - `Fix spelling mistakes in comments and docs` - `Fix spelling mistakes in test names` - `Fix spelling mistakes in error messages, panic messages, logs and tracing` Other source code spelling mistakes are separated into individual commits for easier reviewing: - `Fix the spelling of 'authority'` - `Fix the spelling of 'REASONABLE_HEADERS_IN_JUSTIFICATION_ANCESTRY'` - `Fix the spelling of 'prev_enqueud_messages'` - `Fix the spelling of 'endpoint'` - `Fix the spelling of 'children'` - `Fix the spelling of 'PenpalSiblingSovereignAccount'` - `Fix the spelling of 'PenpalSudoAccount'` - `Fix the spelling of 'insufficient'` - `Fix the spelling of 'PalletXcmExtrinsicsBenchmark'` - `Fix the spelling of 'subtracted'` - `Fix the spelling of 'CandidatePendingAvailability'` - `Fix the spelling of 'exclusive'` - `Fix the spelling of 'until'` - `Fix the spelling of 'discriminator'` - `Fix the spelling of 'nonexistent'` - `Fix the spelling of 'subsystem'` - `Fix the spelling of 'indices'` - `Fix the spelling of 'committed'` - `Fix the spelling of 'topology'` - `Fix the spelling of 'response'` - `Fix the spelling of 'beneficiary'` - `Fix the spelling of 'formatted'` - `Fix the spelling of 'UNKNOWN_PROOF_REQUEST'` - `Fix the spelling of 'succeeded'` - `Fix the spelling of 'reopened'` - `Fix the spelling of 'proposer'` - `Fix the spelling of 'InstantiationNonce'` - `Fix the spelling of 'depositor'` - `Fix the spelling of 'expiration'` - `Fix the spelling of 'phantom'` - `Fix the spelling of 'AggregatedKeyValue'` - `Fix the spelling of 'randomness'` - `Fix the spelling of 'defendant'` - `Fix the spelling of 'AquaticMammal'` - `Fix the spelling of 'transactions'` - `Fix the spelling of 'PassingTracingSubscriber'` - `Fix the spelling of 'TxSignaturePayload'` - `Fix the spelling of 'versioning'` - `Fix the spelling of 'descendant'` - `Fix the spelling of 'overridden'` - `Fix the spelling of 'network'` Let me know if this structure is adequate. **Note:** The usage of the words `Merkle`, `Merkelize`, `Merklization`, `Merkelization`, `Merkleization`, is somewhat inconsistent but I left it as it is. ~~**Note:** In some places the term `Receival` is used to refer to message reception, IMO `Reception` is the correct word here, but I left it as it is.~~ ~~**Note:** In some places the term `Overlayed` is used instead of the more acceptable version `Overlaid` but I also left it as it is.~~ ~~**Note:** In some places the term `Applyable` is used instead of the correct version `Applicable` but I also left it as it is.~~ **Note:** Some usage of British vs American english e.g. `judgement` vs `judgment`, `initialise` vs `initialize`, `optimise` vs `optimize` etc. are both present in different places, but I suppose that's understandable given the number of contributors. ~~**Note:** There is a spelling mistake in `.github/CODEOWNERS` but it triggers errors in CI when I make changes to it, so I left it as it is.~~
859 lines
28 KiB
Rust
859 lines
28 KiB
Rust
// This file is part of Substrate.
|
|
|
|
// Copyright (C) Parity Technologies (UK) Ltd.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
//! The crate's tests.
|
|
|
|
use std::collections::BTreeMap;
|
|
|
|
use frame_support::{
|
|
assert_noop, assert_ok, derive_impl, parameter_types,
|
|
traits::{ConstU32, ConstU64, Contains, Polling, VoteTally},
|
|
};
|
|
use sp_runtime::BuildStorage;
|
|
|
|
use super::*;
|
|
use crate as pallet_conviction_voting;
|
|
|
|
type Block = frame_system::mocking::MockBlock<Test>;
|
|
|
|
frame_support::construct_runtime!(
|
|
pub enum Test
|
|
{
|
|
System: frame_system,
|
|
Balances: pallet_balances,
|
|
Voting: pallet_conviction_voting,
|
|
}
|
|
);
|
|
|
|
// Test that a filtered call can be dispatched.
|
|
pub struct BaseFilter;
|
|
impl Contains<RuntimeCall> for BaseFilter {
|
|
fn contains(call: &RuntimeCall) -> bool {
|
|
!matches!(call, &RuntimeCall::Balances(pallet_balances::Call::force_set_balance { .. }))
|
|
}
|
|
}
|
|
|
|
#[derive_impl(frame_system::config_preludes::TestDefaultConfig)]
|
|
impl frame_system::Config for Test {
|
|
type Block = Block;
|
|
type AccountData = pallet_balances::AccountData<u64>;
|
|
}
|
|
|
|
impl pallet_balances::Config for Test {
|
|
type MaxReserves = ();
|
|
type ReserveIdentifier = [u8; 8];
|
|
type MaxLocks = ConstU32<10>;
|
|
type Balance = u64;
|
|
type RuntimeEvent = RuntimeEvent;
|
|
type DustRemoval = ();
|
|
type ExistentialDeposit = ConstU64<1>;
|
|
type AccountStore = System;
|
|
type WeightInfo = ();
|
|
type FreezeIdentifier = ();
|
|
type MaxFreezes = ();
|
|
type RuntimeHoldReason = ();
|
|
type RuntimeFreezeReason = ();
|
|
}
|
|
|
|
#[derive(Clone, PartialEq, Eq, Debug)]
|
|
pub enum TestPollState {
|
|
Ongoing(TallyOf<Test>, u8),
|
|
Completed(u64, bool),
|
|
}
|
|
use TestPollState::*;
|
|
|
|
parameter_types! {
|
|
pub static Polls: BTreeMap<u8, TestPollState> = vec![
|
|
(1, Completed(1, true)),
|
|
(2, Completed(2, false)),
|
|
(3, Ongoing(Tally::from_parts(0, 0, 0), 0)),
|
|
].into_iter().collect();
|
|
}
|
|
|
|
pub struct TestPolls;
|
|
impl Polling<TallyOf<Test>> for TestPolls {
|
|
type Index = u8;
|
|
type Votes = u64;
|
|
type Moment = u64;
|
|
type Class = u8;
|
|
fn classes() -> Vec<u8> {
|
|
vec![0, 1, 2]
|
|
}
|
|
fn as_ongoing(index: u8) -> Option<(TallyOf<Test>, Self::Class)> {
|
|
Polls::get().remove(&index).and_then(|x| {
|
|
if let TestPollState::Ongoing(t, c) = x {
|
|
Some((t, c))
|
|
} else {
|
|
None
|
|
}
|
|
})
|
|
}
|
|
fn access_poll<R>(
|
|
index: Self::Index,
|
|
f: impl FnOnce(PollStatus<&mut TallyOf<Test>, u64, u8>) -> R,
|
|
) -> R {
|
|
let mut polls = Polls::get();
|
|
let entry = polls.get_mut(&index);
|
|
let r = match entry {
|
|
Some(Ongoing(ref mut tally_mut_ref, class)) =>
|
|
f(PollStatus::Ongoing(tally_mut_ref, *class)),
|
|
Some(Completed(when, succeeded)) => f(PollStatus::Completed(*when, *succeeded)),
|
|
None => f(PollStatus::None),
|
|
};
|
|
Polls::set(polls);
|
|
r
|
|
}
|
|
fn try_access_poll<R>(
|
|
index: Self::Index,
|
|
f: impl FnOnce(PollStatus<&mut TallyOf<Test>, u64, u8>) -> Result<R, DispatchError>,
|
|
) -> Result<R, DispatchError> {
|
|
let mut polls = Polls::get();
|
|
let entry = polls.get_mut(&index);
|
|
let r = match entry {
|
|
Some(Ongoing(ref mut tally_mut_ref, class)) =>
|
|
f(PollStatus::Ongoing(tally_mut_ref, *class)),
|
|
Some(Completed(when, succeeded)) => f(PollStatus::Completed(*when, *succeeded)),
|
|
None => f(PollStatus::None),
|
|
}?;
|
|
Polls::set(polls);
|
|
Ok(r)
|
|
}
|
|
|
|
#[cfg(feature = "runtime-benchmarks")]
|
|
fn create_ongoing(class: Self::Class) -> Result<Self::Index, ()> {
|
|
let mut polls = Polls::get();
|
|
let i = polls.keys().rev().next().map_or(0, |x| x + 1);
|
|
polls.insert(i, Ongoing(Tally::new(0), class));
|
|
Polls::set(polls);
|
|
Ok(i)
|
|
}
|
|
|
|
#[cfg(feature = "runtime-benchmarks")]
|
|
fn end_ongoing(index: Self::Index, approved: bool) -> Result<(), ()> {
|
|
let mut polls = Polls::get();
|
|
match polls.get(&index) {
|
|
Some(Ongoing(..)) => {},
|
|
_ => return Err(()),
|
|
}
|
|
let now = frame_system::Pallet::<Test>::block_number();
|
|
polls.insert(index, Completed(now, approved));
|
|
Polls::set(polls);
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
impl Config for Test {
|
|
type RuntimeEvent = RuntimeEvent;
|
|
type Currency = pallet_balances::Pallet<Self>;
|
|
type VoteLockingPeriod = ConstU64<3>;
|
|
type MaxVotes = ConstU32<3>;
|
|
type WeightInfo = ();
|
|
type MaxTurnout = frame_support::traits::TotalIssuanceOf<Balances, Self::AccountId>;
|
|
type Polls = TestPolls;
|
|
}
|
|
|
|
pub fn new_test_ext() -> sp_io::TestExternalities {
|
|
let mut t = frame_system::GenesisConfig::<Test>::default().build_storage().unwrap();
|
|
pallet_balances::GenesisConfig::<Test> {
|
|
balances: vec![(1, 10), (2, 20), (3, 30), (4, 40), (5, 50), (6, 60)],
|
|
}
|
|
.assimilate_storage(&mut t)
|
|
.unwrap();
|
|
let mut ext = sp_io::TestExternalities::new(t);
|
|
ext.execute_with(|| System::set_block_number(1));
|
|
ext
|
|
}
|
|
|
|
#[test]
|
|
fn params_should_work() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_eq!(Balances::free_balance(42), 0);
|
|
assert_eq!(Balances::total_issuance(), 210);
|
|
});
|
|
}
|
|
|
|
fn next_block() {
|
|
System::set_block_number(System::block_number() + 1);
|
|
}
|
|
|
|
#[allow(dead_code)]
|
|
fn run_to(n: u64) {
|
|
while System::block_number() < n {
|
|
next_block();
|
|
}
|
|
}
|
|
|
|
fn aye(amount: u64, conviction: u8) -> AccountVote<u64> {
|
|
let vote = Vote { aye: true, conviction: conviction.try_into().unwrap() };
|
|
AccountVote::Standard { vote, balance: amount }
|
|
}
|
|
|
|
fn nay(amount: u64, conviction: u8) -> AccountVote<u64> {
|
|
let vote = Vote { aye: false, conviction: conviction.try_into().unwrap() };
|
|
AccountVote::Standard { vote, balance: amount }
|
|
}
|
|
|
|
fn split(aye: u64, nay: u64) -> AccountVote<u64> {
|
|
AccountVote::Split { aye, nay }
|
|
}
|
|
|
|
fn split_abstain(aye: u64, nay: u64, abstain: u64) -> AccountVote<u64> {
|
|
AccountVote::SplitAbstain { aye, nay, abstain }
|
|
}
|
|
|
|
fn tally(index: u8) -> TallyOf<Test> {
|
|
<TestPolls as Polling<TallyOf<Test>>>::as_ongoing(index).expect("No poll").0
|
|
}
|
|
|
|
fn class(index: u8) -> u8 {
|
|
<TestPolls as Polling<TallyOf<Test>>>::as_ongoing(index).expect("No poll").1
|
|
}
|
|
|
|
#[test]
|
|
#[ignore]
|
|
#[should_panic(expected = "No poll")]
|
|
fn unknown_poll_should_panic() {
|
|
let _ = tally(0);
|
|
}
|
|
|
|
#[test]
|
|
#[ignore]
|
|
#[should_panic(expected = "No poll")]
|
|
fn completed_poll_should_panic() {
|
|
let _ = tally(1);
|
|
}
|
|
|
|
#[test]
|
|
fn basic_stuff() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_eq!(tally(3), Tally::from_parts(0, 0, 0));
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn basic_voting_works() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(2, 5)));
|
|
assert_eq!(tally(3), Tally::from_parts(10, 0, 2));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, nay(2, 5)));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 10, 0));
|
|
assert_eq!(Balances::usable_balance(1), 8);
|
|
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(5, 1)));
|
|
assert_eq!(tally(3), Tally::from_parts(5, 0, 5));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, nay(5, 1)));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 5, 0));
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(10, 0)));
|
|
assert_eq!(tally(3), Tally::from_parts(1, 0, 10));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, nay(10, 0)));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 1, 0));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), None, 3));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 0, 0));
|
|
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), class(3), 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn split_voting_works() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, split(10, 0)));
|
|
assert_eq!(tally(3), Tally::from_parts(1, 0, 10));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, split(5, 5)));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 0, 5));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), None, 3));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 0, 0));
|
|
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), class(3), 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn abstain_voting_works() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, split_abstain(0, 0, 10)));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 0, 10));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(2), 3, split_abstain(0, 0, 20)));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 0, 30));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(2), 3, split_abstain(10, 0, 10)));
|
|
assert_eq!(tally(3), Tally::from_parts(1, 0, 30));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
assert_eq!(Balances::usable_balance(2), 0);
|
|
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), None, 3));
|
|
assert_eq!(tally(3), Tally::from_parts(1, 0, 20));
|
|
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(2), None, 3));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 0, 0));
|
|
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), class(3), 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(2), class(3), 2));
|
|
assert_eq!(Balances::usable_balance(2), 20);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn voting_balance_gets_locked() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(2, 5)));
|
|
assert_eq!(tally(3), Tally::from_parts(10, 0, 2));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, nay(2, 5)));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 10, 0));
|
|
assert_eq!(Balances::usable_balance(1), 8);
|
|
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(5, 1)));
|
|
assert_eq!(tally(3), Tally::from_parts(5, 0, 5));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, nay(5, 1)));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 5, 0));
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(10, 0)));
|
|
assert_eq!(tally(3), Tally::from_parts(1, 0, 10));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, nay(10, 0)));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 1, 0));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), None, 3));
|
|
assert_eq!(tally(3), Tally::from_parts(0, 0, 0));
|
|
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), class(3), 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn successful_but_zero_conviction_vote_balance_can_be_unlocked() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(1, 1)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(2), 3, nay(20, 0)));
|
|
let c = class(3);
|
|
Polls::set(vec![(3, Completed(3, false))].into_iter().collect());
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(2), Some(c), 3));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(2), c, 2));
|
|
assert_eq!(Balances::usable_balance(2), 20);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn unsuccessful_conviction_vote_balance_can_be_unlocked() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(1, 1)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(2), 3, nay(20, 0)));
|
|
let c = class(3);
|
|
Polls::set(vec![(3, Completed(3, false))].into_iter().collect());
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(c), 3));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), c, 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn successful_conviction_vote_balance_stays_locked_for_correct_time() {
|
|
new_test_ext().execute_with(|| {
|
|
for i in 1..=5 {
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(i), 3, aye(10, i as u8)));
|
|
}
|
|
let c = class(3);
|
|
Polls::set(vec![(3, Completed(3, true))].into_iter().collect());
|
|
for i in 1..=5 {
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(i), Some(c), 3));
|
|
}
|
|
for block in 1..=(3 + 5 * 3) {
|
|
run_to(block);
|
|
for i in 1..=5 {
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(i), c, i));
|
|
let expired = block >= (3 << (i - 1)) + 3;
|
|
assert_eq!(Balances::usable_balance(i), i * 10 - if expired { 0 } else { 10 });
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn classwise_delegation_works() {
|
|
new_test_ext().execute_with(|| {
|
|
Polls::set(
|
|
vec![
|
|
(0, Ongoing(Tally::new(0), 0)),
|
|
(1, Ongoing(Tally::new(0), 1)),
|
|
(2, Ongoing(Tally::new(0), 2)),
|
|
(3, Ongoing(Tally::new(0), 2)),
|
|
]
|
|
.into_iter()
|
|
.collect(),
|
|
);
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked1x, 5));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 1, 3, Conviction::Locked1x, 5));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 2, 4, Conviction::Locked1x, 5));
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(2), 0, aye(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(2), 1, nay(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(2), 2, nay(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(3), 0, nay(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(3), 1, aye(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(3), 2, nay(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(4), 0, nay(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(4), 1, nay(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(4), 2, aye(10, 0)));
|
|
// 4 hasn't voted yet
|
|
|
|
assert_eq!(
|
|
Polls::get(),
|
|
vec![
|
|
(0, Ongoing(Tally::from_parts(6, 2, 15), 0)),
|
|
(1, Ongoing(Tally::from_parts(6, 2, 15), 1)),
|
|
(2, Ongoing(Tally::from_parts(6, 2, 15), 2)),
|
|
(3, Ongoing(Tally::from_parts(0, 0, 0), 2)),
|
|
]
|
|
.into_iter()
|
|
.collect()
|
|
);
|
|
|
|
// 4 votes nay to 3.
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(4), 3, nay(10, 0)));
|
|
assert_eq!(
|
|
Polls::get(),
|
|
vec![
|
|
(0, Ongoing(Tally::from_parts(6, 2, 15), 0)),
|
|
(1, Ongoing(Tally::from_parts(6, 2, 15), 1)),
|
|
(2, Ongoing(Tally::from_parts(6, 2, 15), 2)),
|
|
(3, Ongoing(Tally::from_parts(0, 6, 0), 2)),
|
|
]
|
|
.into_iter()
|
|
.collect()
|
|
);
|
|
|
|
// Redelegate for class 2 to account 3.
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 2));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 2, 3, Conviction::Locked1x, 5));
|
|
assert_eq!(
|
|
Polls::get(),
|
|
vec![
|
|
(0, Ongoing(Tally::from_parts(6, 2, 15), 0)),
|
|
(1, Ongoing(Tally::from_parts(6, 2, 15), 1)),
|
|
(2, Ongoing(Tally::from_parts(1, 7, 10), 2)),
|
|
(3, Ongoing(Tally::from_parts(0, 1, 0), 2)),
|
|
]
|
|
.into_iter()
|
|
.collect()
|
|
);
|
|
|
|
// Redelegating with a lower lock does not forget previous lock and updates correctly.
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 1));
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 2));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked1x, 3));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 1, 3, Conviction::Locked1x, 3));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 2, 4, Conviction::Locked1x, 3));
|
|
assert_eq!(
|
|
Polls::get(),
|
|
vec![
|
|
(0, Ongoing(Tally::from_parts(4, 2, 13), 0)),
|
|
(1, Ongoing(Tally::from_parts(4, 2, 13), 1)),
|
|
(2, Ongoing(Tally::from_parts(4, 2, 13), 2)),
|
|
(3, Ongoing(Tally::from_parts(0, 4, 0), 2)),
|
|
]
|
|
.into_iter()
|
|
.collect()
|
|
);
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 1, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 2, 1));
|
|
// unlock does nothing since the delegation already took place.
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
|
|
// Redelegating with higher amount extends previous lock.
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked1x, 6));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 4);
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 1));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 1, 3, Conviction::Locked1x, 7));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 1, 1));
|
|
assert_eq!(Balances::usable_balance(1), 3);
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 2));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 2, 4, Conviction::Locked1x, 8));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 2, 1));
|
|
assert_eq!(Balances::usable_balance(1), 2);
|
|
assert_eq!(
|
|
Polls::get(),
|
|
vec![
|
|
(0, Ongoing(Tally::from_parts(7, 2, 16), 0)),
|
|
(1, Ongoing(Tally::from_parts(8, 2, 17), 1)),
|
|
(2, Ongoing(Tally::from_parts(9, 2, 18), 2)),
|
|
(3, Ongoing(Tally::from_parts(0, 9, 0), 2)),
|
|
]
|
|
.into_iter()
|
|
.collect()
|
|
);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn redelegation_after_vote_ending_should_keep_lock() {
|
|
new_test_ext().execute_with(|| {
|
|
Polls::set(vec![(0, Ongoing(Tally::new(0), 0))].into_iter().collect());
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked1x, 5));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(2), 0, aye(10, 1)));
|
|
Polls::set(vec![(0, Completed(1, true))].into_iter().collect());
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 3, Conviction::Locked1x, 3));
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn lock_amalgamation_valid_with_multiple_removed_votes() {
|
|
new_test_ext().execute_with(|| {
|
|
Polls::set(
|
|
vec![
|
|
(0, Ongoing(Tally::new(0), 0)),
|
|
(1, Ongoing(Tally::new(0), 0)),
|
|
(2, Ongoing(Tally::new(0), 0)),
|
|
]
|
|
.into_iter()
|
|
.collect(),
|
|
);
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 0, aye(5, 1)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 1, aye(10, 1)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 2, aye(5, 2)));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
Polls::set(
|
|
vec![(0, Completed(1, true)), (1, Completed(1, true)), (2, Completed(1, true))]
|
|
.into_iter()
|
|
.collect(),
|
|
);
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(0), 0));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(0), 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(0), 2));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
run_to(3);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
run_to(6);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert!(Balances::usable_balance(1) <= 5);
|
|
|
|
run_to(7);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn lock_amalgamation_valid_with_multiple_delegations() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked1x, 5));
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked1x, 10));
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked2x, 5));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
|
|
run_to(3);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
run_to(6);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert!(Balances::usable_balance(1) <= 5);
|
|
|
|
run_to(7);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn lock_amalgamation_valid_with_move_roundtrip_to_delegation() {
|
|
new_test_ext().execute_with(|| {
|
|
Polls::set(vec![(0, Ongoing(Tally::new(0), 0))].into_iter().collect());
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 0, aye(5, 1)));
|
|
Polls::set(vec![(0, Completed(1, true))].into_iter().collect());
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(0), 0));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked1x, 10));
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
Polls::set(vec![(1, Ongoing(Tally::new(0), 0))].into_iter().collect());
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 1, aye(5, 2)));
|
|
Polls::set(vec![(1, Completed(1, true))].into_iter().collect());
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(0), 1));
|
|
|
|
run_to(3);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
run_to(6);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert!(Balances::usable_balance(1) <= 5);
|
|
|
|
run_to(7);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn lock_amalgamation_valid_with_move_roundtrip_to_casting() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked1x, 5));
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
|
|
Polls::set(vec![(0, Ongoing(Tally::new(0), 0))].into_iter().collect());
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 0, aye(10, 1)));
|
|
Polls::set(vec![(0, Completed(1, true))].into_iter().collect());
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(0), 0));
|
|
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked2x, 10));
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
|
|
run_to(3);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
run_to(6);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert!(Balances::usable_balance(1) <= 5);
|
|
|
|
run_to(7);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn lock_aggregation_over_different_classes_with_delegation_works() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::Locked1x, 5));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 1, 2, Conviction::Locked2x, 5));
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 2, 2, Conviction::Locked1x, 10));
|
|
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 1));
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 2));
|
|
|
|
run_to(3);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 1, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 2, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
run_to(6);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 1, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 2, 1));
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
|
|
run_to(7);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 1, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 2, 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn lock_aggregation_over_different_classes_with_casting_works() {
|
|
new_test_ext().execute_with(|| {
|
|
Polls::set(
|
|
vec![
|
|
(0, Ongoing(Tally::new(0), 0)),
|
|
(1, Ongoing(Tally::new(0), 1)),
|
|
(2, Ongoing(Tally::new(0), 2)),
|
|
]
|
|
.into_iter()
|
|
.collect(),
|
|
);
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 0, aye(5, 1)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 1, aye(10, 1)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 2, aye(5, 2)));
|
|
Polls::set(
|
|
vec![(0, Completed(1, true)), (1, Completed(1, true)), (2, Completed(1, true))]
|
|
.into_iter()
|
|
.collect(),
|
|
);
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(0), 0));
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(1), 1));
|
|
assert_ok!(Voting::remove_vote(RuntimeOrigin::signed(1), Some(2), 2));
|
|
|
|
run_to(3);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 1, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 2, 1));
|
|
assert_eq!(Balances::usable_balance(1), 0);
|
|
|
|
run_to(6);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 1, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 2, 1));
|
|
assert_eq!(Balances::usable_balance(1), 5);
|
|
|
|
run_to(7);
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 0, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 1, 1));
|
|
assert_ok!(Voting::unlock(RuntimeOrigin::signed(1), 2, 1));
|
|
assert_eq!(Balances::usable_balance(1), 10);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn errors_with_vote_work() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_noop!(
|
|
Voting::vote(RuntimeOrigin::signed(1), 0, aye(10, 0)),
|
|
Error::<Test>::NotOngoing
|
|
);
|
|
assert_noop!(
|
|
Voting::vote(RuntimeOrigin::signed(1), 1, aye(10, 0)),
|
|
Error::<Test>::NotOngoing
|
|
);
|
|
assert_noop!(
|
|
Voting::vote(RuntimeOrigin::signed(1), 2, aye(10, 0)),
|
|
Error::<Test>::NotOngoing
|
|
);
|
|
assert_noop!(
|
|
Voting::vote(RuntimeOrigin::signed(1), 3, aye(11, 0)),
|
|
Error::<Test>::InsufficientFunds
|
|
);
|
|
|
|
assert_ok!(Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::None, 10));
|
|
assert_noop!(
|
|
Voting::vote(RuntimeOrigin::signed(1), 3, aye(10, 0)),
|
|
Error::<Test>::AlreadyDelegating
|
|
);
|
|
|
|
assert_ok!(Voting::undelegate(RuntimeOrigin::signed(1), 0));
|
|
Polls::set(
|
|
vec![
|
|
(0, Ongoing(Tally::new(0), 0)),
|
|
(1, Ongoing(Tally::new(0), 0)),
|
|
(2, Ongoing(Tally::new(0), 0)),
|
|
(3, Ongoing(Tally::new(0), 0)),
|
|
]
|
|
.into_iter()
|
|
.collect(),
|
|
);
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 0, aye(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 1, aye(10, 0)));
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 2, aye(10, 0)));
|
|
assert_noop!(
|
|
Voting::vote(RuntimeOrigin::signed(1), 3, aye(10, 0)),
|
|
Error::<Test>::MaxVotesReached
|
|
);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn errors_with_delegating_work() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_noop!(
|
|
Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::None, 11),
|
|
Error::<Test>::InsufficientFunds
|
|
);
|
|
assert_noop!(
|
|
Voting::delegate(RuntimeOrigin::signed(1), 3, 2, Conviction::None, 10),
|
|
Error::<Test>::BadClass
|
|
);
|
|
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(10, 0)));
|
|
assert_noop!(
|
|
Voting::delegate(RuntimeOrigin::signed(1), 0, 2, Conviction::None, 10),
|
|
Error::<Test>::AlreadyVoting
|
|
);
|
|
|
|
assert_noop!(Voting::undelegate(RuntimeOrigin::signed(1), 0), Error::<Test>::NotDelegating);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn remove_other_vote_works() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_noop!(
|
|
Voting::remove_other_vote(RuntimeOrigin::signed(2), 1, 0, 3),
|
|
Error::<Test>::NotVoter
|
|
);
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(10, 2)));
|
|
assert_noop!(
|
|
Voting::remove_other_vote(RuntimeOrigin::signed(2), 1, 0, 3),
|
|
Error::<Test>::NoPermission
|
|
);
|
|
Polls::set(vec![(3, Completed(1, true))].into_iter().collect());
|
|
run_to(6);
|
|
assert_noop!(
|
|
Voting::remove_other_vote(RuntimeOrigin::signed(2), 1, 0, 3),
|
|
Error::<Test>::NoPermissionYet
|
|
);
|
|
run_to(7);
|
|
assert_ok!(Voting::remove_other_vote(RuntimeOrigin::signed(2), 1, 0, 3));
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn errors_with_remove_vote_work() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_noop!(
|
|
Voting::remove_vote(RuntimeOrigin::signed(1), Some(0), 3),
|
|
Error::<Test>::NotVoter
|
|
);
|
|
assert_ok!(Voting::vote(RuntimeOrigin::signed(1), 3, aye(10, 2)));
|
|
Polls::set(vec![(3, Completed(1, true))].into_iter().collect());
|
|
assert_noop!(
|
|
Voting::remove_vote(RuntimeOrigin::signed(1), None, 3),
|
|
Error::<Test>::ClassNeeded
|
|
);
|
|
});
|
|
}
|