use crate::simplex::{
scheme::Scheme,
types::{Nullify, Vote},
};
use commonware_codec::{DecodeExt, Encode};
use commonware_cryptography::{certificate, Hasher};
use commonware_p2p::{Receiver, Recipients, Sender};
use commonware_runtime::{spawn_cell, ContextCell, Handle, Spawner};
use std::marker::PhantomData;
use tracing::debug;
pub struct Config<S: certificate::Scheme> {
pub scheme: S,
}
pub struct NullifyOnly<E: Spawner, S: Scheme<H::Digest>, H: Hasher> {
context: ContextCell<E>,
scheme: S,
_hasher: PhantomData<H>,
}
impl<E: Spawner, S: Scheme<H::Digest>, H: Hasher> NullifyOnly<E, S, H> {
pub fn new(context: E, cfg: Config<S>) -> Self {
Self {
context: ContextCell::new(context),
scheme: cfg.scheme,
_hasher: PhantomData,
}
}
pub fn start(mut self, vote_network: (impl Sender, impl Receiver)) -> Handle<()> {
spawn_cell!(self.context, self.run(vote_network).await)
}
async fn run(self, vote_network: (impl Sender, impl Receiver)) {
let (mut sender, mut receiver) = vote_network;
while let Ok((s, msg)) = receiver.recv().await {
let msg = match Vote::<S, H::Digest>::decode(msg) {
Ok(msg) => msg,
Err(err) => {
debug!(?err, sender = ?s, "failed to decode message");
continue;
}
};
if let Vote::Notarize(notarize) = msg {
let nullify = Nullify::sign::<H::Digest>(&self.scheme, notarize.round()).unwrap();
let msg = Vote::<S, H::Digest>::Nullify(nullify).encode();
sender.send(Recipients::All, msg, true).await.unwrap();
}
}
}
}