#![cfg_attr(not(feature = "std"), no_std)] #[frame_support::pallet] pub mod pallet { use scale::{Encode, Decode}; use scale_info::TypeInfo; use frame_system::pallet_prelude::*; use frame_support::pallet_prelude::*; use serai_primitives::*; use validator_sets_primitives::*; #[pallet::config] pub trait Config: frame_system::Config + TypeInfo { type RuntimeEvent: IsType<::RuntimeEvent> + From>; } #[pallet::genesis_config] #[derive(Clone, PartialEq, Eq, Debug, Encode, Decode)] pub struct GenesisConfig { /// Bond requirement to join the initial validator set. /// Every participant at genesis will automatically be assumed to have this much bond. /// This bond cannot be withdrawn however as there's no stake behind it. pub bond: Amount, /// Coins to spawn the network with in the initial validator set. pub coins: Vec, /// List of participants to place in the genesis set. pub participants: Vec, } #[cfg(feature = "std")] impl Default for GenesisConfig { fn default() -> Self { GenesisConfig { bond: Amount(1), coins: vec![], participants: vec![] } } } // Max of 16 coins per validator set // At launch, we'll have BTC, ETH, DAI, and XMR // In the future, these will be split into separate validator sets, so we're already not // planning expansion beyond just a few coins per validator set // The only case which really makes sense for multiple coins in a validator set is: // 1) The coins are small, easy to run, and make no sense to be in their own set // In this case, it's still hard to ask validators to run 16 different nodes // 2) The coins are all on the same network yet there's no DEX on-chain // In these cases, it'd be hard to find and justify 16 different coins from that single chain // This could probably be just 8, yet 16 is a hedge for the unforseen // If necessary, this can be increased with a fork type MaxCoinsPerSet = ConstU32<16>; // Support keys up to 96 bytes (BLS12-381 G2) const MAX_KEY_LEN: u32 = 96; type MaxKeyLen = ConstU32; #[derive(Clone, PartialEq, Eq, Debug, Encode, Decode, TypeInfo, MaxEncodedLen)] pub struct ValidatorSet { bond: Amount, coins: BoundedVec, // Participant and their amount bonded to this set // Limit each set to 100 participants for now participants: BoundedVec<(T::AccountId, Amount), ConstU32<100>>, } #[pallet::pallet] #[pallet::generate_store(pub(super) trait Store)] pub struct Pallet(PhantomData); /// The details of a validator set instance. #[pallet::storage] #[pallet::getter(fn validator_set)] pub type ValidatorSets = StorageMap<_, Twox64Concat, ValidatorSetInstance, ValidatorSet, OptionQuery>; type Key = BoundedVec; /// The key for a given validator set instance coin. #[pallet::storage] #[pallet::getter(fn key)] pub type Keys = StorageMap<_, Twox64Concat, (ValidatorSetInstance, Coin), Key, OptionQuery>; /// If an account has voted for a specific key or not. Prevents them from voting multiple times. #[pallet::storage] #[pallet::getter(fn voted)] pub type Voted = StorageMap<_, Blake2_128Concat, (T::AccountId, Key), (), OptionQuery>; /// How many times a key has been voted for. Once consensus is reached, the keys will be adopted. #[pallet::storage] #[pallet::getter(fn vote_count)] pub type VoteCount = StorageMap<_, Blake2_128Concat, (ValidatorSetInstance, Coin, Key), u16, ValueQuery>; #[pallet::genesis_build] impl GenesisBuild for GenesisConfig { fn build(&self) { let mut participants = Vec::new(); for participant in self.participants.clone() { participants.push((participant, self.bond)); } ValidatorSets::::set( ValidatorSetInstance(Session(0), ValidatorSetIndex(0)), Some(ValidatorSet { bond: self.bond, coins: BoundedVec::try_from(self.coins.clone()).unwrap(), participants: BoundedVec::try_from(participants).unwrap(), }), ); } } #[pallet::event] #[pallet::generate_deposit(pub(super) fn deposit_event)] pub enum Event { Vote { voter: T::AccountId, instance: ValidatorSetInstance, coin: Coin, key: Key, // Amount of votes the key now has votes: u16, }, KeyGen { instance: ValidatorSetInstance, coin: Coin, key: Key, }, } #[pallet::error] pub enum Error { /// Validator Set doesn't exist. NonExistentValidatorSet, /// Non-validator is voting. NotValidator, /// Validator Set already generated keys. AlreadyGeneratedKeys, /// Vvalidator has already voted for these keys. AlreadyVoted, } #[pallet::call] impl Pallet { #[pallet::call_index(0)] #[pallet::weight(0)] // TODO pub fn vote( origin: OriginFor, index: ValidatorSetIndex, coin: Coin, key: Key, ) -> DispatchResult { let signer = ensure_signed(origin)?; // TODO: Do we need to check the key is within the length bounds? // The docs suggest the BoundedVec will create/write, yet not read, which could be an issue // if it can be passed in // TODO: Get session let session: Session = Session(0); // Confirm a key hasn't been set for this set instance let instance = ValidatorSetInstance(session, index); if Keys::::get((instance, coin)).is_some() { Err(Error::::AlreadyGeneratedKeys)?; } // Confirm the signer is a validator in the set let set = ValidatorSets::::get(instance).ok_or(Error::::NonExistentValidatorSet)?; if set.participants.iter().any(|participant| participant.0 == signer) { Err(Error::::NotValidator)?; } // Confirm this signer hasn't already voted for these keys if Voted::::get((&signer, &key)).is_some() { Err(Error::::AlreadyVoted)?; } Voted::::set((&signer, &key), Some(())); // Add their vote let votes = VoteCount::::mutate((instance, coin, &key), |value| { *value += 1; *value }); Self::deposit_event(Event::Vote { voter: signer, instance, coin, key: key.clone(), votes }); // If we've reached consensus, set the key if usize::try_from(votes).unwrap() == set.participants.len() { Keys::::set((instance, coin), Some(key.clone())); Self::deposit_event(Event::KeyGen { instance, coin, key }); } Ok(()) } } // TODO: Support session rotation } pub use pallet::*;