mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-27 17:28:00 +00:00
d0530d0f2a
We were using the wrong syntax and that will be dropped with Rust 2021. The compiler already starts to hint the wrong syntax with warnings. So, we fix this here.
428 lines
14 KiB
Rust
428 lines
14 KiB
Rust
// This file is part of Substrate.
|
|
|
|
// Copyright (C) 2020-2021 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.
|
|
|
|
//! Offences pallet benchmarking.
|
|
|
|
#![cfg_attr(not(feature = "std"), no_std)]
|
|
|
|
mod mock;
|
|
|
|
use sp_std::prelude::*;
|
|
use sp_std::vec;
|
|
|
|
use frame_system::{RawOrigin, Module as System, Config as SystemConfig};
|
|
use frame_benchmarking::{benchmarks, account, impl_benchmark_test_suite};
|
|
use frame_support::traits::{Currency, OnInitialize, ValidatorSet, ValidatorSetWithIdentification};
|
|
|
|
use sp_runtime::{Perbill, traits::{Convert, StaticLookup, Saturating, UniqueSaturatedInto}};
|
|
use sp_staking::offence::{ReportOffence, Offence, OffenceDetails};
|
|
|
|
use pallet_balances::Config as BalancesConfig;
|
|
use pallet_babe::BabeEquivocationOffence;
|
|
use pallet_grandpa::{GrandpaEquivocationOffence, GrandpaTimeSlot};
|
|
use pallet_im_online::{Config as ImOnlineConfig, Module as ImOnline, UnresponsivenessOffence};
|
|
use pallet_offences::{Config as OffencesConfig, Module as Offences};
|
|
use pallet_session::historical::{Config as HistoricalConfig, IdentificationTuple};
|
|
use pallet_session::{Config as SessionConfig, SessionManager};
|
|
use pallet_staking::{
|
|
Module as Staking, Config as StakingConfig, RewardDestination, ValidatorPrefs,
|
|
Exposure, IndividualExposure, ElectionStatus, MAX_NOMINATIONS, Event as StakingEvent
|
|
};
|
|
|
|
const SEED: u32 = 0;
|
|
|
|
const MAX_REPORTERS: u32 = 100;
|
|
const MAX_OFFENDERS: u32 = 100;
|
|
const MAX_NOMINATORS: u32 = 100;
|
|
const MAX_DEFERRED_OFFENCES: u32 = 100;
|
|
|
|
pub struct Module<T: Config>(Offences<T>);
|
|
|
|
pub trait Config:
|
|
SessionConfig
|
|
+ StakingConfig
|
|
+ OffencesConfig
|
|
+ ImOnlineConfig
|
|
+ HistoricalConfig
|
|
+ BalancesConfig
|
|
+ IdTupleConvert<Self>
|
|
{}
|
|
|
|
/// A helper trait to make sure we can convert `IdentificationTuple` coming from historical
|
|
/// and the one required by offences.
|
|
pub trait IdTupleConvert<T: HistoricalConfig + OffencesConfig> {
|
|
/// Convert identification tuple from `historical` trait to the one expected by `offences`.
|
|
fn convert(id: IdentificationTuple<T>) -> <T as OffencesConfig>::IdentificationTuple;
|
|
}
|
|
|
|
impl<T: HistoricalConfig + OffencesConfig> IdTupleConvert<T> for T where
|
|
<T as OffencesConfig>::IdentificationTuple: From<IdentificationTuple<T>>
|
|
{
|
|
fn convert(id: IdentificationTuple<T>) -> <T as OffencesConfig>::IdentificationTuple {
|
|
id.into()
|
|
}
|
|
}
|
|
|
|
type LookupSourceOf<T> = <<T as SystemConfig>::Lookup as StaticLookup>::Source;
|
|
type BalanceOf<T> = <<T as StakingConfig>::Currency as Currency<<T as SystemConfig>::AccountId>>::Balance;
|
|
|
|
struct Offender<T: Config> {
|
|
pub controller: T::AccountId,
|
|
pub stash: T::AccountId,
|
|
pub nominator_stashes: Vec<T::AccountId>,
|
|
}
|
|
|
|
fn bond_amount<T: Config>() -> BalanceOf<T> {
|
|
T::Currency::minimum_balance().saturating_mul(10_000u32.into())
|
|
}
|
|
|
|
fn create_offender<T: Config>(n: u32, nominators: u32) -> Result<Offender<T>, &'static str> {
|
|
let stash: T::AccountId = account("stash", n, SEED);
|
|
let controller: T::AccountId = account("controller", n, SEED);
|
|
let controller_lookup: LookupSourceOf<T> = T::Lookup::unlookup(controller.clone());
|
|
let reward_destination = RewardDestination::Staked;
|
|
let raw_amount = bond_amount::<T>();
|
|
// add twice as much balance to prevent the account from being killed.
|
|
let free_amount = raw_amount.saturating_mul(2u32.into());
|
|
T::Currency::make_free_balance_be(&stash, free_amount);
|
|
let amount: BalanceOf<T> = raw_amount.into();
|
|
Staking::<T>::bond(
|
|
RawOrigin::Signed(stash.clone()).into(),
|
|
controller_lookup.clone(),
|
|
amount.clone(),
|
|
reward_destination.clone(),
|
|
)?;
|
|
|
|
let validator_prefs = ValidatorPrefs {
|
|
commission: Perbill::from_percent(50),
|
|
.. Default::default()
|
|
};
|
|
Staking::<T>::validate(RawOrigin::Signed(controller.clone()).into(), validator_prefs)?;
|
|
|
|
let mut individual_exposures = vec![];
|
|
let mut nominator_stashes = vec![];
|
|
// Create n nominators
|
|
for i in 0 .. nominators {
|
|
let nominator_stash: T::AccountId = account("nominator stash", n * MAX_NOMINATORS + i, SEED);
|
|
let nominator_controller: T::AccountId = account("nominator controller", n * MAX_NOMINATORS + i, SEED);
|
|
let nominator_controller_lookup: LookupSourceOf<T> = T::Lookup::unlookup(nominator_controller.clone());
|
|
T::Currency::make_free_balance_be(&nominator_stash, free_amount.into());
|
|
|
|
Staking::<T>::bond(
|
|
RawOrigin::Signed(nominator_stash.clone()).into(),
|
|
nominator_controller_lookup.clone(),
|
|
amount.clone(),
|
|
reward_destination.clone(),
|
|
)?;
|
|
|
|
let selected_validators: Vec<LookupSourceOf<T>> = vec![controller_lookup.clone()];
|
|
Staking::<T>::nominate(RawOrigin::Signed(nominator_controller.clone()).into(), selected_validators)?;
|
|
|
|
individual_exposures.push(IndividualExposure {
|
|
who: nominator_stash.clone(),
|
|
value: amount.clone(),
|
|
});
|
|
nominator_stashes.push(nominator_stash.clone());
|
|
}
|
|
|
|
let exposure = Exposure {
|
|
total: amount.clone() * n.into(),
|
|
own: amount,
|
|
others: individual_exposures,
|
|
};
|
|
let current_era = 0u32;
|
|
Staking::<T>::add_era_stakers(current_era.into(), stash.clone().into(), exposure);
|
|
|
|
Ok(Offender { controller, stash, nominator_stashes })
|
|
}
|
|
|
|
fn make_offenders<T: Config>(num_offenders: u32, num_nominators: u32) -> Result<
|
|
(Vec<IdentificationTuple<T>>, Vec<Offender<T>>),
|
|
&'static str
|
|
> {
|
|
Staking::<T>::new_session(0);
|
|
|
|
let mut offenders = vec![];
|
|
for i in 0 .. num_offenders {
|
|
let offender = create_offender::<T>(i + 1, num_nominators)?;
|
|
offenders.push(offender);
|
|
}
|
|
|
|
Staking::<T>::start_session(0);
|
|
|
|
let id_tuples = offenders.iter()
|
|
.map(|offender|
|
|
<T as SessionConfig>::ValidatorIdOf::convert(offender.controller.clone())
|
|
.expect("failed to get validator id from account id"))
|
|
.map(|validator_id|
|
|
<T as HistoricalConfig>::FullIdentificationOf::convert(validator_id.clone())
|
|
.map(|full_id| (validator_id, full_id))
|
|
.expect("failed to convert validator id to full identification"))
|
|
.collect::<Vec<IdentificationTuple<T>>>();
|
|
Ok((id_tuples, offenders))
|
|
}
|
|
|
|
fn make_offenders_im_online<T: Config>(num_offenders: u32, num_nominators: u32) -> Result<
|
|
(Vec<pallet_im_online::IdentificationTuple<T>>, Vec<Offender<T>>),
|
|
&'static str
|
|
> {
|
|
Staking::<T>::new_session(0);
|
|
|
|
let mut offenders = vec![];
|
|
for i in 0 .. num_offenders {
|
|
let offender = create_offender::<T>(i + 1, num_nominators)?;
|
|
offenders.push(offender);
|
|
}
|
|
|
|
Staking::<T>::start_session(0);
|
|
|
|
let id_tuples = offenders.iter()
|
|
.map(|offender| <
|
|
<T as ImOnlineConfig>::ValidatorSet as ValidatorSet<T::AccountId>
|
|
>::ValidatorIdOf::convert(offender.controller.clone())
|
|
.expect("failed to get validator id from account id"))
|
|
.map(|validator_id| <
|
|
<T as ImOnlineConfig>::ValidatorSet as ValidatorSetWithIdentification<T::AccountId>
|
|
>::IdentificationOf::convert(validator_id.clone())
|
|
.map(|full_id| (validator_id, full_id))
|
|
.expect("failed to convert validator id to full identification"))
|
|
.collect::<Vec<pallet_im_online::IdentificationTuple<T>>>();
|
|
Ok((id_tuples, offenders))
|
|
}
|
|
|
|
#[cfg(test)]
|
|
fn check_events<T: Config, I: Iterator<Item = <T as SystemConfig>::Event>>(expected: I) {
|
|
let events = System::<T>::events() .into_iter()
|
|
.map(|frame_system::EventRecord { event, .. }| event).collect::<Vec<_>>();
|
|
let expected = expected.collect::<Vec<_>>();
|
|
let lengths = (events.len(), expected.len());
|
|
let length_mismatch = if lengths.0 != lengths.1 {
|
|
fn pretty<D: std::fmt::Debug>(header: &str, ev: &[D]) {
|
|
println!("{}", header);
|
|
for (idx, ev) in ev.iter().enumerate() {
|
|
println!("\t[{:04}] {:?}", idx, ev);
|
|
}
|
|
}
|
|
pretty("--Got:", &events);
|
|
pretty("--Expected:", &expected);
|
|
format!("Mismatching length. Got: {}, expected: {}", lengths.0, lengths.1)
|
|
} else { Default::default() };
|
|
|
|
for (idx, (a, b)) in events.into_iter().zip(expected).enumerate() {
|
|
assert_eq!(a, b, "Mismatch at: {}. {}", idx, length_mismatch);
|
|
}
|
|
|
|
if !length_mismatch.is_empty() {
|
|
panic!("{}", length_mismatch);
|
|
}
|
|
}
|
|
|
|
benchmarks! {
|
|
report_offence_im_online {
|
|
let r in 1 .. MAX_REPORTERS;
|
|
// we skip 1 offender, because in such case there is no slashing
|
|
let o in 2 .. MAX_OFFENDERS;
|
|
let n in 0 .. MAX_NOMINATORS.min(MAX_NOMINATIONS as u32);
|
|
|
|
// Make r reporters
|
|
let mut reporters = vec![];
|
|
for i in 0 .. r {
|
|
let reporter = account("reporter", i, SEED);
|
|
reporters.push(reporter);
|
|
}
|
|
|
|
// make sure reporters actually get rewarded
|
|
Staking::<T>::set_slash_reward_fraction(Perbill::one());
|
|
|
|
let (offenders, raw_offenders) = make_offenders_im_online::<T>(o, n)?;
|
|
let keys = ImOnline::<T>::keys();
|
|
let validator_set_count = keys.len() as u32;
|
|
|
|
let slash_fraction = UnresponsivenessOffence::<T::AccountId>::slash_fraction(
|
|
offenders.len() as u32, validator_set_count,
|
|
);
|
|
let offence = UnresponsivenessOffence {
|
|
session_index: 0,
|
|
validator_set_count,
|
|
offenders,
|
|
};
|
|
assert_eq!(System::<T>::event_count(), 0);
|
|
}: {
|
|
let _ = <T as ImOnlineConfig>::ReportUnresponsiveness::report_offence(
|
|
reporters.clone(),
|
|
offence
|
|
);
|
|
}
|
|
verify {
|
|
// make sure the report was not deferred
|
|
assert!(Offences::<T>::deferred_offences().is_empty());
|
|
let bond_amount: u32 = UniqueSaturatedInto::<u32>::unique_saturated_into(bond_amount::<T>());
|
|
let slash_amount = slash_fraction * bond_amount;
|
|
let reward_amount = slash_amount * (1 + n) / 2;
|
|
let mut slash_events = raw_offenders.into_iter()
|
|
.flat_map(|offender| {
|
|
core::iter::once(offender.stash).chain(offender.nominator_stashes.into_iter())
|
|
})
|
|
.map(|stash| <T as StakingConfig>::Event::from(
|
|
StakingEvent::<T>::Slash(stash, BalanceOf::<T>::from(slash_amount))
|
|
))
|
|
.collect::<Vec<_>>();
|
|
let reward_events = reporters.into_iter()
|
|
.flat_map(|reporter| vec![
|
|
frame_system::Event::<T>::NewAccount(reporter.clone()).into(),
|
|
<T as BalancesConfig>::Event::from(
|
|
pallet_balances::Event::<T>::Endowed(reporter, (reward_amount / r).into())
|
|
).into()
|
|
]);
|
|
|
|
// rewards are applied after first offender and it's nominators
|
|
let slash_rest = slash_events.split_off(1 + n as usize);
|
|
|
|
// make sure that all slashes have been applied
|
|
#[cfg(test)]
|
|
check_events::<T, _>(
|
|
std::iter::empty()
|
|
.chain(slash_events.into_iter().map(Into::into))
|
|
.chain(reward_events)
|
|
.chain(slash_rest.into_iter().map(Into::into))
|
|
.chain(std::iter::once(<T as OffencesConfig>::Event::from(
|
|
pallet_offences::Event::Offence(
|
|
UnresponsivenessOffence::<T>::ID,
|
|
0_u32.to_le_bytes().to_vec(),
|
|
true
|
|
)
|
|
).into()))
|
|
);
|
|
}
|
|
|
|
report_offence_grandpa {
|
|
let n in 0 .. MAX_NOMINATORS.min(MAX_NOMINATIONS as u32);
|
|
|
|
// for grandpa equivocation reports the number of reporters
|
|
// and offenders is always 1
|
|
let reporters = vec![account("reporter", 1, SEED)];
|
|
|
|
// make sure reporters actually get rewarded
|
|
Staking::<T>::set_slash_reward_fraction(Perbill::one());
|
|
|
|
let (mut offenders, raw_offenders) = make_offenders::<T>(1, n)?;
|
|
let keys = ImOnline::<T>::keys();
|
|
|
|
let offence = GrandpaEquivocationOffence {
|
|
time_slot: GrandpaTimeSlot { set_id: 0, round: 0 },
|
|
session_index: 0,
|
|
validator_set_count: keys.len() as u32,
|
|
offender: T::convert(offenders.pop().unwrap()),
|
|
};
|
|
assert_eq!(System::<T>::event_count(), 0);
|
|
}: {
|
|
let _ = Offences::<T>::report_offence(reporters, offence);
|
|
}
|
|
verify {
|
|
// make sure the report was not deferred
|
|
assert!(Offences::<T>::deferred_offences().is_empty());
|
|
// make sure that all slashes have been applied
|
|
assert_eq!(
|
|
System::<T>::event_count(), 0
|
|
+ 1 // offence
|
|
+ 2 // reporter (reward + endowment)
|
|
+ 1 // offenders slashed
|
|
+ n // nominators slashed
|
|
);
|
|
}
|
|
|
|
report_offence_babe {
|
|
let n in 0 .. MAX_NOMINATORS.min(MAX_NOMINATIONS as u32);
|
|
|
|
// for babe equivocation reports the number of reporters
|
|
// and offenders is always 1
|
|
let reporters = vec![account("reporter", 1, SEED)];
|
|
|
|
// make sure reporters actually get rewarded
|
|
Staking::<T>::set_slash_reward_fraction(Perbill::one());
|
|
|
|
let (mut offenders, raw_offenders) = make_offenders::<T>(1, n)?;
|
|
let keys = ImOnline::<T>::keys();
|
|
|
|
let offence = BabeEquivocationOffence {
|
|
slot: 0u64.into(),
|
|
session_index: 0,
|
|
validator_set_count: keys.len() as u32,
|
|
offender: T::convert(offenders.pop().unwrap()),
|
|
};
|
|
assert_eq!(System::<T>::event_count(), 0);
|
|
}: {
|
|
let _ = Offences::<T>::report_offence(reporters, offence);
|
|
}
|
|
verify {
|
|
// make sure the report was not deferred
|
|
assert!(Offences::<T>::deferred_offences().is_empty());
|
|
// make sure that all slashes have been applied
|
|
assert_eq!(
|
|
System::<T>::event_count(), 0
|
|
+ 1 // offence
|
|
+ 2 // reporter (reward + endowment)
|
|
+ 1 // offenders slashed
|
|
+ n // nominators slashed
|
|
);
|
|
}
|
|
|
|
on_initialize {
|
|
let d in 1 .. MAX_DEFERRED_OFFENCES;
|
|
let o = 10;
|
|
let n = 100;
|
|
|
|
Staking::<T>::put_election_status(ElectionStatus::Closed);
|
|
|
|
let mut deferred_offences = vec![];
|
|
let offenders = make_offenders::<T>(o, n)?.0;
|
|
let offence_details = offenders.into_iter()
|
|
.map(|offender| OffenceDetails {
|
|
offender: T::convert(offender),
|
|
reporters: vec![],
|
|
})
|
|
.collect::<Vec<_>>();
|
|
|
|
for i in 0 .. d {
|
|
let fractions = offence_details.iter()
|
|
.map(|_| Perbill::from_percent(100 * (i + 1) / MAX_DEFERRED_OFFENCES))
|
|
.collect::<Vec<_>>();
|
|
deferred_offences.push((offence_details.clone(), fractions.clone(), 0u32));
|
|
}
|
|
|
|
Offences::<T>::set_deferred_offences(deferred_offences);
|
|
assert!(!Offences::<T>::deferred_offences().is_empty());
|
|
}: {
|
|
Offences::<T>::on_initialize(0u32.into());
|
|
}
|
|
verify {
|
|
// make sure that all deferred offences were reported with Ok status.
|
|
assert!(Offences::<T>::deferred_offences().is_empty());
|
|
assert_eq!(
|
|
System::<T>::event_count(), d * (0
|
|
+ o // offenders slashed
|
|
+ o * n // nominators slashed
|
|
));
|
|
}
|
|
}
|
|
|
|
impl_benchmark_test_suite!(
|
|
Module,
|
|
crate::mock::new_test_ext(),
|
|
crate::mock::Test,
|
|
);
|