mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-01 16:39:53 +00:00
6f6c9f7cdf
It's needed by the processor and tributary (coordinator).
96 lines
2.7 KiB
Rust
96 lines
2.7 KiB
Rust
use core::marker::PhantomData;
|
|
|
|
pub use serai_db::*;
|
|
|
|
use crate::{Plan, coins::Coin};
|
|
|
|
#[derive(Debug)]
|
|
pub struct MainDb<C: Coin, D: Db>(D, PhantomData<C>);
|
|
impl<C: Coin, D: Db> MainDb<C, D> {
|
|
pub fn new(db: D) -> Self {
|
|
Self(db, PhantomData)
|
|
}
|
|
|
|
fn main_key(dst: &'static [u8], key: impl AsRef<[u8]>) -> Vec<u8> {
|
|
D::key(b"MAIN", dst, key)
|
|
}
|
|
|
|
fn plan_key(id: &[u8]) -> Vec<u8> {
|
|
Self::main_key(b"plan", id)
|
|
}
|
|
fn signing_key(key: &[u8]) -> Vec<u8> {
|
|
Self::main_key(b"signing", key)
|
|
}
|
|
pub fn save_signing(&mut self, key: &[u8], block_number: u64, time: u64, plan: &Plan<C>) {
|
|
let id = plan.id();
|
|
// Creating a TXN here is arguably an anti-pattern, yet nothing here expects atomicity
|
|
let mut txn = self.0.txn();
|
|
|
|
{
|
|
let mut signing = txn.get(Self::signing_key(key)).unwrap_or(vec![]);
|
|
|
|
// If we've already noted we're signing this, return
|
|
assert_eq!(signing.len() % 32, 0);
|
|
for i in 0 .. (signing.len() / 32) {
|
|
if signing[(i * 32) .. ((i + 1) * 32)] == id {
|
|
return;
|
|
}
|
|
}
|
|
|
|
signing.extend(&id);
|
|
txn.put(Self::signing_key(key), id);
|
|
}
|
|
|
|
{
|
|
let mut buf = block_number.to_le_bytes().to_vec();
|
|
buf.extend(&time.to_le_bytes());
|
|
plan.write(&mut buf).unwrap();
|
|
txn.put(Self::plan_key(&id), &buf);
|
|
}
|
|
|
|
txn.commit();
|
|
}
|
|
|
|
pub fn signing(&self, key: &[u8]) -> Vec<(u64, u64, Plan<C>)> {
|
|
let signing = self.0.get(Self::signing_key(key)).unwrap_or(vec![]);
|
|
let mut res = vec![];
|
|
|
|
assert_eq!(signing.len() % 32, 0);
|
|
for i in 0 .. (signing.len() / 32) {
|
|
let id = &signing[(i * 32) .. ((i + 1) * 32)];
|
|
let buf = self.0.get(Self::plan_key(id)).unwrap();
|
|
|
|
let block_number = u64::from_le_bytes(buf[.. 8].try_into().unwrap());
|
|
let time = u64::from_le_bytes(buf[8 .. 16].try_into().unwrap());
|
|
let plan = Plan::<C>::read::<&[u8]>(&mut &buf[16 ..]).unwrap();
|
|
assert_eq!(id, &plan.id());
|
|
res.push((block_number, time, plan));
|
|
}
|
|
|
|
res
|
|
}
|
|
|
|
pub fn finish_signing(&mut self, key: &[u8], id: [u8; 32]) {
|
|
let mut signing = self.0.get(Self::signing_key(key)).unwrap_or(vec![]);
|
|
assert_eq!(signing.len() % 32, 0);
|
|
|
|
let mut found = false;
|
|
for i in 0 .. (signing.len() / 32) {
|
|
let start = i * 32;
|
|
let end = i + 32;
|
|
if signing[start .. end] == id {
|
|
found = true;
|
|
signing = [&signing[.. start], &signing[end ..]].concat().to_vec();
|
|
break;
|
|
}
|
|
}
|
|
|
|
if !found {
|
|
log::warn!("told to finish signing {} yet wasn't actively signing it", hex::encode(id));
|
|
}
|
|
|
|
let mut txn = self.0.txn();
|
|
txn.put(Self::signing_key(key), signing);
|
|
txn.commit();
|
|
}
|
|
}
|