tor_netdir/
testprovider.rs
1use std::sync::{Arc, Mutex};
4
5use crate::{DirEvent, Error, NetDir, NetDirProvider, Result};
6
7use postage::broadcast::{self, Receiver, Sender};
8use postage::sink::Sink as _;
9
10#[derive(Debug)]
21pub struct TestNetDirProvider {
22 inner: Mutex<Inner>,
24}
25
26#[derive(Debug)]
28struct Inner {
29 current: Option<Arc<NetDir>>,
31 event_tx: Sender<DirEvent>,
33 _event_rx: Receiver<DirEvent>,
35}
36
37#[allow(clippy::new_without_default)]
38impl TestNetDirProvider {
39 pub fn new() -> Self {
41 let (event_tx, _event_rx) = broadcast::channel(128);
42 let inner = Inner {
43 current: None,
44 event_tx,
45 _event_rx,
46 };
47
48 Self {
49 inner: Mutex::new(inner),
50 }
51 }
52
53 pub fn set_netdir(&self, dir: impl Into<Arc<NetDir>>) {
55 let mut inner = self.inner.lock().expect("lock poisoned");
56 inner.current = Some(dir.into());
57 }
58
59 pub async fn set_netdir_and_notify(&self, dir: impl Into<Arc<NetDir>>) {
62 let mut event_tx = {
63 let mut inner = self.inner.lock().expect("lock poisoned");
64 inner.current = Some(dir.into());
65 inner.event_tx.clone()
66 };
67 event_tx
68 .send(DirEvent::NewConsensus)
69 .await
70 .expect("receivers were dropped");
71 }
72}
73
74impl From<NetDir> for TestNetDirProvider {
75 fn from(nd: NetDir) -> Self {
76 let rv = Self::new();
77 rv.set_netdir(nd);
78 rv
79 }
80}
81
82impl NetDirProvider for TestNetDirProvider {
83 fn netdir(&self, _timeliness: crate::Timeliness) -> Result<Arc<NetDir>> {
84 match self.inner.lock().expect("lock poisoned").current.as_ref() {
85 Some(netdir) => Ok(Arc::clone(netdir)),
86 None => Err(Error::NoInfo),
87 }
88 }
89
90 fn events(&self) -> futures::stream::BoxStream<'static, DirEvent> {
91 let inner = self.inner.lock().expect("lock poisoned");
92 let events = inner.event_tx.subscribe();
93 Box::pin(events)
94 }
95
96 fn params(&self) -> Arc<dyn AsRef<crate::params::NetParameters>> {
97 if let Ok(nd) = self.netdir(crate::Timeliness::Unchecked) {
98 nd
99 } else {
100 Arc::new(crate::params::NetParameters::default())
101 }
102 }
103
104 fn protocol_statuses(
105 &self,
106 ) -> Option<(
107 std::time::SystemTime,
108 Arc<tor_netdoc::doc::netstatus::ProtoStatuses>,
109 )> {
110 None
111 }
112}