mirror of
https://github.com/hinto-janai/cuprate.git
synced 2025-01-26 04:15:54 +00:00
19150df355
* dandelion-tower: add/fix workspace lints * fmt * fixes * todos * fixes * fixes * expect reason
354 lines
12 KiB
Rust
354 lines
12 KiB
Rust
//! # Dandelion++ Router
|
|
//!
|
|
//! This module contains [`DandelionRouter`] which is a [`Service`]. It that handles keeping the
|
|
//! current dandelion++ [`State`] and deciding where to send transactions based on their [`TxState`].
|
|
//!
|
|
//! ### What The Router Does Not Do
|
|
//!
|
|
//! It does not handle anything to do with keeping transactions long term, i.e. embargo timers and handling
|
|
//! loops in the stem. It is up to implementers to do this if they decide not to use [`DandelionPool`](crate::pool::DandelionPoolManager)
|
|
use std::{
|
|
collections::HashMap,
|
|
hash::Hash,
|
|
marker::PhantomData,
|
|
pin::Pin,
|
|
task::{ready, Context, Poll},
|
|
time::Instant,
|
|
};
|
|
|
|
use futures::{future::BoxFuture, FutureExt, TryFutureExt, TryStream};
|
|
use rand::{distributions::Bernoulli, prelude::*, thread_rng};
|
|
use tower::Service;
|
|
|
|
use crate::{
|
|
traits::{DiffuseRequest, StemRequest},
|
|
DandelionConfig,
|
|
};
|
|
|
|
/// An error returned from the [`DandelionRouter`]
|
|
#[derive(thiserror::Error, Debug)]
|
|
pub enum DandelionRouterError {
|
|
/// This error is probably recoverable so the request should be retried.
|
|
#[error("Peer chosen to route stem txs to had an err: {0}.")]
|
|
PeerError(tower::BoxError),
|
|
/// The broadcast service returned an error.
|
|
#[error("Broadcast service returned an err: {0}.")]
|
|
BroadcastError(tower::BoxError),
|
|
/// The outbound peer stream returned an error, this is critical.
|
|
#[error("The outbound peer stream returned an err: {0}.")]
|
|
OutboundPeerStreamError(tower::BoxError),
|
|
/// The outbound peer discoverer returned [`None`].
|
|
#[error("The outbound peer discoverer exited.")]
|
|
OutboundPeerDiscoverExited,
|
|
}
|
|
|
|
/// A response from an attempt to retrieve an outbound peer.
|
|
pub enum OutboundPeer<Id, T> {
|
|
/// A peer.
|
|
Peer(Id, T),
|
|
/// The peer store is exhausted and has no more to return.
|
|
Exhausted,
|
|
}
|
|
|
|
/// The dandelion++ state.
|
|
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
|
|
pub enum State {
|
|
/// Fluff state, in this state we are diffusing stem transactions to all peers.
|
|
Fluff,
|
|
/// Stem state, in this state we are stemming stem transactions to a single outbound peer.
|
|
Stem,
|
|
}
|
|
|
|
/// The routing state of a transaction.
|
|
#[derive(Debug, Clone, Eq, PartialEq)]
|
|
pub enum TxState<Id> {
|
|
/// Fluff state.
|
|
Fluff,
|
|
/// Stem state.
|
|
Stem {
|
|
/// The peer who sent us this transaction's Id.
|
|
from: Id,
|
|
},
|
|
/// Local - the transaction originated from our node.
|
|
Local,
|
|
}
|
|
|
|
/// A request to route a transaction.
|
|
pub struct DandelionRouteReq<Tx, Id> {
|
|
/// The transaction.
|
|
pub tx: Tx,
|
|
/// The transaction state.
|
|
pub state: TxState<Id>,
|
|
}
|
|
|
|
/// The dandelion router service.
|
|
pub struct DandelionRouter<P, B, Id, S, Tx> {
|
|
// pub(crate) is for tests
|
|
/// A [`Discover`] where we can get outbound peers from.
|
|
outbound_peer_discover: Pin<Box<P>>,
|
|
/// A [`Service`] which handle broadcasting (diffusing) transactions.
|
|
broadcast_svc: B,
|
|
|
|
/// The current state.
|
|
current_state: State,
|
|
/// The time at which this epoch started.
|
|
epoch_start: Instant,
|
|
|
|
/// The stem our local transactions will be sent to.
|
|
local_route: Option<Id>,
|
|
/// A [`HashMap`] linking peer's Ids to Ids in `stem_peers`.
|
|
stem_routes: HashMap<Id, Id>,
|
|
/// Peers we are using for stemming.
|
|
///
|
|
/// This will contain peers, even in [`State::Fluff`] to allow us to stem [`TxState::Local`]
|
|
/// transactions.
|
|
pub(crate) stem_peers: HashMap<Id, S>,
|
|
|
|
/// The distribution to sample to get the [`State`], true is [`State::Fluff`].
|
|
state_dist: Bernoulli,
|
|
|
|
/// The config.
|
|
config: DandelionConfig,
|
|
|
|
/// The routers tracing span.
|
|
span: tracing::Span,
|
|
|
|
_tx: PhantomData<Tx>,
|
|
}
|
|
|
|
impl<Tx, Id, P, B, S> DandelionRouter<P, B, Id, S, Tx>
|
|
where
|
|
Id: Hash + Eq + Clone,
|
|
P: TryStream<Ok = OutboundPeer<Id, S>, Error = tower::BoxError>,
|
|
B: Service<DiffuseRequest<Tx>, Error = tower::BoxError>,
|
|
B::Future: Send + 'static,
|
|
S: Service<StemRequest<Tx>, Error = tower::BoxError>,
|
|
S::Future: Send + 'static,
|
|
{
|
|
/// Creates a new [`DandelionRouter`], with the provided services and config.
|
|
///
|
|
/// # Panics
|
|
/// This function panics if [`DandelionConfig::fluff_probability`] is not `0.0..=1.0`.
|
|
pub fn new(broadcast_svc: B, outbound_peer_discover: P, config: DandelionConfig) -> Self {
|
|
// get the current state
|
|
let state_dist = Bernoulli::new(config.fluff_probability)
|
|
.expect("Fluff probability was not between 0 and 1");
|
|
|
|
let current_state = if state_dist.sample(&mut thread_rng()) {
|
|
State::Fluff
|
|
} else {
|
|
State::Stem
|
|
};
|
|
|
|
Self {
|
|
outbound_peer_discover: Box::pin(outbound_peer_discover),
|
|
broadcast_svc,
|
|
current_state,
|
|
epoch_start: Instant::now(),
|
|
local_route: None,
|
|
stem_routes: HashMap::new(),
|
|
stem_peers: HashMap::new(),
|
|
state_dist,
|
|
config,
|
|
span: tracing::debug_span!("dandelion_router", state = ?current_state),
|
|
_tx: PhantomData,
|
|
}
|
|
}
|
|
|
|
/// This function gets the number of outbound peers from the [`Discover`] required for the selected [`Graph`](crate::Graph).
|
|
fn poll_prepare_graph(
|
|
&mut self,
|
|
cx: &mut Context<'_>,
|
|
) -> Poll<Result<(), DandelionRouterError>> {
|
|
let peers_needed = match self.current_state {
|
|
State::Stem => self.config.number_of_stems(),
|
|
// When in the fluff state we only need one peer, the one for our txs.
|
|
State::Fluff => 1,
|
|
};
|
|
|
|
while self.stem_peers.len() < peers_needed {
|
|
match ready!(self
|
|
.outbound_peer_discover
|
|
.as_mut()
|
|
.try_poll_next(cx)
|
|
.map_err(DandelionRouterError::OutboundPeerStreamError))
|
|
.ok_or(DandelionRouterError::OutboundPeerDiscoverExited)??
|
|
{
|
|
OutboundPeer::Peer(key, svc) => {
|
|
self.stem_peers.insert(key, svc);
|
|
}
|
|
OutboundPeer::Exhausted => {
|
|
tracing::warn!("Failed to retrieve enough outbound peers for optimal dandelion++, privacy may be degraded.");
|
|
return Poll::Ready(Ok(()));
|
|
}
|
|
}
|
|
}
|
|
|
|
Poll::Ready(Ok(()))
|
|
}
|
|
|
|
fn fluff_tx(&mut self, tx: Tx) -> BoxFuture<'static, Result<State, DandelionRouterError>> {
|
|
self.broadcast_svc
|
|
.call(DiffuseRequest(tx))
|
|
.map_ok(|_| State::Fluff)
|
|
.map_err(DandelionRouterError::BroadcastError)
|
|
.boxed()
|
|
}
|
|
|
|
fn stem_tx(
|
|
&mut self,
|
|
tx: Tx,
|
|
from: &Id,
|
|
) -> BoxFuture<'static, Result<State, DandelionRouterError>> {
|
|
if self.stem_peers.is_empty() {
|
|
tracing::debug!("Stem peers are empty, fluffing stem transaction.");
|
|
return self.fluff_tx(tx);
|
|
}
|
|
|
|
loop {
|
|
let stem_route = self.stem_routes.entry(from.clone()).or_insert_with(|| {
|
|
self.stem_peers
|
|
.iter()
|
|
.choose(&mut thread_rng())
|
|
.expect("No peers in `stem_peers` was poll_ready called?")
|
|
.0
|
|
.clone()
|
|
});
|
|
|
|
let Some(peer) = self.stem_peers.get_mut(stem_route) else {
|
|
self.stem_routes.remove(from);
|
|
continue;
|
|
};
|
|
|
|
return peer
|
|
.call(StemRequest(tx))
|
|
.map_ok(|_| State::Stem)
|
|
.map_err(DandelionRouterError::PeerError)
|
|
.boxed();
|
|
}
|
|
}
|
|
|
|
fn stem_local_tx(&mut self, tx: Tx) -> BoxFuture<'static, Result<State, DandelionRouterError>> {
|
|
if self.stem_peers.is_empty() {
|
|
tracing::warn!("Stem peers are empty, no outbound connections to stem local tx to, fluffing instead, privacy will be degraded.");
|
|
return self.fluff_tx(tx);
|
|
}
|
|
|
|
loop {
|
|
let stem_route = self.local_route.get_or_insert_with(|| {
|
|
self.stem_peers
|
|
.iter()
|
|
.choose(&mut thread_rng())
|
|
.expect("No peers in `stem_peers` was poll_ready called?")
|
|
.0
|
|
.clone()
|
|
});
|
|
|
|
let Some(peer) = self.stem_peers.get_mut(stem_route) else {
|
|
self.local_route.take();
|
|
continue;
|
|
};
|
|
|
|
return peer
|
|
.call(StemRequest(tx))
|
|
.map_ok(|_| State::Stem)
|
|
.map_err(DandelionRouterError::PeerError)
|
|
.boxed();
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<Tx, Id, P, B, S> Service<DandelionRouteReq<Tx, Id>> for DandelionRouter<P, B, Id, S, Tx>
|
|
where
|
|
Id: Hash + Eq + Clone,
|
|
P: TryStream<Ok = OutboundPeer<Id, S>, Error = tower::BoxError>,
|
|
B: Service<DiffuseRequest<Tx>, Error = tower::BoxError>,
|
|
B::Future: Send + 'static,
|
|
S: Service<StemRequest<Tx>, Error = tower::BoxError>,
|
|
S::Future: Send + 'static,
|
|
{
|
|
type Response = State;
|
|
type Error = DandelionRouterError;
|
|
type Future = BoxFuture<'static, Result<State, DandelionRouterError>>;
|
|
|
|
fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
|
if self.epoch_start.elapsed() > self.config.epoch_duration {
|
|
// clear all the stem routing data.
|
|
self.stem_peers.clear();
|
|
self.stem_routes.clear();
|
|
self.local_route.take();
|
|
|
|
self.current_state = if self.state_dist.sample(&mut thread_rng()) {
|
|
State::Fluff
|
|
} else {
|
|
State::Stem
|
|
};
|
|
|
|
self.span
|
|
.record("state", format!("{:?}", self.current_state));
|
|
tracing::debug!(parent: &self.span, "Starting new d++ epoch",);
|
|
|
|
self.epoch_start = Instant::now();
|
|
}
|
|
|
|
let mut peers_pending = false;
|
|
|
|
let span = &self.span;
|
|
|
|
self.stem_peers
|
|
.retain(|_, peer_svc| match peer_svc.poll_ready(cx) {
|
|
Poll::Ready(res) => res
|
|
.inspect_err(|e| {
|
|
tracing::debug!(
|
|
parent: span,
|
|
"Peer returned an error on `poll_ready`: {e}, removing from router.",
|
|
);
|
|
})
|
|
.is_ok(),
|
|
Poll::Pending => {
|
|
// Pending peers should be kept - they have not errored yet.
|
|
peers_pending = true;
|
|
true
|
|
}
|
|
});
|
|
|
|
if peers_pending {
|
|
return Poll::Pending;
|
|
}
|
|
|
|
// now we have removed the failed peers check if we still have enough for the graph chosen.
|
|
ready!(self.poll_prepare_graph(cx)?);
|
|
|
|
ready!(self
|
|
.broadcast_svc
|
|
.poll_ready(cx)
|
|
.map_err(DandelionRouterError::BroadcastError)?);
|
|
|
|
Poll::Ready(Ok(()))
|
|
}
|
|
|
|
fn call(&mut self, req: DandelionRouteReq<Tx, Id>) -> Self::Future {
|
|
tracing::trace!(parent: &self.span, "Handling route request.");
|
|
|
|
match req.state {
|
|
TxState::Fluff => self.fluff_tx(req.tx),
|
|
TxState::Stem { from } => match self.current_state {
|
|
State::Fluff => {
|
|
tracing::debug!(parent: &self.span, "Fluffing stem tx.");
|
|
|
|
self.fluff_tx(req.tx)
|
|
}
|
|
State::Stem => {
|
|
tracing::trace!(parent: &self.span, "Steming transaction");
|
|
|
|
self.stem_tx(req.tx, &from)
|
|
}
|
|
},
|
|
TxState::Local => {
|
|
tracing::debug!(parent: &self.span, "Steming local tx.");
|
|
|
|
self.stem_local_tx(req.tx)
|
|
}
|
|
}
|
|
}
|
|
}
|