use std::io::Read; use scale::{Encode, Decode}; use ciphersuite::{group::GroupEncoding, Ciphersuite, Ristretto}; use serai_client::validator_sets::primitives::{ValidatorSet, KeyPair}; pub use serai_db::*; // Used to determine if an ID is acceptable #[derive(Clone, Copy, PartialEq, Eq, Debug)] pub enum Zone { Dkg, Batch, Sign, } impl Zone { fn label(&self) -> &'static str { match self { Zone::Dkg => "dkg", Zone::Batch => "batch", Zone::Sign => "sign", } } } // A struct to refer to a piece of data all validators will presumably provide a value for. #[derive(Clone, Copy, PartialEq, Eq, Debug)] pub struct DataSpecification { pub zone: Zone, pub label: &'static str, pub id: [u8; 32], pub attempt: u32, } impl DataSpecification { fn as_key(&self, genesis: [u8; 32]) -> Vec { // TODO: Use a proper transcript here to avoid conflicts? [ genesis.as_ref(), self.zone.label().as_bytes(), self.label.as_bytes(), self.id.as_ref(), self.attempt.to_le_bytes().as_ref(), ] .concat() } } #[derive(Debug)] pub struct TributaryDb(pub D); impl TributaryDb { pub fn new(db: D) -> Self { Self(db) } fn tributary_key(dst: &'static [u8], key: impl AsRef<[u8]>) -> Vec { D::key(b"coordinator_tributary", dst, key) } fn block_key(genesis: [u8; 32]) -> Vec { Self::tributary_key(b"block", genesis) } pub fn set_last_block(&mut self, genesis: [u8; 32], block: [u8; 32]) { let mut txn = self.0.txn(); txn.put(Self::block_key(genesis), block); txn.commit(); } pub fn last_block(&self, genesis: [u8; 32]) -> [u8; 32] { self.0.get(Self::block_key(genesis)).map(|last| last.try_into().unwrap()).unwrap_or(genesis) } fn fatal_slash_key(genesis: [u8; 32]) -> Vec { Self::tributary_key(b"fatal_slash", genesis) } pub fn set_fatally_slashed(txn: &mut D::Transaction<'_>, genesis: [u8; 32], id: [u8; 32]) { let key = Self::fatal_slash_key(genesis); let mut existing = txn.get(&key).unwrap_or(vec![]); // don't append if we already have it. if existing.chunks(32).any(|ex_id| ex_id == id) { return; } existing.extend(id); txn.put(key, existing); } fn plan_ids_key(genesis: &[u8], block: u64) -> Vec { Self::tributary_key(b"plan_ids", [genesis, block.to_le_bytes().as_ref()].concat()) } pub fn set_plan_ids( txn: &mut D::Transaction<'_>, genesis: [u8; 32], block: u64, plans: &[[u8; 32]], ) { txn.put(Self::plan_ids_key(&genesis, block), plans.concat()); } pub fn plan_ids(getter: &G, genesis: [u8; 32], block: u64) -> Option> { getter.get(Self::plan_ids_key(&genesis, block)).map(|bytes| { let mut res = vec![]; let mut bytes_ref: &[u8] = bytes.as_ref(); while !bytes_ref.is_empty() { let mut id = [0; 32]; bytes_ref.read_exact(&mut id).unwrap(); res.push(id); } res }) } fn currently_completing_key_pair_key(genesis: [u8; 32]) -> Vec { Self::tributary_key(b"currently_completing_key_pair", genesis) } pub fn save_currently_completing_key_pair( txn: &mut D::Transaction<'_>, genesis: [u8; 32], key_pair: &KeyPair, ) { txn.put(Self::currently_completing_key_pair_key(genesis), key_pair.encode()) } pub fn currently_completing_key_pair(getter: &G, genesis: [u8; 32]) -> Option { getter .get(Self::currently_completing_key_pair_key(genesis)) .map(|bytes| KeyPair::decode(&mut bytes.as_slice()).unwrap()) } pub fn key_pair_key(set: ValidatorSet) -> Vec { Self::tributary_key(b"key_pair", set.encode()) } pub fn set_key_pair(txn: &mut D::Transaction<'_>, set: ValidatorSet, key_pair: &KeyPair) { txn.put(Self::key_pair_key(set), key_pair.encode()); } pub fn key_pair(getter: &G, set: ValidatorSet) -> Option { Some(KeyPair::decode(&mut getter.get(Self::key_pair_key(set))?.as_slice()).unwrap()) } fn recognized_id_key(genesis: [u8; 32], zone: Zone, id: [u8; 32]) -> Vec { Self::tributary_key( b"recognized", [genesis.as_ref(), zone.label().as_bytes(), id.as_ref()].concat(), ) } pub fn recognized_id(getter: &G, genesis: [u8; 32], zone: Zone, id: [u8; 32]) -> bool { getter.get(Self::recognized_id_key(genesis, zone, id)).is_some() } pub fn recognize_id(txn: &mut D::Transaction<'_>, genesis: [u8; 32], zone: Zone, id: [u8; 32]) { txn.put(Self::recognized_id_key(genesis, zone, id), []) } fn attempt_key(genesis: [u8; 32], id: [u8; 32]) -> Vec { let genesis_ref: &[u8] = genesis.as_ref(); Self::tributary_key(b"attempt", [genesis_ref, id.as_ref()].concat()) } pub fn attempt(getter: &G, genesis: [u8; 32], id: [u8; 32]) -> u32 { u32::from_le_bytes( getter.get(Self::attempt_key(genesis, id)).unwrap_or(vec![0; 4]).try_into().unwrap(), ) } // Key for the amount of instances received thus far fn data_received_key(genesis: [u8; 32], data_spec: &DataSpecification) -> Vec { Self::tributary_key(b"data_received", data_spec.as_key(genesis)) } fn data_key( genesis: [u8; 32], data_spec: &DataSpecification, signer: ::G, ) -> Vec { Self::tributary_key( b"data", [data_spec.as_key(genesis).as_slice(), signer.to_bytes().as_ref()].concat(), ) } pub fn data( getter: &G, genesis: [u8; 32], data_spec: &DataSpecification, signer: ::G, ) -> Option> { getter.get(Self::data_key(genesis, data_spec, signer)) } pub fn set_data( txn: &mut D::Transaction<'_>, genesis: [u8; 32], data_spec: &DataSpecification, signer: ::G, data: &[u8], ) -> u16 { let received_key = Self::data_received_key(genesis, data_spec); let mut received = u16::from_le_bytes(txn.get(&received_key).unwrap_or(vec![0; 2]).try_into().unwrap()); received += 1; txn.put(received_key, received.to_le_bytes()); txn.put(Self::data_key(genesis, data_spec, signer), data); received } fn event_key(id: &[u8], index: u32) -> Vec { Self::tributary_key(b"event", [id, index.to_le_bytes().as_ref()].concat()) } pub fn handled_event(getter: &G, id: [u8; 32], index: u32) -> bool { getter.get(Self::event_key(&id, index)).is_some() } pub fn handle_event(txn: &mut D::Transaction<'_>, id: [u8; 32], index: u32) { assert!(!Self::handled_event(txn, id, index)); txn.put(Self::event_key(&id, index), []); } }