mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-01 16:39:53 +00:00
152 lines
5 KiB
Rust
152 lines
5 KiB
Rust
#![cfg_attr(not(feature = "std"), no_std)]
|
|
|
|
#[cfg(feature = "std")]
|
|
use zeroize::Zeroize;
|
|
|
|
use ciphersuite::{group::GroupEncoding, Ciphersuite, Ristretto};
|
|
|
|
use scale::{Encode, Decode, MaxEncodedLen};
|
|
use scale_info::TypeInfo;
|
|
|
|
#[cfg(feature = "borsh")]
|
|
use borsh::{BorshSerialize, BorshDeserialize};
|
|
#[cfg(feature = "serde")]
|
|
use serde::{Serialize, Deserialize};
|
|
|
|
use sp_core::{ConstU32, sr25519::Public, bounded::BoundedVec};
|
|
#[cfg(not(feature = "std"))]
|
|
use sp_std::vec::Vec;
|
|
|
|
use serai_primitives::NetworkId;
|
|
|
|
/// The maximum amount of key shares per set.
|
|
pub const MAX_KEY_SHARES_PER_SET: u32 = 150;
|
|
// Support keys up to 96 bytes (BLS12-381 G2).
|
|
pub const MAX_KEY_LEN: u32 = 96;
|
|
|
|
/// The type used to identify a specific session of validators.
|
|
#[derive(
|
|
Clone, Copy, PartialEq, Eq, Hash, Default, Debug, Encode, Decode, TypeInfo, MaxEncodedLen,
|
|
)]
|
|
#[cfg_attr(feature = "std", derive(Zeroize))]
|
|
#[cfg_attr(feature = "borsh", derive(BorshSerialize, BorshDeserialize))]
|
|
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
|
|
pub struct Session(pub u32);
|
|
|
|
/// The type used to identify a specific validator set during a specific session.
|
|
#[derive(Clone, Copy, PartialEq, Eq, Hash, Debug, Encode, Decode, TypeInfo, MaxEncodedLen)]
|
|
#[cfg_attr(feature = "std", derive(Zeroize))]
|
|
#[cfg_attr(feature = "borsh", derive(BorshSerialize, BorshDeserialize))]
|
|
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
|
|
pub struct ValidatorSet {
|
|
pub session: Session,
|
|
pub network: NetworkId,
|
|
}
|
|
|
|
type MaxKeyLen = ConstU32<MAX_KEY_LEN>;
|
|
/// The type representing a Key from an external network.
|
|
pub type ExternalKey = BoundedVec<u8, MaxKeyLen>;
|
|
|
|
/// The key pair for a validator set.
|
|
///
|
|
/// This is their Ristretto key, used for signing Batches, and their key on the external network.
|
|
#[derive(Clone, PartialEq, Eq, Debug, Encode, Decode, TypeInfo, MaxEncodedLen)]
|
|
#[cfg_attr(feature = "borsh", derive(BorshSerialize, BorshDeserialize))]
|
|
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
|
|
pub struct KeyPair(
|
|
#[cfg_attr(
|
|
feature = "borsh",
|
|
borsh(
|
|
serialize_with = "serai_primitives::borsh_serialize_public",
|
|
deserialize_with = "serai_primitives::borsh_deserialize_public"
|
|
)
|
|
)]
|
|
pub Public,
|
|
#[cfg_attr(
|
|
feature = "borsh",
|
|
borsh(
|
|
serialize_with = "serai_primitives::borsh_serialize_bounded_vec",
|
|
deserialize_with = "serai_primitives::borsh_deserialize_bounded_vec"
|
|
)
|
|
)]
|
|
pub ExternalKey,
|
|
);
|
|
#[cfg(feature = "std")]
|
|
impl Zeroize for KeyPair {
|
|
fn zeroize(&mut self) {
|
|
self.0 .0.zeroize();
|
|
self.1.as_mut().zeroize();
|
|
}
|
|
}
|
|
|
|
/// The MuSig context for a validator set.
|
|
pub fn musig_context(set: ValidatorSet) -> Vec<u8> {
|
|
[b"ValidatorSets-musig_key".as_ref(), &set.encode()].concat()
|
|
}
|
|
|
|
/// The MuSig public key for a validator set.
|
|
///
|
|
/// This function panics on invalid input.
|
|
pub fn musig_key(set: ValidatorSet, set_keys: &[Public]) -> Public {
|
|
let mut keys = Vec::new();
|
|
for key in set_keys {
|
|
keys.push(
|
|
<Ristretto as Ciphersuite>::read_G::<&[u8]>(&mut key.0.as_ref())
|
|
.expect("invalid participant"),
|
|
);
|
|
}
|
|
Public(dkg::musig::musig_key::<Ristretto>(&musig_context(set), &keys).unwrap().to_bytes())
|
|
}
|
|
|
|
/// The message for the set_keys signature.
|
|
pub fn set_keys_message(set: &ValidatorSet, key_pair: &KeyPair) -> Vec<u8> {
|
|
(b"ValidatorSets-set_keys", set, key_pair).encode()
|
|
}
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug, Encode, Decode, TypeInfo, MaxEncodedLen)]
|
|
#[cfg_attr(feature = "borsh", derive(BorshSerialize, BorshDeserialize))]
|
|
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
|
|
pub struct Slash {
|
|
#[cfg_attr(
|
|
feature = "borsh",
|
|
borsh(
|
|
serialize_with = "serai_primitives::borsh_serialize_public",
|
|
deserialize_with = "serai_primitives::borsh_deserialize_public"
|
|
)
|
|
)]
|
|
key: Public,
|
|
points: u32,
|
|
}
|
|
#[derive(Clone, PartialEq, Eq, Debug, Encode, Decode, TypeInfo, MaxEncodedLen)]
|
|
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
|
|
pub struct SlashReport(pub BoundedVec<Slash, ConstU32<{ MAX_KEY_SHARES_PER_SET / 3 }>>);
|
|
|
|
pub fn report_slashes_message(set: &ValidatorSet, slashes: &SlashReport) -> Vec<u8> {
|
|
(b"ValidatorSets-report_slashes", set, slashes).encode()
|
|
}
|
|
|
|
/// For a set of validators whose key shares may exceed the maximum, reduce until they equal the
|
|
/// maximum.
|
|
///
|
|
/// Reduction occurs by reducing each validator in a reverse round-robin.
|
|
pub fn amortize_excess_key_shares(validators: &mut [(Public, u64)]) {
|
|
let total_key_shares = validators.iter().map(|(_, shares)| shares).sum::<u64>();
|
|
for i in 0 .. usize::try_from(total_key_shares.saturating_sub(u64::from(MAX_KEY_SHARES_PER_SET)))
|
|
.unwrap()
|
|
{
|
|
validators[validators.len() - ((i % validators.len()) + 1)].1 -= 1;
|
|
}
|
|
}
|
|
|
|
/// Returns the post-amortization key shares for the top validator.
|
|
///
|
|
/// Panics when `validators == 0`.
|
|
pub fn post_amortization_key_shares_for_top_validator(
|
|
validators: usize,
|
|
top: u64,
|
|
key_shares: u64,
|
|
) -> u64 {
|
|
top -
|
|
(key_shares.saturating_sub(MAX_KEY_SHARES_PER_SET.into()) /
|
|
u64::try_from(validators).unwrap())
|
|
}
|