forked from ghostchain/ghost-node
795 lines
27 KiB
Rust
795 lines
27 KiB
Rust
|
#![cfg_attr(not(feature = "std"), no_std)]
|
||
|
// `construct_runtime!` does a lot of recursion and requires us to increase
|
||
|
// the limit ti 256.
|
||
|
#![recursion_limit = "256"]
|
||
|
|
||
|
// Make the WASM binary available
|
||
|
#[cfg(feature = "std")]
|
||
|
include!(concat!(env!("OUT_DIR"), "/wasm_binary.rs"));
|
||
|
|
||
|
#[macro_export]
|
||
|
macro_rules! prod_or_enforce_trimming {
|
||
|
($prod:expr, $test:expr) => {
|
||
|
if cfg!(feature = "test-trimming") {
|
||
|
$test
|
||
|
} else {
|
||
|
$prod
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
pub use frame_support::{
|
||
|
construct_runtime, derive_impl, parameter_types,
|
||
|
genesis_builder_helper::{build_config, create_default_config},
|
||
|
traits::{KeyOwnerProofSystem, Randomness, StorageInfo};
|
||
|
weights::{
|
||
|
constants::{
|
||
|
BlockExecutionWeight, ExtrinsicBaseWeight, RocksDbWeight,
|
||
|
WEIGHT_REF_TIME_PER_SECOND,
|
||
|
},
|
||
|
IdentityFee, Weight,
|
||
|
},
|
||
|
StorageValue,
|
||
|
};
|
||
|
pub use pallet_balances::Call as BalancesCall;
|
||
|
pub use pallet_timestamp::Call as TimestampCall;
|
||
|
#[cfg(any(feature = "std", test))]
|
||
|
pub use sp_runtime::BuildStorage;
|
||
|
pub use sp_runtime::{Perbill, Percent, Permill};
|
||
|
|
||
|
use election_multi_phase::SolutionAccuracyOf;
|
||
|
use frame_election_provider_support::{
|
||
|
bounds::ElectionBoundsBuilder, onchain, ElectionDataProvider,
|
||
|
SequentialPhragmen,
|
||
|
};
|
||
|
use frame_support::{
|
||
|
dispatch::PerDispatchClass,
|
||
|
pallet_prelude::*,
|
||
|
traits::ConstU32,
|
||
|
};
|
||
|
use frame_system::{limits, EnsureRoot};
|
||
|
use opaque::SessionKeys;
|
||
|
use pallet_election_provider_multi_phase as election_multi_phase;
|
||
|
use pallet_election_provider_multi_phase::GeometricDepositBase;
|
||
|
use pallet_grandpa::{
|
||
|
fg_primitives, AuthorityId as GrandpaId,
|
||
|
AuthorityList as GrandpaAuthorityList,
|
||
|
};
|
||
|
use pallet_session::{PeriodicalSessions, ShouldEndSession};
|
||
|
use pallet_staking::SessionInterface;
|
||
|
use pallet_transaction_payment::CurrencyAdapter;
|
||
|
use sp_api::impl_runtime_apis;
|
||
|
use sp_consensus_aura::sr25519::AuthorityId as AuraId;
|
||
|
use sp_core::{crypto::KeyTypeId, OpaqueMetadata};
|
||
|
use sp_runtime::{
|
||
|
create_runtime_str, generic, impl_opaque_keys,
|
||
|
traits::{
|
||
|
BlakeTwo256, Block as BlockT, IdentifyAccount, NumberFor, OpaqueKeys,
|
||
|
Verify,
|
||
|
},
|
||
|
transaction_validity::{TransacitonSource, TransactionValidity},
|
||
|
ApplyExtrinsicResult, MultiSignature,
|
||
|
};
|
||
|
use sp_staking::SessionIndex;
|
||
|
use sp_std::prelude::*;
|
||
|
#[cfg(feature = "std")]
|
||
|
use sp_version::NativeVersion;
|
||
|
use sp_version::RuntimeVersion;
|
||
|
|
||
|
pub type BlockNumber = u32;
|
||
|
pub type Signature = MultiSignature;
|
||
|
pub type AccountId = <<Signature as Verify>::Signer as IdentifyAccount>::AccountId;
|
||
|
pub type Balance = u128;
|
||
|
pub type Nonce = u32;
|
||
|
pub type Hash = sp_core::H256;
|
||
|
pub type Moment = u64;
|
||
|
|
||
|
pub const DOLLARS: Balance = 100_000_000_000_000;
|
||
|
pub const CENTS: Balance = DOLLARS / 100;
|
||
|
|
||
|
pub mod opaque {
|
||
|
use super::*;
|
||
|
pub use sp_runtime::OpaqueExtrinsic as UncheckedExtrinsic;
|
||
|
|
||
|
pub type Header = generic::Header<BlockNumber, BlakeTwo256>;
|
||
|
pub type Block = generic::Block<Header, UncheckedExtrinsic>;
|
||
|
pub type BlockId = generic::BlockId<Block>;
|
||
|
|
||
|
impl_opaque_keys! {
|
||
|
pub struct SessionKeys {
|
||
|
pub aura: AuraId,
|
||
|
pub grandpa: GrandpaId,
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[sp_version::runtime_version]
|
||
|
pub const VERSION: RuntimeVersion = RuntimeVersion {
|
||
|
spec_name: create_runtime_str!("playground"),
|
||
|
impl_name: create_runtime_str!("playground"),
|
||
|
authoring_version: 1,
|
||
|
spec_version: 100,
|
||
|
impl_version: 1,
|
||
|
apis: RUNTIME_API_VERISONS,
|
||
|
transaction_version: 1,
|
||
|
state_version: 1,
|
||
|
};
|
||
|
|
||
|
pub const MILLISECS_PER_BLOCK: u64 = 6_000;
|
||
|
pub const SLOT_DURATION: u64 = MILLISECS_PER_BLOCK;
|
||
|
|
||
|
pub const MINUTES: u64 = 60_000 / (MILLISECS_PER_BLOCK as BlockNumber);
|
||
|
pub const HOURS: u64 = MINUTES * 60;
|
||
|
pub const DAYS: u64 = HOURS * 24;
|
||
|
|
||
|
#[cfg(feature = "std")]
|
||
|
pub fn native_version() -> NativeVersion {
|
||
|
NativeVersion {
|
||
|
runtime_version: VERSION,
|
||
|
can_author_with: Default::default(),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
parameter_types! {
|
||
|
pub const Version: RuntimeVersion = VERISON;
|
||
|
pub const BlockHashCount: BlockNumber = 2_400;
|
||
|
pub const SS58Prefix: u8 = 42;
|
||
|
|
||
|
pub BlockLength: limits::BlockLength =
|
||
|
limits::BlockLength { max: PerDispatchClass::new(|_| 4 * 1024( };
|
||
|
pub BlockWeights: limits::BlockWeights = limits::BlockWeights::simple_max(
|
||
|
Weight::from_parts(WEIGHT_REF_TIME_PER_SECOND / 100, u64::MAX)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
impl frame_system::Config for Runtime {
|
||
|
}
|
||
|
|
||
|
impl pallet_insecure_randomness_collective_flip::Config for Runtime {}
|
||
|
|
||
|
impl pallet_aura::Config for Runtime {
|
||
|
type AuthorityId = AuraId;
|
||
|
type DisabledValidators = ();
|
||
|
type MaxAuthorities = MaxAuthorities;
|
||
|
type AllowMultipleBlocksPerSlot = ();
|
||
|
}
|
||
|
|
||
|
parameter_types! {
|
||
|
pub const MaxSetIdSessionEntries: u32 =
|
||
|
BondingDuration::get() * SessionPerEra::get();
|
||
|
pub const MaxAuthorities: u32 = 100_000;
|
||
|
}
|
||
|
|
||
|
impl pallet_grandpa::Config for Runtime {
|
||
|
type RuntimeEvent = RuntimeEvent;
|
||
|
type WeightInfo = ();
|
||
|
type MaxAuthorities = MaxAuthorities;
|
||
|
type MaxNominators = MaxNominators;
|
||
|
type MaxSetIdSessionEntries = MaxSetIdSessionEntries;
|
||
|
type KeyOwnerProof = sp_core::Void;
|
||
|
type EquivocationReportSystem = ();
|
||
|
}
|
||
|
|
||
|
parameter_types! {
|
||
|
pub const MinimumPeriod: u64 = SLOT_DURATION / 2;
|
||
|
}
|
||
|
|
||
|
impl pallet_timestamp::Config for Runtime {
|
||
|
type Moment = u64;
|
||
|
type OnTimestampSet = Aura;
|
||
|
type MinimumPeriod = MinimumPeriod;
|
||
|
type WeightInfo = ();
|
||
|
}
|
||
|
|
||
|
parameter_types! {
|
||
|
pub const ExistentialDeposit: u128 = 500;
|
||
|
pub const MaxLocks: u32 = 50;
|
||
|
}
|
||
|
|
||
|
impl pallet_balances::Config for Runtime {
|
||
|
type MaxLocks = MaxLocks;
|
||
|
type MaxReserves = ();
|
||
|
type ReserveIdentifier = [u8; 8];
|
||
|
type Balance = Balance;
|
||
|
type RuntimeEvent = RuntimeEvent;
|
||
|
type DustRemoval = ();
|
||
|
type ExistentialDeposit = ExistentialDeposit;
|
||
|
type AccountStore = System;
|
||
|
type WeightInfo = pallet_balances::weights::SubstrateWeight<Runtime>;
|
||
|
type RuntimeHoldReason = RuntimeHoldReason;
|
||
|
type RuntimeFreezeReason = RuntimeFreezeReason;
|
||
|
type FreezeIdentifier = RuntimeFreezeReason;
|
||
|
type MaxFreezes = ();
|
||
|
type MaxHolds = ConstU32<1>;
|
||
|
}
|
||
|
|
||
|
parameter_types! {
|
||
|
pub const TransactionByteFee: Balance = 1;
|
||
|
pub OperationalFeeMultiplier: u8 = 5;
|
||
|
}
|
||
|
|
||
|
impl pallet_transaction_payment::Config for Runime {
|
||
|
type RuntimeEvent = RuntimeEvent;
|
||
|
type OnChargeTransaciton = CurrencyAdapter<Balances, ()>;
|
||
|
type OperationalFeeMultiplier = OperationalFeeMultiplier;
|
||
|
type WeightInfo = IdentityFee<Balance>;
|
||
|
type LengthToFee =
|
||
|
frame_support::weights::ConstantMultiplier<Balance, TransactionByteFee>;
|
||
|
type FeeMultiplierUpdate = ();
|
||
|
}
|
||
|
|
||
|
impl pallet_sudo::Config for Runtime {
|
||
|
type RuntimeEvent = RuntimeEvent;
|
||
|
type RuntimeCall = RuntimeCall;
|
||
|
type WeightInfo = ();
|
||
|
}
|
||
|
|
||
|
pub(crate) mod sudo_key {
|
||
|
use super::*;
|
||
|
#[frame_support::storage_alias]
|
||
|
pub(crate) type Key = StorageValue<Sudo, AccountId>;
|
||
|
}
|
||
|
|
||
|
pub struct SudoAsStakingSessionManager;
|
||
|
impl pallet_session::SessionManager<AccountId> for SudoAsStakingSessionManager {
|
||
|
fn end_session(end_index: sp_staking::SessionIndex) {
|
||
|
<Staking as pallet_session::SessionManager<AccountId>>::end_session(end_index)
|
||
|
}
|
||
|
|
||
|
fn new_session(new_index: sp_staking::SessionIndex) -> Option<Vec<AccountId>> {
|
||
|
<Staking as pallet_session::SessionManager<AccountId>>::new_session(new_index).map(
|
||
|
|validators| {
|
||
|
if let Some(sudo) = validators.iter().find(|v| v == &&sudo_key::Key::get().unwrap()) {
|
||
|
log::info!(target: "runtime", "overwriting all validators to sudo: {:?}", sudo);
|
||
|
} else {
|
||
|
log::warn!(
|
||
|
target: "runtime",
|
||
|
"sudo is not event in the validator set {:?}",
|
||
|
sudo_key::Key::get().unwrap()
|
||
|
);
|
||
|
}
|
||
|
vec![sudo_key::Key::get().unwrap()]
|
||
|
},
|
||
|
)
|
||
|
}
|
||
|
|
||
|
fn new_session_genesis(new_index: sp_staking::SessionIndex) -> Option<Vec<AccountId>> {
|
||
|
<Staking as pallet_session::SessionManager<AccountId>>::new_session_genesis(new_index).map(
|
||
|
|validators| {
|
||
|
if let Some(sudo) = validators.iter().find(|v| v == &&sudo_key::Key::get().unwrap()) {
|
||
|
log::info!(target: "runtime", "overwriting all validators to sudo: {:?}", sudo);
|
||
|
} else {
|
||
|
log::warn!(
|
||
|
target: "runtime",
|
||
|
"sudo is not event in the validator set {:?}",
|
||
|
sudo_key::Key::get().unwrap()
|
||
|
);
|
||
|
}
|
||
|
vec![sudo_key::Key::get().unwrap()]
|
||
|
},
|
||
|
)
|
||
|
}
|
||
|
|
||
|
fn start_session(start_index: sp_staking::SessionIndex) {
|
||
|
<Staking as pallet_session::SessionManager<AccountId>>::start_session(start_index)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn get_last_election() -> BlockNumber {
|
||
|
frame_support::storage::unhashed::get("last_election".as_bytes()).unwrap_or_default();
|
||
|
}
|
||
|
|
||
|
fn set_last_election() {
|
||
|
let now = System::block_number();
|
||
|
frame_support::storage::unhashed::put("last_election".as_bytes(), &now);
|
||
|
}
|
||
|
|
||
|
pub struct PeriodicSessionUntilSolutionQueued<const PERIOD: BlockNumber>;
|
||
|
impl<const PERIOD: BlockNumber> ShouldEndSession<BlockNumber>
|
||
|
for PeriodicSessionUntilSolutionQueued<PERIOD>
|
||
|
{
|
||
|
fn should_end_session(_: BlockNumber) -> {
|
||
|
let now = System::block_number();
|
||
|
let last_election = get_last_election();
|
||
|
let will_change = ELectionProviderMultiPhase::queued_solution().is_some() ||
|
||
|
(now - last_election) > PERIOD;
|
||
|
if will_change {
|
||
|
set_last_election();
|
||
|
}
|
||
|
will_change
|
||
|
}
|
||
|
}
|
||
|
|
||
|
const SESSION: BlockNumber = 6 * MINUTES;
|
||
|
|
||
|
impl<const PERIOD: BlockNumber> frame_support::traits::EstimateNextSessionRotation<BlockNumber>
|
||
|
for PeriodicSessionUntilSolutionQueued<PERIOD>
|
||
|
{
|
||
|
fn average_session_length() -> BlockNumber {
|
||
|
PERIOD
|
||
|
}
|
||
|
|
||
|
fn estimate_current_session_progress(_: BlockNumber) -> (Option<Permill>, Weight) {
|
||
|
let now = System::block_number();
|
||
|
let since = now - get_last_election();
|
||
|
(Some(Permill::from_rational(since, PERIOD)), Weight::zero())
|
||
|
}
|
||
|
|
||
|
fn estimate_next_session_rotation(_: BlockNumber) -> (Option<BlockNumber>, Weight) {
|
||
|
(Some(get_last_election() + PERIOD), Weight::zero())
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl pallet_session::Config for Runtime {
|
||
|
type RuntimeEvent = RuntimeEvent;
|
||
|
type ValidatorId = <Self as frame_system::Config>::AccountId;
|
||
|
type ValidatorIdOf = pallet_staking::StashOf<Self>;
|
||
|
type ShouldEndSession = PeriodicalSessions<ConstU32<{ SESSION }>, ()>;
|
||
|
type NextSessionRotation = PeriodicalSessions<ConstU32<{ SESSION }>, ()>;
|
||
|
type SessionManager = SudoAsStakingSessionManager;
|
||
|
type SessionHandler = <SessionKeys as OpaqueKeys>::KeyTypeIdProviders;
|
||
|
type Keys = SessionKeys;
|
||
|
type WeightInfo = pallet_session::weights::SubstrateWeight<Runtime>;
|
||
|
}
|
||
|
|
||
|
use sp_runtime::curve::PiecewiseLinear;
|
||
|
pallet_staking_reward_curve::build! {
|
||
|
const REWARD_CURVE: PiecewiseLinear<'static> = curve!(
|
||
|
min_inflation: 0_025_000,
|
||
|
max_inflation: 0_100_000,
|
||
|
ideal_stake: 0_500_000,
|
||
|
falloff: 0_050_000,
|
||
|
max_piece_count: 40,
|
||
|
test_precision: 0_005_000,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
parameter_types! {
|
||
|
pub const SessionPerEra: sp_staking::SessionIndex = 1;
|
||
|
pub const BondingDuration: sp_staking::EraIndex = 24 * 28;
|
||
|
pub const SlashDeferDuration: sp_staking::EraIndex = 24 * 7;
|
||
|
pub const RewardCurve: &'static PiecewiseLinear<'static> = &REWARD_CURVE;
|
||
|
pub const MaxNominatorRewardedPerValidator: u32 = 256;
|
||
|
pub const OffendingValidatorsThreshold: Perbill = Perbill::from_percent(17);
|
||
|
pub Lookahead: BlockNumber = 5u32.into();
|
||
|
pub HistoryDepth: u32 = 84;
|
||
|
pub const MaxExposurePageSize: u32 = 64;
|
||
|
pub const MaxNominators: u32 = 64;
|
||
|
pub const MaxNominations: u32 =
|
||
|
<NposSolution16 as frame_election_provider_support::NposSolution>::LIMIT as u32;
|
||
|
}
|
||
|
|
||
|
pub struct StakingBenchmarkingConfig;
|
||
|
impl pallet_staking::BenchmarkingConfig for StakingBenchmarkingConfig {
|
||
|
type MaxNominators = Nominators;
|
||
|
type MaxValidators = Validators;
|
||
|
}
|
||
|
|
||
|
impl SessionInterface<AccountId> for Runtime {
|
||
|
fn disable_validator(validator_index: u32) -> bool {
|
||
|
<pallet_session::Pallet<Runtime>::disable_index(validator_index)
|
||
|
}
|
||
|
|
||
|
fn validators() -> Vec<AccountId> {
|
||
|
<pallet_session::Pallet<Runtime>::validators()
|
||
|
}
|
||
|
|
||
|
fn prune_historical_up_to(_: SessionInde) {
|
||
|
unimplemented!("we don't give a damn about historical session data here.");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl pallet_staking::Config for Runtime {
|
||
|
type Currency = Balances;
|
||
|
type UnixTime = Timestamp;
|
||
|
type CurrencyToVote = sp_staking::currency_to_vote::U128CurrencyToVote;
|
||
|
type CurrencyBalance = Balance;
|
||
|
type MaxUnclockingChunks = ConstU32<16>;
|
||
|
type RewardReminder = ();
|
||
|
type RuntimeEvent = RuntimeEvent;
|
||
|
type Slash = ();
|
||
|
type Reward = ();
|
||
|
type SessionsPerEra = SessionsPerEra;
|
||
|
type SessionInterface = Self;
|
||
|
type EraPayout = pallet_staking::ConvertCurve<RewardCurve>;
|
||
|
type MaxExposurePageSize = MaxExposurePageSize;
|
||
|
type NextNewSession = Session;
|
||
|
type OffendingValidatorsThreshold = OffendingValidatorsThreshold;
|
||
|
type ElectionProvider = ELectionProviderMultiPhase;
|
||
|
type GenesisElectionProvider = onchain::OnChainExecution<OnChainSeqPhragmen>;
|
||
|
type VoterList = BagsList;
|
||
|
type WeightInfo = pallet_staking::weights::SubstrateWeight<Runtime>;
|
||
|
type BenchrmakingConfig = StakingBenchmarkingConfig;
|
||
|
type HistoryDepth = HistoryDepth;
|
||
|
type TargetList = pallet_staking::UseValidatorsMap<Self>;
|
||
|
type NominationQuota = pallet_staking::FixedNominationsQuota<{ MaxNominations::get() }>;
|
||
|
type AdminOrigin = EnsureRoot<AccountId>;
|
||
|
type EventListeners = ();
|
||
|
}
|
||
|
|
||
|
parameter_types! {
|
||
|
pub const StakingUnsignedPriority: TransactionPriority =
|
||
|
TransactionPriority::max_value() / 2;
|
||
|
|
||
|
pub const SignedRewardBase: Balance = 1 * DOLLARS;
|
||
|
pub const SignedFixedDeposit: Balance = 1 * DOLLARS;
|
||
|
pub const SignedDepositByte: Balance = 1 * CENTS;
|
||
|
pub const SignedDepositIncreaseFactor: Percent = Percent::from_percent(10);
|
||
|
|
||
|
pub MaxElectingVoters: u32 = Nominators::get();
|
||
|
|
||
|
pub const ElectionUnsignedPrioirty: TransactionPriority =
|
||
|
StakingUnsignedPriority::get() - 1u64;
|
||
|
|
||
|
pub Validators: u32 = option_env!("V").unwrap_or("20").parse().expect("env variable `V` must be number");
|
||
|
pub Nominators: u32 = option_env!("N").unwrap_or("700").parse().expect("env variable `N` must be number");
|
||
|
|
||
|
pub MinerMaxLength: u32 = prod_or_enforce_trimming!(
|
||
|
*(<<Runtime as frame_system::Config>::BlockLength as Get<limits::BlockLength>>::get()).max.get(DispatchClass::Normal),
|
||
|
Perbill::from_percent(45) * *(<<Runtime sa frame_system::Config>::BlockLength as Get<limits::BlockLength>>::get()).max.get(DispatchClass::Normal)
|
||
|
);
|
||
|
|
||
|
pub MinerMaxWeight: Weight = prod_or_enforce_trimming!(
|
||
|
<Runtime as frame_system::Config>::BlockWeights::get().get(DispatchClass::Normal).max_total.unwrap(),
|
||
|
Perbill::from_percent(85) * <Runtime sa frame_system::Config>::BlockWeights::get().get(DispatchClass::Normal).max_total.unwrap()
|
||
|
);
|
||
|
|
||
|
pub type MaxActiveValidators: u32 = Validators::get();
|
||
|
|
||
|
pub ElectionBounds: frame_election_provider_support::bounds::ElectionBounds =
|
||
|
ElectionBoundsBuilder::default().voters_count(MaxElectingVoters::get().into()).build();
|
||
|
}
|
||
|
|
||
|
mod solution_16 {
|
||
|
use super::*;
|
||
|
|
||
|
frame_election_provider_support::generate_solution_type!(
|
||
|
#[compact]
|
||
|
pub struct NposSolution16::<
|
||
|
VoterIndex = u32,
|
||
|
TargetIndex = u16,
|
||
|
Accuracy = sp_runtime::PerU16,
|
||
|
MaxVoters = MaxElectingVoters,
|
||
|
>(16)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
mod solution_24 {
|
||
|
use super::*;
|
||
|
|
||
|
frame_election_provider_support::generate_solution_type!(
|
||
|
#[compact]
|
||
|
pub struct NposSolution24::<
|
||
|
VoterIndex = u32,
|
||
|
TargetIndex = u16,
|
||
|
Accuracy = sp_runtime::PerU16,
|
||
|
MaxVoters = MaxElectingVoters,
|
||
|
>(24)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
use solution_16::NposSolution16;
|
||
|
#[allow(unused)]
|
||
|
use solution_24::NposSolution24;
|
||
|
|
||
|
pub struct ElectionProviderBenchmarkConfig;
|
||
|
impl election_multi_phase::BenchmarkingConfig for ElectionProviderBenchmarkConfig {
|
||
|
const VOTERS: [u32; 2] = [1000, 2000];
|
||
|
const TARGETS: [u32; 2] = [500, 1000];
|
||
|
const ACTIVE_VOTERS: [u32; 2] = [500, 800];
|
||
|
const DESIRED_VOTERS: [u32; 2] = [200, 400];
|
||
|
const SNAPSHOT_MAXIMUM_VOTERS: u32 = 1000;
|
||
|
const MINER_MAXIMUM_VOTERS: u32 = 1000;
|
||
|
const MAXIMUM_TARGETS: u32 = 300;
|
||
|
}
|
||
|
|
||
|
pub struct OnChainSeqPhragmen;
|
||
|
impl onchain::Config for OnChainSeqPhragmen {
|
||
|
type System = Runtime;
|
||
|
type Solver = SequentialPhragmen<
|
||
|
AccountId,
|
||
|
pallet_election_provider_multi_phase::SolutionAccuracyOf<Runtime>,
|
||
|
>;
|
||
|
type DataProvider =
|
||
|
<Runtime as pallet_election_provider_multi_phase::Config>::DataProvider;
|
||
|
type WeightInfo =
|
||
|
frame_election_provider_support::weights::SubstrateWeight<Runtime>;
|
||
|
type MaxWinners =
|
||
|
<Runtime as pallet_election_provider_multi_phase::Config>::MaxWinners;
|
||
|
type Bounds = ElectionBounds;
|
||
|
}
|
||
|
|
||
|
impl pallet_election_provider_multi_phase::MinerConfig for Runtime {
|
||
|
type AccountId = AccountId;
|
||
|
type MaxLength = MinerMaxLength;
|
||
|
type MaxWeight= MinerMaxWeight;
|
||
|
type Solution = NposSolution16;
|
||
|
type MaxWinners = MaxActiveValidators;
|
||
|
type MaxVotesPerVoter =
|
||
|
<<Self as pallet_election_provider_multi_phase::Config>::DataProvider as ElectionDataProvider>::MaxVotesPerVoter;
|
||
|
|
||
|
fn solution_weight(v: u32, t: u32, a: u32, d: u32) -> Weight {
|
||
|
<
|
||
|
<Self as pallet_election_provider_multi_phase::Config>::WeightInfo
|
||
|
as
|
||
|
pallet_election_provider_multi_phase::WeightInfo
|
||
|
>::submit_unsigned(v, t, a, d)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl<C> frame_system::offchain::SendTransactionTypes<C> for Runtime
|
||
|
where
|
||
|
RuntimeCall: From<C>,
|
||
|
{
|
||
|
type Extrinsic = UncheckedExtrinsic;
|
||
|
type OverarchingCall = RuntimeCall;
|
||
|
}
|
||
|
|
||
|
pub struct IncPerRound<const S: u32, const I: u32>;
|
||
|
impl<const S: u32, const I: u32> frame_support::traits::Get<u32> for IncPerRound<S, I> {
|
||
|
fn get() -> u32 {
|
||
|
S + (ELectionProviderMultiPhase::round() * I)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl pallet_election_provider_multi_phase::Config for Runtime {
|
||
|
type RuntimeEvent = RuntimeEvent;
|
||
|
type Currency = Balances;
|
||
|
type EstimateCallFee = TransactionPayment;
|
||
|
type MinerConfig = Self;
|
||
|
type SignedMaxRefunds = ();
|
||
|
type UnsignedPhase = ConstU32<{ SESSION / 2 }>;
|
||
|
type SignedPhase = ConstU32<{ SESSION / 2 }>;
|
||
|
type BetterSignedThreshold = ();
|
||
|
type BetterUnsignedThreshold = ();
|
||
|
type OffchainRepeat = ();
|
||
|
type MinerTxPriority = ElectionUnsignedPrioirty;
|
||
|
type SignedMaxSubmissions = ConstU32<10>;
|
||
|
type SignedRewardBase = SignedRewardBase;
|
||
|
type SignedDepositBase =
|
||
|
GeometricDepositBase<Balance, SignedFixedDeposit, SignedDepositIncreaseFactor>;
|
||
|
type SignedDepositByte = SignedDepositByte;
|
||
|
type SignedDepositWeight = ();
|
||
|
type SignedMaxWeight = MinerMaxWeight;
|
||
|
type SlashHandler = ();
|
||
|
type RewardHandler = ();
|
||
|
type DataProvider = Staking;
|
||
|
type Fallback = onchain::OnChainExecution<OnChainSeqPhragmen>;
|
||
|
type GovernanceFallback = onchain::OnChainExecution<OnChainSeqPhragmen>;
|
||
|
type Solver = SequentialPhragmen<AccountId, SolutionAccuracyOf<Self>, ()>;
|
||
|
type WeightInfo pallet_election_provider_multi_phase::weights::SubstrateWeight<Self>;
|
||
|
type ForceOrigin = EnsureRoot<Self::AccountId>;
|
||
|
type MaxWinners = MaxActiveValidators;
|
||
|
type BenchmarkingConfig = ElectionProviderBenchmarkConfig;
|
||
|
type ElectionBounds = ElectionBounds;
|
||
|
}
|
||
|
|
||
|
pub mod voter_bags;
|
||
|
parameter_types! {
|
||
|
pub const BagThreshold: &'static [u64] = &voter_bags::THRESHOLDS;
|
||
|
}
|
||
|
|
||
|
impl pallet_bags_list::Config for Runtime {
|
||
|
type RuntimeEvent = RuntimeEvent;
|
||
|
type ScoreProvider = Staking;
|
||
|
type Score = u64;
|
||
|
type WeightInfo = pallet_bags_list::weights::SubstrateWeight<Runtime>;
|
||
|
type BagThreshold = BagThreshold;
|
||
|
}
|
||
|
|
||
|
construct_runtime!(
|
||
|
pub enum Runtime
|
||
|
{
|
||
|
System: frame_system,
|
||
|
RandomnessCollectiveFlip: pallet_insecure_randomness_collective_flip,
|
||
|
Timestamp: pallet_timestamp,
|
||
|
Sudo: pallet_sudo,
|
||
|
Aura: pallet_aura,
|
||
|
Grandpa: pallet_grandpa,
|
||
|
Balances: pallet_balances,
|
||
|
Staking: pallet_staking,
|
||
|
BagsList: pallet_bags_list,
|
||
|
TransactionPayment: pallet_transaction_payment,
|
||
|
ELectionProviderMultiPhase: election_multi_phase,
|
||
|
}
|
||
|
);
|
||
|
|
||
|
pub type Address = sp_runtime::MultiAddress<AccountId, ()>;
|
||
|
pub type Header = generic::Header<BlockNumber, BlakeTwo256>;
|
||
|
pub type Block = generic::Block<Header, UncheckedExtrinsic>;
|
||
|
pub type SignedExtra = (
|
||
|
frame_system::CheckSpecVersion<Runtime>,
|
||
|
frame_system::CheckTxVersion<Runtime>,
|
||
|
frame_system::CheckGenesis<Runtime>,
|
||
|
frame_system::CheckEra<Runtime>,
|
||
|
frame_system::CheckNonce<Runtime>,
|
||
|
frame_system::CheckWeight<Runtime>,
|
||
|
pallet_transaction_payment::ChargeTransactionPayment<Runtime>
|
||
|
);
|
||
|
|
||
|
pub type UncheckedExtrinsic =
|
||
|
generic::UncheckedExtrinsic<Address, RuntimeCall, Signature, SignedExtra>;
|
||
|
pub type Executive = frame_executive::Executive<
|
||
|
Runtime,
|
||
|
Block,
|
||
|
frame_system::ChainContext<Runtime>,
|
||
|
Runtime,
|
||
|
AllPalletsWithSystem
|
||
|
>;
|
||
|
|
||
|
impl_runtime_apis! {
|
||
|
impl sp_api::Core<Block> for Runtime {
|
||
|
fn version() -> RuntimeVersion {
|
||
|
VERSION
|
||
|
}
|
||
|
|
||
|
fn execute_block(block: Block) {
|
||
|
Executive::execute_block(block);
|
||
|
}
|
||
|
|
||
|
fn initialize_block(header: &<Block as BlockT>::Header) {
|
||
|
Executive::initialize_block(header)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl sp_api::Metadata<Block> for Runtime {
|
||
|
fn metadata() -> OpaqueMetadata {
|
||
|
OpaqueMetadata::new(Runtime::metadata().into())
|
||
|
}
|
||
|
|
||
|
fn metadata_at_version(version: u32) -> Option<OpaqueMetadata> {
|
||
|
Runtime::metadata_at_version(version)
|
||
|
}
|
||
|
|
||
|
fn metadata_versions() -> sp_std::vec::Vec<u32> {
|
||
|
Runtime::metadata_versions()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl sp_block_builder::BlockBuilder<Block> for Runtime {
|
||
|
fn apply_extrinsic(extrinsic: <Block as BlockT>::Extrinsic) -> ApplyExtrinsicResult {
|
||
|
Executive::apply_extrinsic(extrinsic)
|
||
|
}
|
||
|
|
||
|
fn finalize_block() -> <Block as BlockT>::Header {
|
||
|
Executive::finalize_block()
|
||
|
}
|
||
|
|
||
|
fn inherent_extrinsics(
|
||
|
data: inherents::InherentData,
|
||
|
) -> Vec<<Block as BlockT>::Extrinsic> {
|
||
|
data.create_extrinsics()
|
||
|
}
|
||
|
|
||
|
fn check_inherents(
|
||
|
block: Block,
|
||
|
data: inherents::InherentData,
|
||
|
) -> inherents::CheckInherentsResult {
|
||
|
data.check_extrinsics(&block)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl sp_transaction_pool::runtime_api::TaggedTransactionQueue<Block> for Runtime {
|
||
|
fn validate_transaction(
|
||
|
source: TransactionSource,
|
||
|
tx: <Block as BlockT>::Extrinsic,
|
||
|
block_hash: <Block as BlockT>::Hash,
|
||
|
) -> TransactionValidity {
|
||
|
Executive::validate_transaction(source, tx, block_hash)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl sp_offchain::OffchainWorkerApi<Block> for Runtime {
|
||
|
fn offchain_worker(header: &<Block as BlockT>::Header) {
|
||
|
Executive::offchain_worker(header)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl sp_consensus_aura::AuraApi<Block, AuraId> for Runtime {
|
||
|
fn slot_duration() -> sp_consensus_aura::SlotDuration {
|
||
|
sp_consensus_aura::SlotDuration::from_millis(Aura::slot_duration())
|
||
|
}
|
||
|
|
||
|
fn authorities() -> Vec<AuraId> {
|
||
|
Aura::authorities().into_iter()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl sp_session::SessionKeys<Block> for Runtime {
|
||
|
fn generate_session_keys(seed: Option<Vec<u8>>) -> Vec<u8> {
|
||
|
opaque::SessionKeys::generate(seed)
|
||
|
}
|
||
|
|
||
|
fn decode_session_keys(
|
||
|
encoded: Vec<u8>,
|
||
|
) -> Option<Vec<(Vec<u8>, KeyTypeId)>> {
|
||
|
opaque::SessionKeys::decode_into_raw_public_keys(&encoded)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl fg_primitives::GrandpaApi<Block> for Runtime {
|
||
|
fn grandpa_authorities() -> GrandpaAuthorityList {
|
||
|
Grandpa::grandpa_authorities()
|
||
|
}
|
||
|
|
||
|
fn current_set_id() -> fg_primitives::SetId {
|
||
|
Grandpa::current_set_id()
|
||
|
}
|
||
|
|
||
|
fn submit_report_equivocation_unsigned_extrinsic(
|
||
|
_equivocation_proof: fg_primitives::EquivocationProof<
|
||
|
<Block as BlockT>::Hash,
|
||
|
NumberFor<Block>,
|
||
|
>,
|
||
|
_key_owner_proof: fg_primitives::OpaqueKeyOwnershipProof,
|
||
|
) -> Option<()> {
|
||
|
None
|
||
|
}
|
||
|
|
||
|
fn generate_key_ownership_proof(
|
||
|
_set_id: fg_primitives::SetId,
|
||
|
_authority_id: GrandpaId,
|
||
|
) -> Option<fg_primitives::OpaqueKeyOwnershipProof> {
|
||
|
None
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl frame_system_rpc_runtime_api::AccountNonceApi<Block, AccountId, Nonce> for Runtime {
|
||
|
fn account_nonce(account: AccountId) -> Nonce {
|
||
|
System::account_nonce(account)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl pallet_transaction_payment_rpc_runtime_api::TransactionPaymentApi<Block, Balance> for Runtime {
|
||
|
fn query_info(
|
||
|
uxt: <Block as BlockT>::Extrinsic,
|
||
|
len: u32,
|
||
|
) -> pallet_transaction_payment_rpc_runtime_api::RuntimeDispatchInfo<Balance> {
|
||
|
TransactionPayment::query_info(uxt, len)
|
||
|
}
|
||
|
|
||
|
fn query_fee_details(
|
||
|
uxt: <Block as BlockT>::Extrinsic,
|
||
|
len: u32,
|
||
|
) -> pallet_transaction_payment::FeeDetails<Balance> {
|
||
|
TransactionPayment::query_fee_details(uxt, len)
|
||
|
}
|
||
|
|
||
|
fn query_weight_to_fee(weight: Weight) -> Balance {
|
||
|
TransactionPayment::weight_to_fee(weight)
|
||
|
}
|
||
|
|
||
|
fn query_length_to_fee(length: u32) -> Balance {
|
||
|
TransactionPayment::length_to_fee(length)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl pallet_transaction_payment_rpc_runtime_api::TransactionPaymentCallApi<Block, Balance, RuntimeCall> for Runtime {
|
||
|
fn query_call_info(call: RuntimeCall, len: u32) -> pallet_transaction_payment_rpc_runtime_api::RuntimeDispatchInfo<Balance> {
|
||
|
TransactionPayment::query_call_info(call, len)
|
||
|
}
|
||
|
|
||
|
fn query_call_fee_details(call: RuntimeCall, len: u32) -> pallet_transaction_payment::FeeDetails<Balance> {
|
||
|
TransactionPayment::query_call_fee_details(call, len)
|
||
|
}
|
||
|
|
||
|
fn query_weight_to_fee(weight: Weight) -> Balance {
|
||
|
TransactionPayment::weight_to_fee(weight)
|
||
|
}
|
||
|
|
||
|
fn query_length_to_fee(length: u32) -> Balance {
|
||
|
TransactionPayment::length_to_fee(length)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl sp_genesis_builder::GenesisBuilder<Block> for Runtime {
|
||
|
fn create_default_config() -> Vec<u8> {
|
||
|
create_default_config::<RuntimeGenesisConfig>()
|
||
|
}
|
||
|
|
||
|
fn build_config(config: Vec<u8>) -> sp_genesis_builder::Result {
|
||
|
build_config::<RuntimeGenesisConfig>(config)
|
||
|
}
|
||
|
}
|
||
|
}
|