use std::collections::HashMap; use zeroize::Zeroize; use rand_core::{RngCore, SeedableRng}; use rand_chacha::ChaCha8Rng; use transcript::{Transcript, RecommendedTranscript}; use serde::{Serialize, Deserialize}; use dkg::{Participant, ThresholdParams}; use serai_primitives::WithAmount; use in_instructions_primitives::InInstruction; use tokens_primitives::OutInstruction; use validator_sets_primitives::ValidatorSetInstance; #[derive(Clone, Copy, PartialEq, Eq, Debug, Zeroize, Serialize, Deserialize)] pub struct SubstrateContext { pub time: u64, pub coin_latest_block_number: u64, } pub mod key_gen { use super::*; #[derive(Clone, Copy, PartialEq, Eq, Hash, Debug, Zeroize, Serialize, Deserialize)] pub struct KeyGenId { pub set: ValidatorSetInstance, pub attempt: u32, } #[derive(Clone, PartialEq, Eq, Debug, Serialize, Deserialize)] pub enum CoordinatorMessage { // Instructs the Processor to begin the key generation process. GenerateKey { id: KeyGenId, params: ThresholdParams }, // Received commitments for the specified key generation protocol. Commitments { id: KeyGenId, commitments: HashMap> }, // Received shares for the specified key generation protocol. Shares { id: KeyGenId, shares: HashMap> }, // Confirm a key. ConfirmKey { context: SubstrateContext, id: KeyGenId }, } #[derive(Clone, PartialEq, Eq, Debug, Serialize, Deserialize)] pub enum ProcessorMessage { // Created commitments for the specified key generation protocol. Commitments { id: KeyGenId, commitments: Vec }, // Created shares for the specified key generation protocol. Shares { id: KeyGenId, shares: HashMap> }, // Resulting key from the specified key generation protocol. GeneratedKey { id: KeyGenId, key: Vec }, } } pub mod sign { use super::*; #[derive(Clone, PartialEq, Eq, Hash, Debug, Zeroize, Serialize, Deserialize)] pub struct SignId { pub key: Vec, pub id: [u8; 32], pub attempt: u32, } impl SignId { /// Determine a signing set for a given signing session. // TODO: Replace with ROAST or the first available group of signers. // https://github.com/serai-dex/serai/issues/163 pub fn signing_set(&self, params: &ThresholdParams) -> Vec { let mut transcript = RecommendedTranscript::new(b"SignId signing_set"); transcript.domain_separate(b"SignId"); transcript.append_message(b"key", &self.key); transcript.append_message(b"id", self.id); transcript.append_message(b"attempt", self.attempt.to_le_bytes()); let mut candidates = (1 ..= params.n()).map(|i| Participant::new(i).unwrap()).collect::>(); let mut rng = ChaCha8Rng::from_seed(transcript.rng_seed(b"signing_set")); while candidates.len() > params.t().into() { candidates.swap_remove( usize::try_from(rng.next_u64() % u64::try_from(candidates.len()).unwrap()).unwrap(), ); } candidates } } #[derive(Clone, PartialEq, Eq, Debug, Serialize, Deserialize)] pub enum CoordinatorMessage { // Received preprocesses for the specified signing protocol. Preprocesses { id: SignId, preprocesses: HashMap> }, // Received shares for the specified signing protocol. Shares { id: SignId, shares: HashMap> }, // Completed a signing protocol already. Completed { key: Vec, id: [u8; 32], tx: Vec }, } #[derive(Clone, PartialEq, Eq, Debug, Zeroize, Serialize, Deserialize)] pub enum ProcessorMessage { // Created preprocess for the specified signing protocol. Preprocess { id: SignId, preprocess: Vec }, // Signed share for the specified signing protocol. Share { id: SignId, share: Vec }, // Completed a signing protocol already. Completed { key: Vec, id: [u8; 32], tx: Vec }, } impl CoordinatorMessage { pub fn key(&self) -> &[u8] { match self { CoordinatorMessage::Preprocesses { id, .. } => &id.key, CoordinatorMessage::Shares { id, .. } => &id.key, CoordinatorMessage::Completed { key, .. } => key, } } } } pub mod substrate { use super::*; #[derive(Clone, PartialEq, Eq, Debug, Zeroize, Serialize, Deserialize)] pub enum CoordinatorMessage { BlockAcknowledged { context: SubstrateContext, key: Vec, block: Vec }, Burns { context: SubstrateContext, burns: Vec> }, } #[derive(Clone, PartialEq, Eq, Debug, Zeroize, Serialize, Deserialize)] pub enum ProcessorMessage { Update { key: Vec, block: Vec, instructions: Vec> }, } } #[derive(Clone, PartialEq, Eq, Debug, Serialize, Deserialize)] pub enum CoordinatorMessage { KeyGen(key_gen::CoordinatorMessage), Sign(sign::CoordinatorMessage), Substrate(substrate::CoordinatorMessage), } #[derive(Clone, PartialEq, Eq, Debug, Serialize, Deserialize)] pub enum ProcessorMessage { KeyGen(key_gen::ProcessorMessage), Sign(sign::ProcessorMessage), Substrate(substrate::ProcessorMessage), }