mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-30 06:07:25 +00:00
44d5aba80d
* Create a macro which automates creation of benchmark test suites. * bump impl_version * allow unused on test_bench_by_name * use proper doctest ignore attribute * Explicitly hand the Module to the test suite Much better practice than depending on it showing up implicitly in the namespace. * explicitly import what we need into `mod tests` * bench_module is `ident` not `tt` Co-authored-by: Guillaume Thiolliere <gui.thiolliere@gmail.com> * allow end users to specify arguments for new_test_ext This turned out to be surprisingly easy. On reflection, it turns out that of course the compiler can't eagerly evaluate the function call, but needs to paste it in everywhere desired. * enable explicitly specifying the path to the benchmarks invocation also enable optional trailing commas * Revert "bump impl_version" This reverts commit 0209e4de33fd43873f8cfc6875815d0fd6151e63. * list failing benchmark tests and the errors which caused the failure * harden benchmark tests against internal panics * suppress warning about ignored profiles unfortunately, setting the profile here doesn't do anything; we'd need to set it in every leaf package anyway. However, as this was just making the default explicit anyway, I think it's safe enough to remove entirely. * impl_benchmark_test_suite for assets * impl_benchmark_test_suite for balances * impl_benchmark_test_suite for bounties * impl_benchmark_test_suite for Collective * impl_benchmark_test_suite for Contracts * impl_benchmark_test_suite for Democracy * don't impl_benchmark_test_suite for Elections-Phragmen * impl_benchmark_test_suite for Identity Note that Identity tests currently fail. They failed in an identical way before this change, so as far as I'm concerned, the status quo is good enough for now. * impl_benchmark_test_suite for ImOnline * impl_benchmark_test_suite for indices For this crate also, the test suite fails identically with and without this change, so we can say that this change is not the cause of the tests' failure to compile. * impl_benchmark_test_suite for lottery * impl_benchmark_test_suite for merkle-mountain-range * impl_benchmark_test_suite for Multisig These tests fail identically with and without the change, so the change seems unlikely to be the origin of the failures. * impl_benchmark_test_suite for offences * impl_benchmark_test_suite for Proxy Fails identically with and without this change. * impl_benchmark_test_suite for scheduler * impl_benchmark_test_suite for session It turns out to be important to be able to exclude items marked `#[extra]` sometimes. Who knew? * impl_benchmark_test_suite for staking * impl_benchmark_test_suite for system * impl_benchmark_test_suite for timestamp * impl_benchmark_test_suite for tips * impl_benchmark_test_suite for treasury * impl_benchmark_test_suite for utility Note that benchmark tests fail identically before and after this change. * impl_benchmark_test_suite for vesting * fix wrong module name in impl_benchmark_test_suite in Offences * address line length nits * enable optional keyword argument: exec_name Took a _lot_ of macro-wrangling to get the functionality that I want, but now you have the option to pass in ```rust impl_benchmark_test_suite!( Elections, crate::tests::ExtBuilder::default().desired_members(13).desired_runners_up(7), crate::tests::Test, exec_name = build_and_execute, ); ``` and have it expand out properly. A selected fragment of the expansion: ```rust fn test_benchmarks() { crate::tests::ExtBuilder::default() .desired_members(13) .desired_runners_up(7) .build_and_execute(|| { ``` * get rid of dead code Co-authored-by: Guillaume Thiolliere <gui.thiolliere@gmail.com>
178 lines
5.4 KiB
Rust
178 lines
5.4 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.
|
|
|
|
//! Benchmarks for the Session Pallet.
|
|
// This is separated into its own crate due to cyclic dependency issues.
|
|
|
|
#![cfg_attr(not(feature = "std"), no_std)]
|
|
|
|
mod mock;
|
|
|
|
use sp_std::prelude::*;
|
|
use sp_std::vec;
|
|
|
|
use frame_benchmarking::{benchmarks, impl_benchmark_test_suite};
|
|
use frame_support::{
|
|
codec::Decode,
|
|
storage::StorageValue,
|
|
traits::{KeyOwnerProofSystem, OnInitialize},
|
|
};
|
|
use frame_system::RawOrigin;
|
|
use pallet_session::{historical::Module as Historical, Module as Session, *};
|
|
use pallet_staking::{
|
|
benchmarking::create_validator_with_nominators, testing_utils::create_validators,
|
|
MAX_NOMINATIONS, RewardDestination,
|
|
};
|
|
use sp_runtime::traits::{One, StaticLookup};
|
|
|
|
const MAX_VALIDATORS: u32 = 1000;
|
|
|
|
pub struct Module<T: Config>(pallet_session::Module<T>);
|
|
pub trait Config: pallet_session::Config + pallet_session::historical::Config + pallet_staking::Config {}
|
|
|
|
impl<T: Config> OnInitialize<T::BlockNumber> for Module<T> {
|
|
fn on_initialize(n: T::BlockNumber) -> frame_support::weights::Weight {
|
|
pallet_session::Module::<T>::on_initialize(n)
|
|
}
|
|
}
|
|
|
|
benchmarks! {
|
|
set_keys {
|
|
let n = MAX_NOMINATIONS as u32;
|
|
let (v_stash, _) = create_validator_with_nominators::<T>(
|
|
n,
|
|
MAX_NOMINATIONS as u32,
|
|
false,
|
|
RewardDestination::Staked,
|
|
)?;
|
|
let v_controller = pallet_staking::Module::<T>::bonded(&v_stash).ok_or("not stash")?;
|
|
let keys = T::Keys::default();
|
|
let proof: Vec<u8> = vec![0,1,2,3];
|
|
// Whitelist controller account from further DB operations.
|
|
let v_controller_key = frame_system::Account::<T>::hashed_key_for(&v_controller);
|
|
frame_benchmarking::benchmarking::add_to_whitelist(v_controller_key.into());
|
|
}: _(RawOrigin::Signed(v_controller), keys, proof)
|
|
|
|
purge_keys {
|
|
let n = MAX_NOMINATIONS as u32;
|
|
let (v_stash, _) = create_validator_with_nominators::<T>(
|
|
n,
|
|
MAX_NOMINATIONS as u32,
|
|
false,
|
|
RewardDestination::Staked
|
|
)?;
|
|
let v_controller = pallet_staking::Module::<T>::bonded(&v_stash).ok_or("not stash")?;
|
|
let keys = T::Keys::default();
|
|
let proof: Vec<u8> = vec![0,1,2,3];
|
|
Session::<T>::set_keys(RawOrigin::Signed(v_controller.clone()).into(), keys, proof)?;
|
|
// Whitelist controller account from further DB operations.
|
|
let v_controller_key = frame_system::Account::<T>::hashed_key_for(&v_controller);
|
|
frame_benchmarking::benchmarking::add_to_whitelist(v_controller_key.into());
|
|
}: _(RawOrigin::Signed(v_controller))
|
|
|
|
#[extra]
|
|
check_membership_proof_current_session {
|
|
let n in 2 .. MAX_VALIDATORS as u32;
|
|
|
|
let (key, key_owner_proof1) = check_membership_proof_setup::<T>(n);
|
|
let key_owner_proof2 = key_owner_proof1.clone();
|
|
}: {
|
|
Historical::<T>::check_proof(key, key_owner_proof1);
|
|
}
|
|
verify {
|
|
assert!(Historical::<T>::check_proof(key, key_owner_proof2).is_some());
|
|
}
|
|
|
|
#[extra]
|
|
check_membership_proof_historical_session {
|
|
let n in 2 .. MAX_VALIDATORS as u32;
|
|
|
|
let (key, key_owner_proof1) = check_membership_proof_setup::<T>(n);
|
|
|
|
// skip to the next session so that the session is historical
|
|
// and the membership merkle proof must be checked.
|
|
Session::<T>::rotate_session();
|
|
|
|
let key_owner_proof2 = key_owner_proof1.clone();
|
|
}: {
|
|
Historical::<T>::check_proof(key, key_owner_proof1);
|
|
}
|
|
verify {
|
|
assert!(Historical::<T>::check_proof(key, key_owner_proof2).is_some());
|
|
}
|
|
}
|
|
|
|
/// Sets up the benchmark for checking a membership proof. It creates the given
|
|
/// number of validators, sets random session keys and then creates a membership
|
|
/// proof for the first authority and returns its key and the proof.
|
|
fn check_membership_proof_setup<T: Config>(
|
|
n: u32,
|
|
) -> (
|
|
(sp_runtime::KeyTypeId, &'static [u8; 32]),
|
|
sp_session::MembershipProof,
|
|
) {
|
|
pallet_staking::ValidatorCount::put(n);
|
|
|
|
// create validators and set random session keys
|
|
for (n, who) in create_validators::<T>(n, 1000)
|
|
.unwrap()
|
|
.into_iter()
|
|
.enumerate()
|
|
{
|
|
use rand::RngCore;
|
|
use rand::SeedableRng;
|
|
|
|
let validator = T::Lookup::lookup(who).unwrap();
|
|
let controller = pallet_staking::Module::<T>::bonded(validator).unwrap();
|
|
|
|
let keys = {
|
|
let mut keys = [0u8; 128];
|
|
|
|
// we keep the keys for the first validator as 0x00000...
|
|
if n > 0 {
|
|
let mut rng = rand::rngs::StdRng::seed_from_u64(n as u64);
|
|
rng.fill_bytes(&mut keys);
|
|
}
|
|
|
|
keys
|
|
};
|
|
|
|
let keys: T::Keys = Decode::decode(&mut &keys[..]).unwrap();
|
|
let proof: Vec<u8> = vec![];
|
|
|
|
Session::<T>::set_keys(RawOrigin::Signed(controller).into(), keys, proof).unwrap();
|
|
}
|
|
|
|
Module::<T>::on_initialize(T::BlockNumber::one());
|
|
|
|
// skip sessions until the new validator set is enacted
|
|
while Session::<T>::validators().len() < n as usize {
|
|
Session::<T>::rotate_session();
|
|
}
|
|
|
|
let key = (sp_runtime::KeyTypeId(*b"babe"), &[0u8; 32]);
|
|
|
|
(key, Historical::<T>::prove(key).unwrap())
|
|
}
|
|
|
|
impl_benchmark_test_suite!(
|
|
Module,
|
|
crate::mock::new_test_ext(),
|
|
crate::mock::Test,
|
|
extra = false,
|
|
);
|