2023-01-05 03:52:41 +00:00
|
|
|
#![cfg_attr(not(feature = "std"), no_std)]
|
|
|
|
|
2023-12-16 22:44:08 +00:00
|
|
|
use core::marker::PhantomData;
|
|
|
|
|
|
|
|
use scale::{Encode, Decode};
|
|
|
|
use scale_info::TypeInfo;
|
|
|
|
|
|
|
|
use sp_std::{vec, vec::Vec};
|
|
|
|
use sp_core::sr25519::{Public, Signature};
|
|
|
|
use sp_application_crypto::RuntimePublic;
|
|
|
|
use sp_session::{ShouldEndSession, GetSessionNumber, GetValidatorCount};
|
|
|
|
use sp_runtime::{KeyTypeId, ConsensusEngineId, traits::IsMember};
|
|
|
|
use sp_staking::offence::{ReportOffence, Offence, OffenceError};
|
|
|
|
|
|
|
|
use frame_system::{pallet_prelude::*, RawOrigin};
|
|
|
|
use frame_support::{
|
|
|
|
pallet_prelude::*,
|
|
|
|
traits::{DisabledValidators, KeyOwnerProofSystem, FindAuthor},
|
|
|
|
BoundedVec, WeakBoundedVec, StoragePrefixedMap,
|
|
|
|
};
|
|
|
|
|
|
|
|
use serai_primitives::*;
|
|
|
|
pub use validator_sets_primitives as primitives;
|
|
|
|
use primitives::*;
|
|
|
|
|
|
|
|
use coins_pallet::{Pallet as Coins, AllowMint};
|
|
|
|
use dex_pallet::Pallet as Dex;
|
|
|
|
|
|
|
|
use pallet_babe::{
|
|
|
|
Pallet as Babe, AuthorityId as BabeAuthorityId, EquivocationOffence as BabeEquivocationOffence,
|
|
|
|
};
|
|
|
|
use pallet_grandpa::{
|
|
|
|
Pallet as Grandpa, AuthorityId as GrandpaAuthorityId,
|
|
|
|
EquivocationOffence as GrandpaEquivocationOffence,
|
|
|
|
};
|
|
|
|
|
|
|
|
#[derive(Debug, Encode, Decode, TypeInfo, PartialEq, Eq, Clone)]
|
|
|
|
pub struct MembershipProof<T: pallet::Config>(pub Public, pub PhantomData<T>);
|
|
|
|
impl<T: pallet::Config> GetSessionNumber for MembershipProof<T> {
|
|
|
|
fn session(&self) -> u32 {
|
|
|
|
let current = Pallet::<T>::session(NetworkId::Serai).unwrap().0;
|
|
|
|
if Babe::<T>::is_member(&BabeAuthorityId::from(self.0)) {
|
|
|
|
current
|
|
|
|
} else {
|
|
|
|
// if it isn't in the current session, it should have been in the previous one.
|
|
|
|
current - 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl<T: pallet::Config> GetValidatorCount for MembershipProof<T> {
|
|
|
|
// We only implement and this interface to satisfy trait requirements
|
|
|
|
// Although this might return the wrong count if the offender was in the previous set, we don't
|
|
|
|
// rely on it and Substrate only relies on it to offer economic calculations we also don't rely
|
|
|
|
// on
|
|
|
|
fn validator_count(&self) -> u32 {
|
2023-12-17 01:54:24 +00:00
|
|
|
u32::try_from(Babe::<T>::authorities().len()).unwrap()
|
2023-12-16 22:44:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-17 01:54:24 +00:00
|
|
|
#[allow(
|
|
|
|
deprecated,
|
|
|
|
clippy::let_unit_value,
|
|
|
|
clippy::cast_possible_truncation,
|
|
|
|
clippy::ignored_unit_patterns
|
|
|
|
)] // TODO
|
2023-01-05 03:52:41 +00:00
|
|
|
#[frame_support::pallet]
|
|
|
|
pub mod pallet {
|
2023-11-22 11:22:46 +00:00
|
|
|
use super::*;
|
|
|
|
|
2023-01-05 03:52:41 +00:00
|
|
|
#[pallet::config]
|
2023-10-10 10:53:24 +00:00
|
|
|
pub trait Config:
|
2023-10-12 04:51:18 +00:00
|
|
|
frame_system::Config<AccountId = Public>
|
2023-10-22 07:59:21 +00:00
|
|
|
+ coins_pallet::Config
|
2023-12-05 13:52:50 +00:00
|
|
|
+ dex_pallet::Config
|
2023-11-22 11:22:46 +00:00
|
|
|
+ pallet_babe::Config
|
|
|
|
+ pallet_grandpa::Config
|
2023-10-12 04:51:18 +00:00
|
|
|
+ TypeInfo
|
2023-10-10 10:53:24 +00:00
|
|
|
{
|
2023-01-05 03:52:41 +00:00
|
|
|
type RuntimeEvent: IsType<<Self as frame_system::Config>::RuntimeEvent> + From<Event<Self>>;
|
2023-11-22 11:22:46 +00:00
|
|
|
|
|
|
|
type ShouldEndSession: ShouldEndSession<BlockNumberFor<Self>>;
|
2023-01-05 03:52:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[pallet::genesis_config]
|
2023-01-20 16:00:18 +00:00
|
|
|
#[derive(Clone, PartialEq, Eq, Debug, Encode, Decode)]
|
2023-01-05 03:52:41 +00:00
|
|
|
pub struct GenesisConfig<T: Config> {
|
2023-10-13 04:04:28 +00:00
|
|
|
/// Networks to spawn Serai with, and the stake requirement per key share.
|
2023-10-10 10:53:24 +00:00
|
|
|
///
|
|
|
|
/// Every participant at genesis will automatically be assumed to have this much stake.
|
|
|
|
/// This stake cannot be withdrawn however as there's no actual stake behind it.
|
2023-10-13 04:04:28 +00:00
|
|
|
pub networks: Vec<(NetworkId, Amount)>,
|
2023-03-25 05:30:53 +00:00
|
|
|
/// List of participants to place in the initial validator sets.
|
2023-01-05 03:52:41 +00:00
|
|
|
pub participants: Vec<T::AccountId>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Config> Default for GenesisConfig<T> {
|
|
|
|
fn default() -> Self {
|
2023-10-13 04:12:10 +00:00
|
|
|
GenesisConfig { networks: Default::default(), participants: Default::default() }
|
2023-01-05 03:52:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[pallet::pallet]
|
|
|
|
pub struct Pallet<T>(PhantomData<T>);
|
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
/// The current session for a network.
|
|
|
|
// Uses Identity for the lookup to avoid a hash of a severely limited fixed key-space.
|
2023-01-05 03:52:41 +00:00
|
|
|
#[pallet::storage]
|
2023-11-22 11:22:46 +00:00
|
|
|
#[pallet::getter(fn session)]
|
2023-10-10 10:53:24 +00:00
|
|
|
pub type CurrentSession<T: Config> = StorageMap<_, Identity, NetworkId, Session, OptionQuery>;
|
|
|
|
impl<T: Config> Pallet<T> {
|
2023-11-22 11:22:46 +00:00
|
|
|
pub fn latest_decided_session(network: NetworkId) -> Option<Session> {
|
|
|
|
let session = Self::session(network);
|
|
|
|
// we already decided about the next session for serai.
|
2023-10-10 10:53:24 +00:00
|
|
|
if network == NetworkId::Serai {
|
2023-11-22 11:22:46 +00:00
|
|
|
return session.map(|s| Session(s.0 + 1));
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
2023-11-22 11:22:46 +00:00
|
|
|
session
|
2023-10-22 00:06:53 +00:00
|
|
|
}
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
|
|
|
|
2023-10-13 02:44:10 +00:00
|
|
|
/// The allocation required per key share.
|
2023-10-10 10:53:24 +00:00
|
|
|
// Uses Identity for the lookup to avoid a hash of a severely limited fixed key-space.
|
|
|
|
#[pallet::storage]
|
2023-10-13 02:44:10 +00:00
|
|
|
#[pallet::getter(fn allocation_per_key_share)]
|
|
|
|
pub type AllocationPerKeyShare<T: Config> =
|
|
|
|
StorageMap<_, Identity, NetworkId, Amount, OptionQuery>;
|
2024-06-24 11:41:25 +00:00
|
|
|
/// The validators selected to be in-set (and their key shares), regardless of if removed.
|
|
|
|
///
|
|
|
|
/// This method allows iterating over all validators and their stake.
|
2023-10-10 10:53:24 +00:00
|
|
|
#[pallet::storage]
|
2023-12-23 02:09:18 +00:00
|
|
|
#[pallet::getter(fn participants_for_latest_decided_set)]
|
2023-12-04 12:04:44 +00:00
|
|
|
pub(crate) type Participants<T: Config> = StorageMap<
|
2023-10-10 10:53:24 +00:00
|
|
|
_,
|
|
|
|
Identity,
|
|
|
|
NetworkId,
|
2023-11-22 11:22:46 +00:00
|
|
|
BoundedVec<(Public, u64), ConstU32<{ MAX_KEY_SHARES_PER_SET }>>,
|
2023-12-04 12:04:44 +00:00
|
|
|
OptionQuery,
|
2023-10-10 10:53:24 +00:00
|
|
|
>;
|
2024-06-24 11:41:25 +00:00
|
|
|
/// The validators selected to be in-set, regardless of if removed.
|
|
|
|
///
|
|
|
|
/// This method allows quickly checking for presence in-set and looking up a validator's key
|
|
|
|
/// shares.
|
2023-12-06 10:39:00 +00:00
|
|
|
// Uses Identity for NetworkId to avoid a hash of a severely limited fixed key-space.
|
2023-10-10 10:53:24 +00:00
|
|
|
#[pallet::storage]
|
2023-12-04 12:04:44 +00:00
|
|
|
pub(crate) type InSet<T: Config> =
|
2023-12-06 10:39:00 +00:00
|
|
|
StorageDoubleMap<_, Identity, NetworkId, Blake2_128Concat, Public, u64, OptionQuery>;
|
2023-12-04 12:04:44 +00:00
|
|
|
|
2023-10-22 00:06:53 +00:00
|
|
|
impl<T: Config> Pallet<T> {
|
|
|
|
// This exists as InSet, for Serai, is the validators set for the next session, *not* the
|
|
|
|
// current set's validators
|
|
|
|
#[inline]
|
|
|
|
fn in_active_serai_set(account: Public) -> bool {
|
2023-11-22 11:22:46 +00:00
|
|
|
// TODO: is_member is internally O(n). Update Babe to use an O(1) storage lookup?
|
|
|
|
Babe::<T>::is_member(&BabeAuthorityId::from(account))
|
2023-10-22 00:06:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns true if the account is included in an active set.
|
2023-12-04 12:04:44 +00:00
|
|
|
///
|
|
|
|
/// This will still include participants which were removed from the DKG.
|
2023-10-22 00:06:53 +00:00
|
|
|
pub fn in_active_set(network: NetworkId, account: Public) -> bool {
|
|
|
|
if network == NetworkId::Serai {
|
|
|
|
Self::in_active_serai_set(account)
|
|
|
|
} else {
|
2023-12-06 10:39:00 +00:00
|
|
|
InSet::<T>::contains_key(network, account)
|
2023-10-22 00:06:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns true if the account has been definitively included in an active or upcoming set.
|
2023-12-04 12:04:44 +00:00
|
|
|
///
|
|
|
|
/// This will still include participants which were removed from the DKG.
|
2023-10-22 00:06:53 +00:00
|
|
|
pub fn in_set(network: NetworkId, account: Public) -> bool {
|
2023-12-06 10:39:00 +00:00
|
|
|
if InSet::<T>::contains_key(network, account) {
|
2023-10-22 00:06:53 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if network == NetworkId::Serai {
|
|
|
|
return Self::in_active_serai_set(account);
|
|
|
|
}
|
|
|
|
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns true if the account is present in the latest decided set.
|
|
|
|
///
|
|
|
|
/// This is useful when working with `allocation` and `total_allocated_stake`, which return the
|
|
|
|
/// latest information.
|
|
|
|
pub fn in_latest_decided_set(network: NetworkId, account: Public) -> bool {
|
2023-12-06 10:39:00 +00:00
|
|
|
InSet::<T>::contains_key(network, account)
|
2023-10-22 00:06:53 +00:00
|
|
|
}
|
|
|
|
}
|
2023-10-10 10:53:24 +00:00
|
|
|
|
2023-10-20 20:58:44 +00:00
|
|
|
/// The total stake allocated to this network by the active set of validators.
|
|
|
|
#[pallet::storage]
|
2023-10-22 00:06:53 +00:00
|
|
|
#[pallet::getter(fn total_allocated_stake)]
|
2023-10-20 20:58:44 +00:00
|
|
|
pub type TotalAllocatedStake<T: Config> = StorageMap<_, Identity, NetworkId, Amount, OptionQuery>;
|
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
/// The current amount allocated to a validator set by a validator.
|
|
|
|
#[pallet::storage]
|
|
|
|
#[pallet::getter(fn allocation)]
|
|
|
|
pub type Allocations<T: Config> =
|
|
|
|
StorageMap<_, Blake2_128Concat, (NetworkId, Public), Amount, OptionQuery>;
|
|
|
|
/// A sorted view of the current allocations premised on the underlying DB itself being sorted.
|
2023-10-10 20:50:48 +00:00
|
|
|
/*
|
|
|
|
This uses Identity so we can take advantage of the DB's lexicographic ordering to iterate over
|
|
|
|
the key space from highest-to-lowest allocated.
|
|
|
|
|
|
|
|
This does remove the protection using a hash algorithm here offers against spam attacks (by
|
|
|
|
flooding the DB with layers, increasing lookup time and merkle proof sizes, not that we use
|
|
|
|
merkle proofs as Polkadot does).
|
|
|
|
|
|
|
|
Since amounts are represented with just 8 bytes, only 16 nibbles are presents. This caps the
|
|
|
|
potential depth caused by spam at 16 layers (as the underlying DB operates on nibbles).
|
|
|
|
|
|
|
|
While there is an entire 32-byte public key after this, a Blake hash of the key is inserted
|
|
|
|
after the amount to prevent the key from also being used to cause layer spam.
|
|
|
|
|
|
|
|
There's also a minimum stake requirement, which further reduces the potential for spam.
|
|
|
|
*/
|
2023-10-10 10:53:24 +00:00
|
|
|
#[pallet::storage]
|
|
|
|
type SortedAllocations<T: Config> =
|
2023-10-10 20:50:48 +00:00
|
|
|
StorageMap<_, Identity, (NetworkId, [u8; 8], [u8; 16], Public), (), OptionQuery>;
|
2023-10-10 10:53:24 +00:00
|
|
|
impl<T: Config> Pallet<T> {
|
2023-10-10 20:50:48 +00:00
|
|
|
#[inline]
|
|
|
|
fn sorted_allocation_key(
|
|
|
|
network: NetworkId,
|
|
|
|
key: Public,
|
|
|
|
amount: Amount,
|
|
|
|
) -> (NetworkId, [u8; 8], [u8; 16], Public) {
|
2024-02-20 01:50:04 +00:00
|
|
|
let amount = reverse_lexicographic_order(amount.0.to_be_bytes());
|
2023-10-10 20:50:48 +00:00
|
|
|
let hash = sp_io::hashing::blake2_128(&(network, amount, key).encode());
|
|
|
|
(network, amount, hash, key)
|
|
|
|
}
|
2023-10-13 02:44:10 +00:00
|
|
|
fn recover_amount_from_sorted_allocation_key(key: &[u8]) -> Amount {
|
|
|
|
let distance_from_end = 8 + 16 + 32;
|
|
|
|
let start_pos = key.len() - distance_from_end;
|
|
|
|
let mut raw: [u8; 8] = key[start_pos .. (start_pos + 8)].try_into().unwrap();
|
|
|
|
for byte in &mut raw {
|
|
|
|
*byte = !*byte;
|
|
|
|
}
|
|
|
|
Amount(u64::from_be_bytes(raw))
|
|
|
|
}
|
|
|
|
fn recover_key_from_sorted_allocation_key(key: &[u8]) -> Public {
|
|
|
|
Public(key[(key.len() - 32) ..].try_into().unwrap())
|
|
|
|
}
|
2023-10-16 05:47:15 +00:00
|
|
|
// Returns if this validator already had an allocation set.
|
|
|
|
fn set_allocation(network: NetworkId, key: Public, amount: Amount) -> bool {
|
2023-10-10 10:53:24 +00:00
|
|
|
let prior = Allocations::<T>::take((network, key));
|
2023-10-10 20:50:48 +00:00
|
|
|
if let Some(amount) = prior {
|
|
|
|
SortedAllocations::<T>::remove(Self::sorted_allocation_key(network, key, amount));
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
|
|
|
if amount.0 != 0 {
|
|
|
|
Allocations::<T>::set((network, key), Some(amount));
|
2023-10-10 20:50:48 +00:00
|
|
|
SortedAllocations::<T>::set(Self::sorted_allocation_key(network, key, amount), Some(()));
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
2023-10-16 05:47:15 +00:00
|
|
|
prior.is_some()
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
|
|
|
}
|
2023-01-05 03:52:41 +00:00
|
|
|
|
2023-12-06 10:39:00 +00:00
|
|
|
// Doesn't use PrefixIterator as we need to yield the keys *and* values
|
|
|
|
// PrefixIterator only yields the values
|
2023-10-13 03:05:29 +00:00
|
|
|
struct SortedAllocationsIter<T: Config> {
|
|
|
|
_t: PhantomData<T>,
|
|
|
|
prefix: Vec<u8>,
|
|
|
|
last: Vec<u8>,
|
|
|
|
}
|
|
|
|
impl<T: Config> SortedAllocationsIter<T> {
|
|
|
|
fn new(network: NetworkId) -> Self {
|
|
|
|
let mut prefix = SortedAllocations::<T>::final_prefix().to_vec();
|
|
|
|
prefix.extend(&network.encode());
|
|
|
|
Self { _t: PhantomData, prefix: prefix.clone(), last: prefix }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl<T: Config> Iterator for SortedAllocationsIter<T> {
|
|
|
|
type Item = (Public, Amount);
|
|
|
|
fn next(&mut self) -> Option<Self::Item> {
|
|
|
|
let next = sp_io::storage::next_key(&self.last)?;
|
|
|
|
if !next.starts_with(&self.prefix) {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
let key = Pallet::<T>::recover_key_from_sorted_allocation_key(&next);
|
|
|
|
let amount = Pallet::<T>::recover_amount_from_sorted_allocation_key(&next);
|
|
|
|
self.last = next;
|
|
|
|
Some((key, amount))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-12 03:42:15 +00:00
|
|
|
/// Pending deallocations, keyed by the Session they become unlocked on.
|
|
|
|
#[pallet::storage]
|
2023-12-16 22:44:08 +00:00
|
|
|
type PendingDeallocations<T: Config> = StorageDoubleMap<
|
|
|
|
_,
|
|
|
|
Blake2_128Concat,
|
|
|
|
(NetworkId, Public),
|
|
|
|
Identity,
|
|
|
|
Session,
|
|
|
|
Amount,
|
|
|
|
OptionQuery,
|
|
|
|
>;
|
2023-10-12 03:42:15 +00:00
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
/// The generated key pair for a given validator set instance.
|
2023-01-05 03:52:41 +00:00
|
|
|
#[pallet::storage]
|
2023-03-31 00:24:11 +00:00
|
|
|
#[pallet::getter(fn keys)]
|
2023-03-28 09:45:54 +00:00
|
|
|
pub type Keys<T: Config> = StorageMap<_, Twox64Concat, ValidatorSet, KeyPair, OptionQuery>;
|
2023-01-05 03:52:41 +00:00
|
|
|
|
2024-01-29 08:48:53 +00:00
|
|
|
/// The key for validator sets which can (and still need to) publish their slash reports.
|
|
|
|
#[pallet::storage]
|
|
|
|
pub type PendingSlashReport<T: Config> = StorageMap<_, Identity, NetworkId, Public, OptionQuery>;
|
|
|
|
|
2023-12-16 22:44:08 +00:00
|
|
|
/// Disabled validators.
|
|
|
|
#[pallet::storage]
|
|
|
|
pub type SeraiDisabledIndices<T: Config> = StorageMap<_, Identity, u32, Public, OptionQuery>;
|
|
|
|
|
2023-04-15 04:40:33 +00:00
|
|
|
#[pallet::event]
|
|
|
|
#[pallet::generate_deposit(pub(super) fn deposit_event)]
|
|
|
|
pub enum Event<T: Config> {
|
2023-10-22 07:28:42 +00:00
|
|
|
NewSet {
|
|
|
|
set: ValidatorSet,
|
|
|
|
},
|
2023-12-04 12:04:44 +00:00
|
|
|
ParticipantRemoved {
|
|
|
|
set: ValidatorSet,
|
|
|
|
removed: T::AccountId,
|
|
|
|
},
|
2023-10-22 07:28:42 +00:00
|
|
|
KeyGen {
|
|
|
|
set: ValidatorSet,
|
|
|
|
key_pair: KeyPair,
|
|
|
|
},
|
2024-01-29 08:48:53 +00:00
|
|
|
AcceptedHandover {
|
|
|
|
set: ValidatorSet,
|
|
|
|
},
|
|
|
|
SetRetired {
|
|
|
|
set: ValidatorSet,
|
|
|
|
},
|
2023-10-22 07:28:42 +00:00
|
|
|
AllocationIncreased {
|
|
|
|
validator: T::AccountId,
|
|
|
|
network: NetworkId,
|
|
|
|
amount: Amount,
|
|
|
|
},
|
|
|
|
AllocationDecreased {
|
|
|
|
validator: T::AccountId,
|
|
|
|
network: NetworkId,
|
|
|
|
amount: Amount,
|
|
|
|
delayed_until: Option<Session>,
|
|
|
|
},
|
2023-10-22 07:59:21 +00:00
|
|
|
DeallocationClaimed {
|
|
|
|
validator: T::AccountId,
|
|
|
|
network: NetworkId,
|
|
|
|
session: Session,
|
|
|
|
},
|
2023-04-15 04:40:33 +00:00
|
|
|
}
|
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
impl<T: Config> Pallet<T> {
|
|
|
|
fn new_set(network: NetworkId) {
|
2023-12-05 13:52:50 +00:00
|
|
|
// TODO: prevent new set if it doesn't have enough stake for economic security.
|
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
// Update CurrentSession
|
2023-11-22 11:22:46 +00:00
|
|
|
let session = {
|
2023-12-17 05:01:41 +00:00
|
|
|
let new_session =
|
|
|
|
CurrentSession::<T>::get(network).map_or(Session(0), |session| Session(session.0 + 1));
|
2023-10-11 06:10:35 +00:00
|
|
|
CurrentSession::<T>::set(network, Some(new_session));
|
|
|
|
new_session
|
2023-10-10 10:53:24 +00:00
|
|
|
};
|
2023-05-13 06:02:47 +00:00
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
// Clear the current InSet
|
2023-12-06 10:39:00 +00:00
|
|
|
assert_eq!(
|
|
|
|
InSet::<T>::clear_prefix(network, MAX_KEY_SHARES_PER_SET, None).maybe_cursor,
|
|
|
|
None
|
|
|
|
);
|
2023-01-05 03:52:41 +00:00
|
|
|
|
2023-10-13 02:44:10 +00:00
|
|
|
let allocation_per_key_share = Self::allocation_per_key_share(network).unwrap().0;
|
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
let mut participants = vec![];
|
2024-03-24 03:30:51 +00:00
|
|
|
{
|
|
|
|
let mut iter = SortedAllocationsIter::<T>::new(network);
|
|
|
|
let mut key_shares = 0;
|
|
|
|
while key_shares < u64::from(MAX_KEY_SHARES_PER_SET) {
|
|
|
|
let Some((key, amount)) = iter.next() else { break };
|
|
|
|
|
|
|
|
let these_key_shares =
|
|
|
|
(amount.0 / allocation_per_key_share).min(u64::from(MAX_KEY_SHARES_PER_SET));
|
|
|
|
participants.push((key, these_key_shares));
|
|
|
|
|
|
|
|
key_shares += these_key_shares;
|
|
|
|
}
|
|
|
|
amortize_excess_key_shares(&mut participants);
|
|
|
|
}
|
2023-10-10 10:53:24 +00:00
|
|
|
|
2024-03-24 03:30:51 +00:00
|
|
|
for (key, shares) in &participants {
|
|
|
|
InSet::<T>::set(network, key, Some(*shares));
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let set = ValidatorSet { network, session };
|
|
|
|
Pallet::<T>::deposit_event(Event::NewSet { set });
|
2023-11-22 11:22:46 +00:00
|
|
|
|
2023-12-04 12:04:44 +00:00
|
|
|
Participants::<T>::set(network, Some(participants.try_into().unwrap()));
|
2023-01-05 03:52:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[pallet::error]
|
|
|
|
pub enum Error<T> {
|
|
|
|
/// Validator Set doesn't exist.
|
|
|
|
NonExistentValidatorSet,
|
2023-10-13 02:44:10 +00:00
|
|
|
/// Not enough allocation to obtain a key share in the set.
|
2023-10-10 10:53:24 +00:00
|
|
|
InsufficientAllocation,
|
2023-10-13 02:44:10 +00:00
|
|
|
/// Trying to deallocate more than allocated.
|
|
|
|
NotEnoughAllocated,
|
2023-10-13 03:47:00 +00:00
|
|
|
/// Allocation would cause the validator set to no longer achieve fault tolerance.
|
|
|
|
AllocationWouldRemoveFaultTolerance,
|
Support multiple key shares per validator (#416)
* Update the coordinator to give key shares based on weight, not based on existence
Participants are now identified by their starting index. While this compiles,
the following is unimplemented:
1) A conversion for DKG `i` values. It assumes the threshold `i` values used
will be identical for the MuSig signature used to confirm the DKG.
2) Expansion from compressed values to full values before forwarding to the
processor.
* Add a fn to the DkgConfirmer to convert `i` values as needed
Also removes TODOs regarding Serai ensuring validator key uniqueness +
validity. The current infra achieves both.
* Have the Tributary DB track participation by shares, not by count
* Prevent a node from obtaining 34% of the maximum amount of key shares
This is actually mainly intended to set a bound on message sizes in the
coordinator. Message sizes are amplified by the amount of key shares held, so
setting an upper bound on said amount lets it determine constants. While that
upper bound could be 150, that'd be unreasonable and increase the potential for
DoS attacks.
* Correct the mechanism to detect if sufficient accumulation has occured
It used to check if the latest accumulation hit the required threshold. Now,
accumulations may jump past the required threshold. The required mechanism is
to check the threshold wasn't prior met and is now met.
* Finish updating the coordinator to handle a multiple key share per validator environment
* Adjust stategy re: preventing noce reuse in DKG Confirmer
* Add TODOs regarding dropped transactions, add possible TODO fix
* Update tests/coordinator
This doesn't add new multi-key-share tests, it solely updates the existing
single key-share tests to compile and run, with the necessary fixes to the
coordinator.
* Update processor key_gen to handle generating multiple key shares at once
* Update SubstrateSigner
* Update signer, clippy
* Update processor tests
* Update processor docker tests
2023-11-04 23:26:13 +00:00
|
|
|
/// Allocation would cause the validator set to never be able to achieve fault tolerance.
|
|
|
|
AllocationWouldPreventFaultTolerance,
|
2023-10-10 10:53:24 +00:00
|
|
|
/// Deallocation would remove the participant from the set, despite the validator not
|
|
|
|
/// specifying so.
|
|
|
|
DeallocationWouldRemoveParticipant,
|
2023-10-13 03:05:29 +00:00
|
|
|
/// Deallocation would cause the validator set to no longer achieve fault tolerance.
|
|
|
|
DeallocationWouldRemoveFaultTolerance,
|
2023-10-22 07:59:21 +00:00
|
|
|
/// Deallocation to be claimed doesn't exist.
|
|
|
|
NonExistentDeallocation,
|
2023-01-05 03:52:41 +00:00
|
|
|
/// Validator Set already generated keys.
|
|
|
|
AlreadyGeneratedKeys,
|
2023-05-13 06:02:47 +00:00
|
|
|
/// An invalid MuSig signature was provided.
|
|
|
|
BadSignature,
|
2023-10-10 10:53:24 +00:00
|
|
|
/// Validator wasn't registered or active.
|
|
|
|
NonExistentValidator,
|
2023-12-05 13:52:50 +00:00
|
|
|
/// Deallocation would take the stake below what is required.
|
|
|
|
DeallocationWouldRemoveEconomicSecurity,
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
|
|
|
|
2023-11-22 11:22:46 +00:00
|
|
|
#[pallet::hooks]
|
|
|
|
impl<T: Config> Hooks<BlockNumberFor<T>> for Pallet<T> {
|
|
|
|
fn on_initialize(n: BlockNumberFor<T>) -> Weight {
|
|
|
|
if T::ShouldEndSession::should_end_session(n) {
|
|
|
|
Self::rotate_session();
|
|
|
|
// TODO: set the proper weights
|
|
|
|
T::BlockWeights::get().max_block
|
|
|
|
} else {
|
|
|
|
Weight::zero()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
#[pallet::genesis_build]
|
|
|
|
impl<T: Config> BuildGenesisConfig for GenesisConfig<T> {
|
|
|
|
fn build(&self) {
|
2023-10-13 04:04:28 +00:00
|
|
|
for (id, stake) in self.networks.clone() {
|
|
|
|
AllocationPerKeyShare::<T>::set(id, Some(stake));
|
2023-10-10 10:53:24 +00:00
|
|
|
for participant in self.participants.clone() {
|
2023-10-16 05:47:15 +00:00
|
|
|
if Pallet::<T>::set_allocation(id, participant, stake) {
|
|
|
|
panic!("participants contained duplicates");
|
|
|
|
}
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
|
|
|
Pallet::<T>::new_set(id);
|
|
|
|
}
|
|
|
|
}
|
2023-05-13 06:02:47 +00:00
|
|
|
}
|
|
|
|
|
2023-01-05 03:52:41 +00:00
|
|
|
impl<T: Config> Pallet<T> {
|
2023-10-22 07:59:21 +00:00
|
|
|
fn account() -> T::AccountId {
|
2023-11-05 17:02:34 +00:00
|
|
|
system_address(b"ValidatorSets").into()
|
2023-05-13 06:02:47 +00:00
|
|
|
}
|
2023-01-05 03:52:41 +00:00
|
|
|
|
Support multiple key shares per validator (#416)
* Update the coordinator to give key shares based on weight, not based on existence
Participants are now identified by their starting index. While this compiles,
the following is unimplemented:
1) A conversion for DKG `i` values. It assumes the threshold `i` values used
will be identical for the MuSig signature used to confirm the DKG.
2) Expansion from compressed values to full values before forwarding to the
processor.
* Add a fn to the DkgConfirmer to convert `i` values as needed
Also removes TODOs regarding Serai ensuring validator key uniqueness +
validity. The current infra achieves both.
* Have the Tributary DB track participation by shares, not by count
* Prevent a node from obtaining 34% of the maximum amount of key shares
This is actually mainly intended to set a bound on message sizes in the
coordinator. Message sizes are amplified by the amount of key shares held, so
setting an upper bound on said amount lets it determine constants. While that
upper bound could be 150, that'd be unreasonable and increase the potential for
DoS attacks.
* Correct the mechanism to detect if sufficient accumulation has occured
It used to check if the latest accumulation hit the required threshold. Now,
accumulations may jump past the required threshold. The required mechanism is
to check the threshold wasn't prior met and is now met.
* Finish updating the coordinator to handle a multiple key share per validator environment
* Adjust stategy re: preventing noce reuse in DKG Confirmer
* Add TODOs regarding dropped transactions, add possible TODO fix
* Update tests/coordinator
This doesn't add new multi-key-share tests, it solely updates the existing
single key-share tests to compile and run, with the necessary fixes to the
coordinator.
* Update processor key_gen to handle generating multiple key shares at once
* Update SubstrateSigner
* Update signer, clippy
* Update processor tests
* Update processor docker tests
2023-11-04 23:26:13 +00:00
|
|
|
// is_bft returns if the network is able to survive any single node becoming byzantine.
|
2023-10-22 07:59:21 +00:00
|
|
|
fn is_bft(network: NetworkId) -> bool {
|
|
|
|
let allocation_per_key_share = AllocationPerKeyShare::<T>::get(network).unwrap().0;
|
2023-05-13 06:02:47 +00:00
|
|
|
|
2023-10-22 07:59:21 +00:00
|
|
|
let mut validators_len = 0;
|
|
|
|
let mut top = None;
|
|
|
|
let mut key_shares = 0;
|
|
|
|
for (_, amount) in SortedAllocationsIter::<T>::new(network) {
|
|
|
|
validators_len += 1;
|
2023-05-13 06:02:47 +00:00
|
|
|
|
2023-10-22 07:59:21 +00:00
|
|
|
key_shares += amount.0 / allocation_per_key_share;
|
|
|
|
if top.is_none() {
|
|
|
|
top = Some(key_shares);
|
|
|
|
}
|
2023-05-13 06:02:47 +00:00
|
|
|
|
2023-10-22 07:59:21 +00:00
|
|
|
if key_shares > u64::from(MAX_KEY_SHARES_PER_SET) {
|
|
|
|
break;
|
|
|
|
}
|
2023-01-05 03:52:41 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 07:59:21 +00:00
|
|
|
let Some(top) = top else { return false };
|
2023-10-13 04:31:23 +00:00
|
|
|
|
2024-04-13 00:38:27 +00:00
|
|
|
// key_shares may be over MAX_KEY_SHARES_PER_SET, which will cause a round robin reduction of
|
2023-10-22 07:59:21 +00:00
|
|
|
// each validator's key shares until their sum is MAX_KEY_SHARES_PER_SET
|
|
|
|
// post_amortization_key_shares_for_top_validator yields what the top validator's key shares
|
|
|
|
// would be after such a reduction, letting us evaluate this correctly
|
|
|
|
let top = post_amortization_key_shares_for_top_validator(validators_len, top, key_shares);
|
|
|
|
(top * 3) < key_shares.min(MAX_KEY_SHARES_PER_SET.into())
|
2023-10-13 04:31:23 +00:00
|
|
|
}
|
2023-01-05 03:52:41 +00:00
|
|
|
|
2023-10-22 07:59:21 +00:00
|
|
|
fn increase_allocation(
|
2023-10-10 10:53:24 +00:00
|
|
|
network: NetworkId,
|
|
|
|
account: T::AccountId,
|
|
|
|
amount: Amount,
|
2023-10-13 03:47:00 +00:00
|
|
|
) -> DispatchResult {
|
|
|
|
let old_allocation = Self::allocation((network, account)).unwrap_or(Amount(0)).0;
|
|
|
|
let new_allocation = old_allocation + amount.0;
|
|
|
|
let allocation_per_key_share = Self::allocation_per_key_share(network).unwrap().0;
|
|
|
|
if new_allocation < allocation_per_key_share {
|
2023-10-13 02:44:10 +00:00
|
|
|
Err(Error::<T>::InsufficientAllocation)?;
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
2023-10-13 03:47:00 +00:00
|
|
|
|
|
|
|
let increased_key_shares =
|
|
|
|
(old_allocation / allocation_per_key_share) < (new_allocation / allocation_per_key_share);
|
|
|
|
|
Support multiple key shares per validator (#416)
* Update the coordinator to give key shares based on weight, not based on existence
Participants are now identified by their starting index. While this compiles,
the following is unimplemented:
1) A conversion for DKG `i` values. It assumes the threshold `i` values used
will be identical for the MuSig signature used to confirm the DKG.
2) Expansion from compressed values to full values before forwarding to the
processor.
* Add a fn to the DkgConfirmer to convert `i` values as needed
Also removes TODOs regarding Serai ensuring validator key uniqueness +
validity. The current infra achieves both.
* Have the Tributary DB track participation by shares, not by count
* Prevent a node from obtaining 34% of the maximum amount of key shares
This is actually mainly intended to set a bound on message sizes in the
coordinator. Message sizes are amplified by the amount of key shares held, so
setting an upper bound on said amount lets it determine constants. While that
upper bound could be 150, that'd be unreasonable and increase the potential for
DoS attacks.
* Correct the mechanism to detect if sufficient accumulation has occured
It used to check if the latest accumulation hit the required threshold. Now,
accumulations may jump past the required threshold. The required mechanism is
to check the threshold wasn't prior met and is now met.
* Finish updating the coordinator to handle a multiple key share per validator environment
* Adjust stategy re: preventing noce reuse in DKG Confirmer
* Add TODOs regarding dropped transactions, add possible TODO fix
* Update tests/coordinator
This doesn't add new multi-key-share tests, it solely updates the existing
single key-share tests to compile and run, with the necessary fixes to the
coordinator.
* Update processor key_gen to handle generating multiple key shares at once
* Update SubstrateSigner
* Update signer, clippy
* Update processor tests
* Update processor docker tests
2023-11-04 23:26:13 +00:00
|
|
|
// Check if the net exhibited the ability to handle any single node becoming byzantine
|
2023-10-13 03:47:00 +00:00
|
|
|
let mut was_bft = None;
|
|
|
|
if increased_key_shares {
|
|
|
|
was_bft = Some(Self::is_bft(network));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Increase the allocation now
|
2023-10-10 10:53:24 +00:00
|
|
|
Self::set_allocation(network, account, Amount(new_allocation));
|
2023-10-22 07:28:42 +00:00
|
|
|
Self::deposit_event(Event::AllocationIncreased { validator: account, network, amount });
|
2023-10-13 03:47:00 +00:00
|
|
|
|
Support multiple key shares per validator (#416)
* Update the coordinator to give key shares based on weight, not based on existence
Participants are now identified by their starting index. While this compiles,
the following is unimplemented:
1) A conversion for DKG `i` values. It assumes the threshold `i` values used
will be identical for the MuSig signature used to confirm the DKG.
2) Expansion from compressed values to full values before forwarding to the
processor.
* Add a fn to the DkgConfirmer to convert `i` values as needed
Also removes TODOs regarding Serai ensuring validator key uniqueness +
validity. The current infra achieves both.
* Have the Tributary DB track participation by shares, not by count
* Prevent a node from obtaining 34% of the maximum amount of key shares
This is actually mainly intended to set a bound on message sizes in the
coordinator. Message sizes are amplified by the amount of key shares held, so
setting an upper bound on said amount lets it determine constants. While that
upper bound could be 150, that'd be unreasonable and increase the potential for
DoS attacks.
* Correct the mechanism to detect if sufficient accumulation has occured
It used to check if the latest accumulation hit the required threshold. Now,
accumulations may jump past the required threshold. The required mechanism is
to check the threshold wasn't prior met and is now met.
* Finish updating the coordinator to handle a multiple key share per validator environment
* Adjust stategy re: preventing noce reuse in DKG Confirmer
* Add TODOs regarding dropped transactions, add possible TODO fix
* Update tests/coordinator
This doesn't add new multi-key-share tests, it solely updates the existing
single key-share tests to compile and run, with the necessary fixes to the
coordinator.
* Update processor key_gen to handle generating multiple key shares at once
* Update SubstrateSigner
* Update signer, clippy
* Update processor tests
* Update processor docker tests
2023-11-04 23:26:13 +00:00
|
|
|
// Error if the net no longer can handle any single node becoming byzantine
|
2023-10-13 03:47:00 +00:00
|
|
|
if let Some(was_bft) = was_bft {
|
|
|
|
if was_bft && (!Self::is_bft(network)) {
|
|
|
|
Err(Error::<T>::AllocationWouldRemoveFaultTolerance)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Support multiple key shares per validator (#416)
* Update the coordinator to give key shares based on weight, not based on existence
Participants are now identified by their starting index. While this compiles,
the following is unimplemented:
1) A conversion for DKG `i` values. It assumes the threshold `i` values used
will be identical for the MuSig signature used to confirm the DKG.
2) Expansion from compressed values to full values before forwarding to the
processor.
* Add a fn to the DkgConfirmer to convert `i` values as needed
Also removes TODOs regarding Serai ensuring validator key uniqueness +
validity. The current infra achieves both.
* Have the Tributary DB track participation by shares, not by count
* Prevent a node from obtaining 34% of the maximum amount of key shares
This is actually mainly intended to set a bound on message sizes in the
coordinator. Message sizes are amplified by the amount of key shares held, so
setting an upper bound on said amount lets it determine constants. While that
upper bound could be 150, that'd be unreasonable and increase the potential for
DoS attacks.
* Correct the mechanism to detect if sufficient accumulation has occured
It used to check if the latest accumulation hit the required threshold. Now,
accumulations may jump past the required threshold. The required mechanism is
to check the threshold wasn't prior met and is now met.
* Finish updating the coordinator to handle a multiple key share per validator environment
* Adjust stategy re: preventing noce reuse in DKG Confirmer
* Add TODOs regarding dropped transactions, add possible TODO fix
* Update tests/coordinator
This doesn't add new multi-key-share tests, it solely updates the existing
single key-share tests to compile and run, with the necessary fixes to the
coordinator.
* Update processor key_gen to handle generating multiple key shares at once
* Update SubstrateSigner
* Update signer, clippy
* Update processor tests
* Update processor docker tests
2023-11-04 23:26:13 +00:00
|
|
|
// The above is_bft calls are only used to check a BFT net doesn't become non-BFT
|
|
|
|
// Check here if this call would prevent a non-BFT net from *ever* becoming BFT
|
|
|
|
if (new_allocation / allocation_per_key_share) >= (MAX_KEY_SHARES_PER_SET / 3).into() {
|
|
|
|
Err(Error::<T>::AllocationWouldPreventFaultTolerance)?;
|
|
|
|
}
|
|
|
|
|
2024-06-24 11:41:25 +00:00
|
|
|
// If they're in the current set, and the current set has completed its handover (so its
|
|
|
|
// currently being tracked by TotalAllocatedStake), update the TotalAllocatedStake
|
|
|
|
if let Some(session) = Self::session(network) {
|
|
|
|
if InSet::<T>::contains_key(network, account) && Self::handover_completed(network, session)
|
|
|
|
{
|
|
|
|
TotalAllocatedStake::<T>::set(
|
|
|
|
network,
|
|
|
|
Some(Amount(TotalAllocatedStake::<T>::get(network).unwrap_or(Amount(0)).0 + amount.0)),
|
|
|
|
);
|
|
|
|
}
|
2023-10-20 20:58:44 +00:00
|
|
|
}
|
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-12-16 22:44:08 +00:00
|
|
|
fn session_to_unlock_on_for_current_set(network: NetworkId) -> Option<Session> {
|
|
|
|
let mut to_unlock_on = Self::session(network)?;
|
|
|
|
// Move to the next session, as deallocating currently in-use stake is obviously invalid
|
|
|
|
to_unlock_on.0 += 1;
|
|
|
|
if network == NetworkId::Serai {
|
|
|
|
// Since the next Serai set will already have been decided, we can only deallocate one
|
|
|
|
// session later
|
|
|
|
to_unlock_on.0 += 1;
|
|
|
|
}
|
|
|
|
// Increase the session by one, creating a cooldown period
|
|
|
|
to_unlock_on.0 += 1;
|
|
|
|
Some(to_unlock_on)
|
|
|
|
}
|
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
/// Decreases a validator's allocation to a set.
|
|
|
|
///
|
|
|
|
/// Errors if the capacity provided by this allocation is in use.
|
|
|
|
///
|
2023-10-12 04:51:18 +00:00
|
|
|
/// Errors if a partial decrease of allocation which puts the remaining allocation below the
|
|
|
|
/// minimum requirement.
|
2023-10-10 10:53:24 +00:00
|
|
|
///
|
|
|
|
/// The capacity prior provided by the allocation is immediately removed, in order to ensure it
|
|
|
|
/// doesn't become used (preventing deallocation).
|
2023-10-12 04:51:18 +00:00
|
|
|
///
|
|
|
|
/// Returns if the amount is immediately eligible for deallocation.
|
2023-10-22 07:59:21 +00:00
|
|
|
fn decrease_allocation(
|
2023-10-10 10:53:24 +00:00
|
|
|
network: NetworkId,
|
|
|
|
account: T::AccountId,
|
|
|
|
amount: Amount,
|
2023-10-13 03:47:00 +00:00
|
|
|
) -> Result<bool, DispatchError> {
|
2023-12-05 13:52:50 +00:00
|
|
|
// Check it's safe to decrease this set's stake by this amount
|
|
|
|
let new_total_staked = Self::total_allocated_stake(network)
|
|
|
|
.unwrap()
|
|
|
|
.0
|
|
|
|
.checked_sub(amount.0)
|
|
|
|
.ok_or(Error::<T>::NotEnoughAllocated)?;
|
|
|
|
let required_stake = Self::required_stake_for_network(network);
|
|
|
|
if new_total_staked < required_stake {
|
|
|
|
Err(Error::<T>::DeallocationWouldRemoveEconomicSecurity)?;
|
|
|
|
}
|
2023-10-10 10:53:24 +00:00
|
|
|
|
2023-10-13 03:05:29 +00:00
|
|
|
let old_allocation =
|
|
|
|
Self::allocation((network, account)).ok_or(Error::<T>::NonExistentValidator)?.0;
|
|
|
|
let new_allocation =
|
|
|
|
old_allocation.checked_sub(amount.0).ok_or(Error::<T>::NotEnoughAllocated)?;
|
|
|
|
|
2023-10-10 10:53:24 +00:00
|
|
|
// If we're not removing the entire allocation, yet the allocation is no longer at or above
|
2023-10-13 02:44:10 +00:00
|
|
|
// the threshold for a key share, error
|
2023-10-13 03:05:29 +00:00
|
|
|
let allocation_per_key_share = Self::allocation_per_key_share(network).unwrap().0;
|
|
|
|
if (new_allocation != 0) && (new_allocation < allocation_per_key_share) {
|
2023-10-10 10:53:24 +00:00
|
|
|
Err(Error::<T>::DeallocationWouldRemoveParticipant)?;
|
|
|
|
}
|
2023-10-13 03:05:29 +00:00
|
|
|
|
2023-10-13 03:47:00 +00:00
|
|
|
let decreased_key_shares =
|
|
|
|
(old_allocation / allocation_per_key_share) > (new_allocation / allocation_per_key_share);
|
2023-10-13 03:05:29 +00:00
|
|
|
|
|
|
|
// If this decreases the validator's key shares, error if the new set is unable to handle
|
|
|
|
// byzantine faults
|
2023-10-13 03:47:00 +00:00
|
|
|
let mut was_bft = None;
|
|
|
|
if decreased_key_shares {
|
|
|
|
was_bft = Some(Self::is_bft(network));
|
2023-10-13 03:05:29 +00:00
|
|
|
}
|
2023-10-10 10:53:24 +00:00
|
|
|
|
|
|
|
// Decrease the allocation now
|
2023-10-22 00:06:53 +00:00
|
|
|
// Since we don't also update TotalAllocatedStake here, TotalAllocatedStake may be greater
|
|
|
|
// than the sum of all allocations, according to the Allocations StorageMap
|
|
|
|
// This is intentional as this allocation has only been queued for deallocation at this time
|
2023-10-10 10:53:24 +00:00
|
|
|
Self::set_allocation(network, account, Amount(new_allocation));
|
|
|
|
|
2023-10-13 03:47:00 +00:00
|
|
|
if let Some(was_bft) = was_bft {
|
|
|
|
if was_bft && (!Self::is_bft(network)) {
|
|
|
|
Err(Error::<T>::DeallocationWouldRemoveFaultTolerance)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-05 14:36:41 +00:00
|
|
|
// If we're not in-set, allow immediate deallocation
|
|
|
|
if !Self::in_set(network, account) {
|
2023-10-22 07:28:42 +00:00
|
|
|
Self::deposit_event(Event::AllocationDecreased {
|
|
|
|
validator: account,
|
|
|
|
network,
|
|
|
|
amount,
|
|
|
|
delayed_until: None,
|
|
|
|
});
|
2023-10-12 04:51:18 +00:00
|
|
|
return Ok(true);
|
|
|
|
}
|
|
|
|
|
2023-10-22 07:59:21 +00:00
|
|
|
// Set it to PendingDeallocations, letting it be released upon a future session
|
2023-10-22 00:06:53 +00:00
|
|
|
// This unwrap should be fine as this account is active, meaning a session has occurred
|
2023-12-16 22:44:08 +00:00
|
|
|
let to_unlock_on = Self::session_to_unlock_on_for_current_set(network).unwrap();
|
2023-10-12 03:42:15 +00:00
|
|
|
let existing =
|
2023-12-16 22:44:08 +00:00
|
|
|
PendingDeallocations::<T>::get((network, account), to_unlock_on).unwrap_or(Amount(0));
|
2023-10-12 03:42:15 +00:00
|
|
|
PendingDeallocations::<T>::set(
|
2023-12-16 22:44:08 +00:00
|
|
|
(network, account),
|
|
|
|
to_unlock_on,
|
2023-10-12 03:42:15 +00:00
|
|
|
Some(Amount(existing.0 + amount.0)),
|
|
|
|
);
|
2023-10-10 10:53:24 +00:00
|
|
|
|
2023-10-22 07:28:42 +00:00
|
|
|
Self::deposit_event(Event::AllocationDecreased {
|
|
|
|
validator: account,
|
|
|
|
network,
|
|
|
|
amount,
|
|
|
|
delayed_until: Some(to_unlock_on),
|
|
|
|
});
|
|
|
|
|
2023-10-12 04:51:18 +00:00
|
|
|
Ok(false)
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
|
|
|
|
2023-10-12 03:42:15 +00:00
|
|
|
// Checks if this session has completed the handover from the prior session.
|
|
|
|
fn handover_completed(network: NetworkId, session: Session) -> bool {
|
2023-10-22 00:06:53 +00:00
|
|
|
let Some(current_session) = Self::session(network) else { return false };
|
2024-06-21 12:39:17 +00:00
|
|
|
|
|
|
|
// If the session we've been queried about is old, it must have completed its handover
|
|
|
|
if current_session.0 > session.0 {
|
2023-10-12 03:42:15 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// If the session we've been queried about has yet to start, it can't have completed its
|
|
|
|
// handover
|
|
|
|
if current_session.0 < session.0 {
|
|
|
|
return false;
|
|
|
|
}
|
2024-06-21 12:39:17 +00:00
|
|
|
|
|
|
|
// Handover is automatically complete for Serai as it doesn't have a handover protocol
|
|
|
|
if network == NetworkId::Serai {
|
|
|
|
return true;
|
2023-10-12 03:42:15 +00:00
|
|
|
}
|
2024-06-21 12:39:17 +00:00
|
|
|
|
|
|
|
// The current session must have set keys for its handover to be completed
|
|
|
|
if !Keys::<T>::contains_key(ValidatorSet { network, session }) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This must be the first session (which has set keys) OR the prior session must have been
|
|
|
|
// retired (signified by its keys no longer being present)
|
|
|
|
(session.0 == 0) ||
|
|
|
|
(!Keys::<T>::contains_key(ValidatorSet { network, session: Session(session.0 - 1) }))
|
2023-10-12 03:42:15 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 07:59:21 +00:00
|
|
|
fn new_session() {
|
2023-10-13 03:59:21 +00:00
|
|
|
for network in serai_primitives::NETWORKS {
|
2023-10-22 00:06:53 +00:00
|
|
|
// If this network hasn't started sessions yet, don't start one now
|
|
|
|
let Some(current_session) = Self::session(network) else { continue };
|
2023-11-22 11:22:46 +00:00
|
|
|
// Only spawn a new set if:
|
|
|
|
// - This is Serai, as we need to rotate Serai upon a new session (per Babe)
|
|
|
|
// - The current set was actually established with a completed handover protocol
|
|
|
|
if (network == NetworkId::Serai) || Self::handover_completed(network, current_session) {
|
2023-10-10 10:53:24 +00:00
|
|
|
Pallet::<T>::new_set(network);
|
2023-12-05 13:52:50 +00:00
|
|
|
// let the Dex know session is rotated.
|
|
|
|
Dex::<T>::on_new_session(network);
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-21 12:39:17 +00:00
|
|
|
// TODO: This is called retire_set, yet just starts retiring the set
|
|
|
|
// Update the nomenclature within this function
|
2023-10-14 20:47:25 +00:00
|
|
|
pub fn retire_set(set: ValidatorSet) {
|
2024-01-29 08:48:53 +00:00
|
|
|
// If the prior prior set didn't report, emit they're retired now
|
|
|
|
if PendingSlashReport::<T>::get(set.network).is_some() {
|
|
|
|
Self::deposit_event(Event::SetRetired {
|
|
|
|
set: ValidatorSet { network: set.network, session: Session(set.session.0 - 1) },
|
|
|
|
});
|
|
|
|
}
|
2024-02-24 19:51:06 +00:00
|
|
|
|
2024-06-24 11:41:25 +00:00
|
|
|
// Serai doesn't set keys and network slashes are handled by BABE/GRANDPA
|
2024-02-24 19:51:06 +00:00
|
|
|
if set.network != NetworkId::Serai {
|
|
|
|
// This overwrites the prior value as the prior to-report set's stake presumably just
|
|
|
|
// unlocked, making their report unenforceable
|
|
|
|
let keys = Keys::<T>::take(set).unwrap();
|
|
|
|
PendingSlashReport::<T>::set(set.network, Some(keys.0));
|
|
|
|
}
|
2024-01-29 08:48:53 +00:00
|
|
|
|
|
|
|
// We're retiring this set because the set after it accepted the handover
|
|
|
|
Self::deposit_event(Event::AcceptedHandover {
|
|
|
|
set: ValidatorSet { network: set.network, session: Session(set.session.0 + 1) },
|
|
|
|
});
|
2024-06-24 11:41:25 +00:00
|
|
|
|
|
|
|
// Update the total allocated stake to be for the current set
|
|
|
|
let participants =
|
|
|
|
Participants::<T>::get(set.network).expect("set retired without a new set");
|
|
|
|
let total_stake = participants.iter().fold(0, |acc, (addr, _)| {
|
|
|
|
acc + Allocations::<T>::get((set.network, addr)).unwrap_or(Amount(0)).0
|
|
|
|
});
|
|
|
|
TotalAllocatedStake::<T>::set(set.network, Some(Amount(total_stake)));
|
2023-10-11 03:55:59 +00:00
|
|
|
}
|
2023-10-12 03:42:15 +00:00
|
|
|
|
|
|
|
/// Take the amount deallocatable.
|
|
|
|
///
|
|
|
|
/// `session` refers to the Session the stake becomes deallocatable on.
|
2023-10-22 07:59:21 +00:00
|
|
|
fn take_deallocatable_amount(
|
2023-10-12 03:42:15 +00:00
|
|
|
network: NetworkId,
|
|
|
|
session: Session,
|
|
|
|
key: Public,
|
|
|
|
) -> Option<Amount> {
|
|
|
|
// Check this Session has properly started, completing the handover from the prior session.
|
|
|
|
if !Self::handover_completed(network, session) {
|
|
|
|
return None;
|
|
|
|
}
|
2023-12-16 22:44:08 +00:00
|
|
|
PendingDeallocations::<T>::take((network, key), session)
|
2023-10-12 03:42:15 +00:00
|
|
|
}
|
2023-11-22 11:22:46 +00:00
|
|
|
|
|
|
|
fn rotate_session() {
|
2023-12-16 22:44:08 +00:00
|
|
|
// next serai validators that is in the queue.
|
|
|
|
let now_validators = Participants::<T>::get(NetworkId::Serai)
|
2023-12-04 12:04:44 +00:00
|
|
|
.expect("no Serai participants upon rotate_session");
|
2023-11-22 11:22:46 +00:00
|
|
|
let prior_serai_session = Self::session(NetworkId::Serai).unwrap();
|
|
|
|
|
|
|
|
// TODO: T::SessionHandler::on_before_session_ending() was here.
|
|
|
|
// end the current serai session.
|
|
|
|
Self::retire_set(ValidatorSet { network: NetworkId::Serai, session: prior_serai_session });
|
|
|
|
|
|
|
|
// make a new session and get the next validator set.
|
|
|
|
Self::new_session();
|
|
|
|
|
|
|
|
// Update Babe and Grandpa
|
|
|
|
let session = prior_serai_session.0 + 1;
|
2023-12-16 22:44:08 +00:00
|
|
|
let next_validators = Participants::<T>::get(NetworkId::Serai).unwrap();
|
2023-11-22 11:22:46 +00:00
|
|
|
Babe::<T>::enact_epoch_change(
|
|
|
|
WeakBoundedVec::force_from(
|
2023-12-16 22:44:08 +00:00
|
|
|
now_validators.iter().copied().map(|(id, w)| (BabeAuthorityId::from(id), w)).collect(),
|
2023-11-22 11:22:46 +00:00
|
|
|
None,
|
|
|
|
),
|
|
|
|
WeakBoundedVec::force_from(
|
2023-12-17 01:54:24 +00:00
|
|
|
next_validators.iter().copied().map(|(id, w)| (BabeAuthorityId::from(id), w)).collect(),
|
2023-11-22 11:22:46 +00:00
|
|
|
None,
|
|
|
|
),
|
|
|
|
Some(session),
|
|
|
|
);
|
|
|
|
Grandpa::<T>::new_session(
|
|
|
|
true,
|
|
|
|
session,
|
2024-02-24 19:51:06 +00:00
|
|
|
now_validators.into_iter().map(|(id, w)| (GrandpaAuthorityId::from(id), w)).collect(),
|
2023-11-22 11:22:46 +00:00
|
|
|
);
|
2023-12-16 22:44:08 +00:00
|
|
|
|
|
|
|
// Clear SeraiDisabledIndices, only preserving keys still present in the new session
|
|
|
|
// First drain so we don't mutate as we iterate
|
|
|
|
let mut disabled = vec![];
|
|
|
|
for (_, validator) in SeraiDisabledIndices::<T>::drain() {
|
|
|
|
disabled.push(validator);
|
|
|
|
}
|
|
|
|
for disabled in disabled {
|
|
|
|
Self::disable_serai_validator(disabled);
|
|
|
|
}
|
2023-11-22 11:22:46 +00:00
|
|
|
}
|
2023-12-05 13:52:50 +00:00
|
|
|
|
|
|
|
/// Returns the required stake in terms SRI for a given `Balance`.
|
|
|
|
pub fn required_stake(balance: &Balance) -> SubstrateAmount {
|
|
|
|
use dex_pallet::HigherPrecisionBalance;
|
|
|
|
|
|
|
|
// This is inclusive to an increase in accuracy
|
2024-02-20 01:50:04 +00:00
|
|
|
let sri_per_coin = Dex::<T>::security_oracle_value(balance.coin).unwrap_or(Amount(0));
|
2023-12-05 13:52:50 +00:00
|
|
|
|
|
|
|
// See dex-pallet for the reasoning on these
|
|
|
|
let coin_decimals = balance.coin.decimals().max(5);
|
|
|
|
let accuracy_increase = HigherPrecisionBalance::from(SubstrateAmount::pow(10, coin_decimals));
|
|
|
|
|
|
|
|
let total_coin_value = u64::try_from(
|
|
|
|
HigherPrecisionBalance::from(balance.amount.0) *
|
|
|
|
HigherPrecisionBalance::from(sri_per_coin.0) /
|
|
|
|
accuracy_increase,
|
|
|
|
)
|
|
|
|
.unwrap_or(u64::MAX);
|
|
|
|
|
|
|
|
// required stake formula (COIN_VALUE * 1.5) + margin(20%)
|
|
|
|
let required_stake = total_coin_value.saturating_mul(3).saturating_div(2);
|
|
|
|
required_stake.saturating_add(total_coin_value.saturating_div(5))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the current total required stake for a given `network`.
|
|
|
|
pub fn required_stake_for_network(network: NetworkId) -> SubstrateAmount {
|
|
|
|
let mut total_required = 0;
|
|
|
|
for coin in network.coins() {
|
|
|
|
let supply = Coins::<T>::supply(coin);
|
|
|
|
total_required += Self::required_stake(&Balance { coin: *coin, amount: Amount(supply) });
|
|
|
|
}
|
|
|
|
total_required
|
|
|
|
}
|
2023-12-16 22:44:08 +00:00
|
|
|
|
|
|
|
fn can_slash_serai_validator(validator: Public) -> bool {
|
|
|
|
// Checks if they're active or actively deallocating (letting us still slash them)
|
|
|
|
// We could check if they're upcoming/still allocating, yet that'd mean the equivocation is
|
|
|
|
// invalid (as they aren't actively signing anything) or severely dated
|
|
|
|
// It's not an edge case worth being comprehensive to due to the complexity of being so
|
|
|
|
Babe::<T>::is_member(&BabeAuthorityId::from(validator)) ||
|
|
|
|
PendingDeallocations::<T>::iter_prefix((NetworkId::Serai, validator)).next().is_some()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn slash_serai_validator(validator: Public) {
|
|
|
|
let network = NetworkId::Serai;
|
|
|
|
|
|
|
|
let mut allocation = Self::allocation((network, validator)).unwrap_or(Amount(0));
|
|
|
|
// reduce the current allocation to 0.
|
|
|
|
Self::set_allocation(network, validator, Amount(0));
|
|
|
|
|
|
|
|
// Take the pending deallocation from the current session
|
|
|
|
allocation.0 += PendingDeallocations::<T>::take(
|
|
|
|
(network, validator),
|
|
|
|
Self::session_to_unlock_on_for_current_set(network).unwrap(),
|
|
|
|
)
|
|
|
|
.unwrap_or(Amount(0))
|
|
|
|
.0;
|
|
|
|
|
|
|
|
// Reduce the TotalAllocatedStake for the network, if in set
|
|
|
|
// TotalAllocatedStake is the sum of allocations and pending deallocations from the current
|
|
|
|
// session, since pending deallocations can still be slashed and therefore still contribute
|
|
|
|
// to economic security, hence the allocation calculations above being above and the ones
|
|
|
|
// below being below
|
|
|
|
if InSet::<T>::contains_key(NetworkId::Serai, validator) {
|
|
|
|
let current_staked = Self::total_allocated_stake(network).unwrap();
|
|
|
|
TotalAllocatedStake::<T>::set(network, Some(current_staked - allocation));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clear any other pending deallocations.
|
|
|
|
for (_, pending) in PendingDeallocations::<T>::drain_prefix((network, validator)) {
|
|
|
|
allocation.0 += pending.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// burn the allocation from the stake account
|
|
|
|
Coins::<T>::burn(
|
|
|
|
RawOrigin::Signed(Self::account()).into(),
|
|
|
|
Balance { coin: Coin::Serai, amount: allocation },
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Disable a Serai validator, preventing them from further authoring blocks.
|
|
|
|
///
|
|
|
|
/// Returns true if the validator-to-disable was actually a validator.
|
|
|
|
/// Returns false if they weren't.
|
|
|
|
fn disable_serai_validator(validator: Public) -> bool {
|
|
|
|
if let Some(index) =
|
|
|
|
Babe::<T>::authorities().into_iter().position(|(id, _)| id.into_inner() == validator)
|
|
|
|
{
|
|
|
|
SeraiDisabledIndices::<T>::set(u32::try_from(index).unwrap(), Some(validator));
|
|
|
|
|
|
|
|
let session = Self::session(NetworkId::Serai).unwrap();
|
|
|
|
Self::deposit_event(Event::ParticipantRemoved {
|
|
|
|
set: ValidatorSet { network: NetworkId::Serai, session },
|
|
|
|
removed: validator,
|
|
|
|
});
|
|
|
|
|
|
|
|
true
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
2023-10-10 10:53:24 +00:00
|
|
|
}
|
2023-10-22 07:59:21 +00:00
|
|
|
|
|
|
|
#[pallet::call]
|
|
|
|
impl<T: Config> Pallet<T> {
|
|
|
|
#[pallet::call_index(0)]
|
|
|
|
#[pallet::weight(0)] // TODO
|
|
|
|
pub fn set_keys(
|
|
|
|
origin: OriginFor<T>,
|
|
|
|
network: NetworkId,
|
2024-06-02 23:58:29 +00:00
|
|
|
removed_participants: BoundedVec<Public, ConstU32<{ MAX_KEY_SHARES_PER_SET / 3 }>>,
|
2023-10-22 07:59:21 +00:00
|
|
|
key_pair: KeyPair,
|
|
|
|
signature: Signature,
|
|
|
|
) -> DispatchResult {
|
|
|
|
ensure_none(origin)?;
|
|
|
|
|
|
|
|
// signature isn't checked as this is an unsigned transaction, and validate_unsigned
|
|
|
|
// (called by pre_dispatch) checks it
|
|
|
|
let _ = signature;
|
|
|
|
|
2023-12-15 04:45:15 +00:00
|
|
|
let session = Self::session(network).unwrap();
|
|
|
|
let set = ValidatorSet { network, session };
|
2023-10-22 07:59:21 +00:00
|
|
|
|
|
|
|
Keys::<T>::set(set, Some(key_pair.clone()));
|
|
|
|
|
2023-12-15 04:45:15 +00:00
|
|
|
// This does not remove from TotalAllocatedStake or InSet in order to:
|
|
|
|
// 1) Not decrease the stake present in this set. This means removed participants are
|
|
|
|
// still liable for the economic security of the external network. This prevents
|
|
|
|
// a decided set, which is economically secure, from falling below the threshold.
|
|
|
|
// 2) Not allow parties removed to immediately deallocate, per commentary on deallocation
|
|
|
|
// scheduling (https://github.com/serai-dex/serai/issues/394).
|
|
|
|
for removed in removed_participants {
|
|
|
|
Self::deposit_event(Event::ParticipantRemoved { set, removed });
|
|
|
|
}
|
|
|
|
Self::deposit_event(Event::KeyGen { set, key_pair });
|
2023-12-04 12:04:44 +00:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2024-01-29 08:48:53 +00:00
|
|
|
#[pallet::call_index(1)]
|
|
|
|
#[pallet::weight(0)] // TODO
|
|
|
|
pub fn report_slashes(
|
|
|
|
origin: OriginFor<T>,
|
|
|
|
network: NetworkId,
|
|
|
|
slashes: BoundedVec<(Public, u32), ConstU32<{ MAX_KEY_SHARES_PER_SET / 3 }>>,
|
|
|
|
signature: Signature,
|
|
|
|
) -> DispatchResult {
|
|
|
|
ensure_none(origin)?;
|
|
|
|
|
|
|
|
// signature isn't checked as this is an unsigned transaction, and validate_unsigned
|
|
|
|
// (called by pre_dispatch) checks it
|
|
|
|
let _ = signature;
|
|
|
|
|
|
|
|
// TODO: Handle slashes
|
|
|
|
let _ = slashes;
|
|
|
|
|
|
|
|
// Emit set retireed
|
|
|
|
Pallet::<T>::deposit_event(Event::SetRetired {
|
|
|
|
set: ValidatorSet { network, session: Session(Self::session(network).unwrap().0 - 1) },
|
|
|
|
});
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-12-04 12:04:44 +00:00
|
|
|
#[pallet::call_index(2)]
|
|
|
|
#[pallet::weight(0)] // TODO
|
2023-10-22 07:59:21 +00:00
|
|
|
pub fn allocate(origin: OriginFor<T>, network: NetworkId, amount: Amount) -> DispatchResult {
|
|
|
|
let validator = ensure_signed(origin)?;
|
|
|
|
Coins::<T>::transfer_internal(
|
|
|
|
validator,
|
|
|
|
Self::account(),
|
|
|
|
Balance { coin: Coin::Serai, amount },
|
|
|
|
)?;
|
|
|
|
Self::increase_allocation(network, validator, amount)
|
|
|
|
}
|
|
|
|
|
2023-12-04 12:04:44 +00:00
|
|
|
#[pallet::call_index(3)]
|
2023-10-22 07:59:21 +00:00
|
|
|
#[pallet::weight(0)] // TODO
|
|
|
|
pub fn deallocate(origin: OriginFor<T>, network: NetworkId, amount: Amount) -> DispatchResult {
|
|
|
|
let account = ensure_signed(origin)?;
|
|
|
|
|
|
|
|
let can_immediately_deallocate = Self::decrease_allocation(network, account, amount)?;
|
|
|
|
if can_immediately_deallocate {
|
|
|
|
Coins::<T>::transfer_internal(
|
|
|
|
Self::account(),
|
|
|
|
account,
|
|
|
|
Balance { coin: Coin::Serai, amount },
|
|
|
|
)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-12-04 12:04:44 +00:00
|
|
|
#[pallet::call_index(4)]
|
2023-10-22 07:59:21 +00:00
|
|
|
#[pallet::weight((0, DispatchClass::Operational))] // TODO
|
|
|
|
pub fn claim_deallocation(
|
|
|
|
origin: OriginFor<T>,
|
|
|
|
network: NetworkId,
|
|
|
|
session: Session,
|
|
|
|
) -> DispatchResult {
|
|
|
|
let account = ensure_signed(origin)?;
|
|
|
|
let Some(amount) = Self::take_deallocatable_amount(network, session, account) else {
|
|
|
|
Err(Error::<T>::NonExistentDeallocation)?
|
|
|
|
};
|
|
|
|
Coins::<T>::transfer_internal(
|
|
|
|
Self::account(),
|
|
|
|
account,
|
|
|
|
Balance { coin: Coin::Serai, amount },
|
|
|
|
)?;
|
2023-10-22 09:37:23 +00:00
|
|
|
Self::deposit_event(Event::DeallocationClaimed { validator: account, network, session });
|
2023-10-22 07:59:21 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[pallet::validate_unsigned]
|
|
|
|
impl<T: Config> ValidateUnsigned for Pallet<T> {
|
|
|
|
type Call = Call<T>;
|
|
|
|
|
|
|
|
fn validate_unsigned(_: TransactionSource, call: &Self::Call) -> TransactionValidity {
|
|
|
|
// Match to be exhaustive
|
2023-12-04 12:04:44 +00:00
|
|
|
match call {
|
2023-12-15 04:45:15 +00:00
|
|
|
Call::set_keys { network, ref removed_participants, ref key_pair, ref signature } => {
|
|
|
|
let network = *network;
|
2023-12-04 12:04:44 +00:00
|
|
|
|
2023-12-15 04:45:15 +00:00
|
|
|
// Don't allow the Serai set to set_keys, as they have no reason to do so
|
|
|
|
if network == NetworkId::Serai {
|
2023-12-04 12:04:44 +00:00
|
|
|
Err(InvalidTransaction::Custom(0))?;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Confirm this set has a session
|
2023-12-15 04:45:15 +00:00
|
|
|
let Some(current_session) = Self::session(network) else {
|
2023-12-04 12:04:44 +00:00
|
|
|
Err(InvalidTransaction::Custom(1))?
|
|
|
|
};
|
2023-12-15 04:45:15 +00:00
|
|
|
|
|
|
|
let set = ValidatorSet { network, session: current_session };
|
|
|
|
|
2023-12-04 12:04:44 +00:00
|
|
|
// Confirm it has yet to set keys
|
|
|
|
if Keys::<T>::get(set).is_some() {
|
2023-12-15 04:45:15 +00:00
|
|
|
Err(InvalidTransaction::Stale)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is a needed precondition as this uses storage variables for the latest decided
|
|
|
|
// session on this assumption
|
|
|
|
assert_eq!(Pallet::<T>::latest_decided_session(network), Some(current_session));
|
|
|
|
|
|
|
|
// This does not slash the removed participants as that'll be done at the end of the
|
|
|
|
// set's lifetime
|
|
|
|
let mut removed = hashbrown::HashSet::new();
|
|
|
|
for participant in removed_participants {
|
|
|
|
// Confirm this wasn't duplicated
|
|
|
|
if removed.contains(&participant.0) {
|
|
|
|
Err(InvalidTransaction::Custom(2))?;
|
|
|
|
}
|
|
|
|
removed.insert(participant.0);
|
2023-12-04 12:04:44 +00:00
|
|
|
}
|
|
|
|
|
2023-12-15 04:45:15 +00:00
|
|
|
let participants =
|
2023-12-04 12:04:44 +00:00
|
|
|
Participants::<T>::get(network).expect("session existed without participants");
|
|
|
|
|
2023-12-15 04:45:15 +00:00
|
|
|
let mut all_key_shares = 0;
|
|
|
|
let mut signers = vec![];
|
2023-12-04 12:04:44 +00:00
|
|
|
let mut signing_key_shares = 0;
|
2023-12-15 04:45:15 +00:00
|
|
|
for participant in participants {
|
|
|
|
let participant = participant.0;
|
|
|
|
let shares = InSet::<T>::get(network, participant)
|
|
|
|
.expect("participant from Participants wasn't InSet");
|
|
|
|
all_key_shares += shares;
|
|
|
|
|
|
|
|
if removed.contains(&participant.0) {
|
|
|
|
continue;
|
2023-12-04 12:04:44 +00:00
|
|
|
}
|
2023-12-15 04:45:15 +00:00
|
|
|
|
|
|
|
signers.push(participant);
|
2023-12-04 12:04:44 +00:00
|
|
|
signing_key_shares += shares;
|
|
|
|
}
|
|
|
|
|
2023-12-15 04:45:15 +00:00
|
|
|
{
|
|
|
|
let f = all_key_shares - signing_key_shares;
|
|
|
|
if signing_key_shares < ((2 * f) + 1) {
|
|
|
|
Err(InvalidTransaction::Custom(3))?;
|
|
|
|
}
|
2023-12-04 12:04:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Verify the signature with the MuSig key of the signers
|
2023-12-15 04:45:15 +00:00
|
|
|
// We theoretically don't need set_keys_message to bind to removed_participants, as the
|
|
|
|
// key we're signing with effectively already does so, yet there's no reason not to
|
|
|
|
if !musig_key(set, &signers)
|
|
|
|
.verify(&set_keys_message(&set, removed_participants, key_pair), signature)
|
2023-12-04 12:04:44 +00:00
|
|
|
{
|
|
|
|
Err(InvalidTransaction::BadProof)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
ValidTransaction::with_tag_prefix("ValidatorSets")
|
2024-01-29 08:48:53 +00:00
|
|
|
.and_provides((0, set))
|
2023-12-04 12:04:44 +00:00
|
|
|
.longevity(u64::MAX)
|
|
|
|
.propagate(true)
|
|
|
|
.build()
|
|
|
|
}
|
2024-01-29 08:48:53 +00:00
|
|
|
Call::report_slashes { network, ref slashes, ref signature } => {
|
|
|
|
let network = *network;
|
|
|
|
// Don't allow Serai to publish a slash report as BABE/GRANDPA handles slashes directly
|
|
|
|
if network == NetworkId::Serai {
|
|
|
|
Err(InvalidTransaction::Custom(0))?;
|
|
|
|
}
|
|
|
|
let Some(key) = PendingSlashReport::<T>::take(network) else {
|
|
|
|
// Assumed already published
|
|
|
|
Err(InvalidTransaction::Stale)?
|
|
|
|
};
|
|
|
|
// There must have been a previous session is PendingSlashReport is populated
|
|
|
|
let set =
|
|
|
|
ValidatorSet { network, session: Session(Self::session(network).unwrap().0 - 1) };
|
|
|
|
if !key.verify(&report_slashes_message(&set, slashes), signature) {
|
|
|
|
Err(InvalidTransaction::BadProof)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
ValidTransaction::with_tag_prefix("ValidatorSets")
|
|
|
|
.and_provides((1, set))
|
|
|
|
.longevity(MAX_KEY_SHARES_PER_SET.into())
|
|
|
|
.propagate(true)
|
|
|
|
.build()
|
|
|
|
}
|
2023-10-22 07:59:21 +00:00
|
|
|
Call::allocate { .. } | Call::deallocate { .. } | Call::claim_deallocation { .. } => {
|
|
|
|
Err(InvalidTransaction::Call)?
|
|
|
|
}
|
|
|
|
Call::__Ignore(_, _) => unreachable!(),
|
2023-11-19 02:27:06 +00:00
|
|
|
}
|
2023-10-22 07:59:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Explicitly provide a pre-dispatch which calls validate_unsigned
|
|
|
|
fn pre_dispatch(call: &Self::Call) -> Result<(), TransactionValidityError> {
|
|
|
|
Self::validate_unsigned(TransactionSource::InBlock, call).map(|_| ()).map_err(Into::into)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-05 13:52:50 +00:00
|
|
|
impl<T: Config> AllowMint for Pallet<T> {
|
|
|
|
fn is_allowed(balance: &Balance) -> bool {
|
|
|
|
// get the required stake
|
|
|
|
let current_required = Self::required_stake_for_network(balance.coin.network());
|
|
|
|
let new_required = current_required + Self::required_stake(balance);
|
|
|
|
|
|
|
|
// get the total stake for the network & compare.
|
|
|
|
let staked = Self::total_allocated_stake(balance.coin.network()).unwrap_or(Amount(0));
|
|
|
|
staked.0 >= new_required
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-16 22:44:08 +00:00
|
|
|
#[rustfmt::skip]
|
|
|
|
impl<T: Config, V: Into<Public> + From<Public>> KeyOwnerProofSystem<(KeyTypeId, V)> for Pallet<T> {
|
|
|
|
type Proof = MembershipProof<T>;
|
|
|
|
type IdentificationTuple = Public;
|
|
|
|
|
|
|
|
fn prove(key: (KeyTypeId, V)) -> Option<Self::Proof> {
|
|
|
|
Some(MembershipProof(key.1.into(), PhantomData))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_proof(key: (KeyTypeId, V), proof: Self::Proof) -> Option<Self::IdentificationTuple> {
|
|
|
|
let validator = key.1.into();
|
|
|
|
|
|
|
|
// check the offender and the proof offender are the same.
|
|
|
|
if validator != proof.0 {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check validator is valid
|
|
|
|
if !Self::can_slash_serai_validator(validator) {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(validator)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Config> ReportOffence<Public, Public, BabeEquivocationOffence<Public>> for Pallet<T> {
|
|
|
|
/// Report an `offence` and reward given `reporters`.
|
|
|
|
fn report_offence(
|
|
|
|
_: Vec<Public>,
|
|
|
|
offence: BabeEquivocationOffence<Public>,
|
|
|
|
) -> Result<(), OffenceError> {
|
|
|
|
// slash the offender
|
|
|
|
let offender = offence.offender;
|
|
|
|
Self::slash_serai_validator(offender);
|
|
|
|
|
|
|
|
// disable it
|
|
|
|
Self::disable_serai_validator(offender);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn is_known_offence(
|
|
|
|
offenders: &[Public],
|
|
|
|
_: &<BabeEquivocationOffence<Public> as Offence<Public>>::TimeSlot,
|
|
|
|
) -> bool {
|
|
|
|
for offender in offenders {
|
|
|
|
// It's not a known offence if we can still slash them
|
|
|
|
if Self::can_slash_serai_validator(*offender) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Config> ReportOffence<Public, Public, GrandpaEquivocationOffence<Public>> for Pallet<T> {
|
|
|
|
/// Report an `offence` and reward given `reporters`.
|
|
|
|
fn report_offence(
|
|
|
|
_: Vec<Public>,
|
|
|
|
offence: GrandpaEquivocationOffence<Public>,
|
|
|
|
) -> Result<(), OffenceError> {
|
|
|
|
// slash the offender
|
|
|
|
let offender = offence.offender;
|
|
|
|
Self::slash_serai_validator(offender);
|
|
|
|
|
|
|
|
// disable it
|
|
|
|
Self::disable_serai_validator(offender);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn is_known_offence(
|
|
|
|
offenders: &[Public],
|
|
|
|
_slot: &<GrandpaEquivocationOffence<Public> as Offence<Public>>::TimeSlot,
|
|
|
|
) -> bool {
|
|
|
|
for offender in offenders {
|
|
|
|
if Self::can_slash_serai_validator(*offender) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Config> FindAuthor<Public> for Pallet<T> {
|
|
|
|
fn find_author<'a, I>(digests: I) -> Option<Public>
|
|
|
|
where
|
|
|
|
I: 'a + IntoIterator<Item = (ConsensusEngineId, &'a [u8])>,
|
|
|
|
{
|
|
|
|
let i = Babe::<T>::find_author(digests)?;
|
|
|
|
Some(Babe::<T>::authorities()[i as usize].0.clone().into())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-22 11:22:46 +00:00
|
|
|
impl<T: Config> DisabledValidators for Pallet<T> {
|
2023-12-16 22:44:08 +00:00
|
|
|
fn is_disabled(index: u32) -> bool {
|
|
|
|
SeraiDisabledIndices::<T>::get(index).is_some()
|
2023-10-22 07:59:21 +00:00
|
|
|
}
|
|
|
|
}
|
2023-01-05 03:52:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub use pallet::*;
|