// Copyright 2020 Parity Technologies (UK) Ltd.
// This file is part of Substrate.
// Substrate is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// Substrate is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with Substrate. If not, see .
//! Fuzzing for staking pallet.
#![no_main]
use libfuzzer_sys::fuzz_target;
use mock::Test;
use pallet_staking::testing_utils::{
self, USER, get_seq_phragmen_solution, get_weak_solution, setup_chain_stakers,
set_validator_count, signed_account,
};
use frame_support::assert_ok;
use sp_runtime::{traits::Dispatchable, DispatchError};
mod mock;
#[repr(u32)]
#[allow(dead_code)]
#[derive(Debug, Clone, Copy)]
enum Mode {
/// Initial submission. This will be rather cheap.
InitialSubmission,
/// A better submission that will replace the previous ones. This is the most expensive.
StrongerSubmission,
/// A weak submission that will be rejected. This will be rather cheap.
WeakerSubmission,
}
pub fn new_test_ext() -> Result {
frame_system::GenesisConfig::default().build_storage::().map(Into::into)
}
fuzz_target!(|do_reduce: bool| {
let ext = new_test_ext();
let mode: Mode = unsafe { std::mem::transmute(testing_utils::random(0, 2)) };
let num_validators = testing_utils::random(50, 500);
let num_nominators = testing_utils::random(200, 2000);
let edge_per_voter = testing_utils::random(1, 16);
let to_elect = testing_utils::random(10, num_validators);
println!("+++ instance with params {} / {} / {} / {:?} / {}",
num_nominators,
num_validators,
edge_per_voter,
mode,
to_elect,
);
ext.unwrap_or_default().execute_with(|| {
// initial setup
set_validator_count::(to_elect);
setup_chain_stakers::(
num_validators,
num_nominators,
edge_per_voter,
);
println!("++ Chain setup done.");
// stuff to submit
let (winners, compact, score) = match mode {
Mode::InitialSubmission => {
/* No need to setup anything */
get_seq_phragmen_solution::(do_reduce)
},
Mode::StrongerSubmission => {
let (winners, compact, score) = get_weak_solution::(false);
assert_ok!(
>::submit_election_solution(
signed_account::(USER),
winners,
compact,
score,
)
);
get_seq_phragmen_solution::(do_reduce)
},
Mode::WeakerSubmission => {
let (winners, compact, score) = get_seq_phragmen_solution::(do_reduce);
assert_ok!(
>::submit_election_solution(
signed_account::(USER),
winners,
compact,
score,
)
);
get_weak_solution::(false)
}
};
println!("++ Submission ready.");
// must have chosen correct number of winners.
assert_eq!(winners.len() as u32, >::validator_count());
// final call and origin
let call = pallet_staking::Call::::submit_election_solution(
winners,
compact,
score,
);
let caller = signed_account::(USER);
// actually submit
match mode {
Mode::WeakerSubmission => {
assert_eq!(
call.dispatch(caller.into()).unwrap_err(),
DispatchError::Module { index: 0, error: 11, message: Some("PhragmenWeakSubmission") },
);
},
_ => assert_ok!(call.dispatch(caller.into())),
};
})
});