use derive_deftly::define_derive_deftly;
use futures::channel::mpsc;
define_derive_deftly! {
SomeMockRuntime for struct, expect items, beta_deftly:
$(
${when fmeta(mock(task))}
impl <$tgens> Spawn for $ttype {
fn spawn_obj(&self, future: FutureObj<'static, ()>) -> Result<(), SpawnError> {
self.$fname.spawn_obj(future)
}
}
impl <$tgens> Blocking for $ttype {
type ThreadHandle<T: Send + 'static> = <$ftype as Blocking>::ThreadHandle<T>;
fn spawn_blocking<F, T>(&self, f: F) -> <$ftype as Blocking>::ThreadHandle<T>
where
F: FnOnce() -> T + Send + 'static,
T: Send + 'static {
self.$fname.spawn_blocking(f)
}
fn reenter_block_on<F>(&self, future: F) -> F::Output
where
F: Future,
F::Output: Send + 'static
{
self.$fname.reenter_block_on(future)
}
}
)
$(
${when any(fmeta(mock(toplevel)), fmeta(mock(toplevel_where)))}
impl <$tgens> ToplevelBlockOn for $ttype
where ${fmeta(mock(toplevel_where)) as token_stream, default {}}
{
fn block_on<F: Future>(&self, future: F) -> F::Output {
self.$fname.block_on(future)
}
}
)
$(
${when fmeta(mock(net))}
#[async_trait]
impl <$tgens> NetStreamProvider for $ttype {
type Stream = <$ftype as NetStreamProvider>::Stream;
type Listener = <$ftype as NetStreamProvider>::Listener;
async fn connect(&self, addr: &SocketAddr) -> IoResult<Self::Stream> {
self.$fname.connect(addr).await
}
async fn listen(&self, addr: &SocketAddr) -> IoResult<Self::Listener> {
self.$fname.listen(addr).await
}
}
#[async_trait]
impl <$tgens> NetStreamProvider<tor_general_addr::unix::SocketAddr> for $ttype {
type Stream = FakeStream;
type Listener = FakeListener<tor_general_addr::unix::SocketAddr>;
async fn connect(&self, _addr: &tor_general_addr::unix::SocketAddr) -> IoResult<Self::Stream> {
Err(tor_general_addr::unix::NoAfUnixSocketSupport::default().into())
}
async fn listen(&self, _addr: &tor_general_addr::unix::SocketAddr) -> IoResult<Self::Listener> {
Err(tor_general_addr::unix::NoAfUnixSocketSupport::default().into())
}
}
impl <$tgens> TlsProvider<<$ftype as NetStreamProvider>::Stream> for $ttype {
type Connector = <$ftype as TlsProvider<
<$ftype as NetStreamProvider>::Stream
>>::Connector;
type TlsStream = <$ftype as TlsProvider<
<$ftype as NetStreamProvider>::Stream
>>::TlsStream;
fn tls_connector(&self) -> Self::Connector {
self.$fname.tls_connector()
}
fn supports_keying_material_export(&self) -> bool {
self.$fname.supports_keying_material_export()
}
}
#[async_trait]
impl <$tgens> UdpProvider for $ttype {
type UdpSocket = <$ftype as UdpProvider>::UdpSocket;
#[inline]
async fn bind(&self, addr: &SocketAddr) -> IoResult<Self::UdpSocket> {
self.$fname.bind(addr).await
}
}
)
$(
${when fmeta(mock(sleep))}
impl <$tgens> SleepProvider for $ttype {
type SleepFuture = <$ftype as SleepProvider>::SleepFuture;
fn sleep(&self, dur: Duration) -> Self::SleepFuture {
self.$fname.sleep(dur)
}
fn now(&self) -> Instant {
self.$fname.now()
}
fn wallclock(&self) -> SystemTime {
self.$fname.wallclock()
}
fn block_advance<T: Into<String>>(&self, reason: T) {
self.$fname.block_advance(reason);
}
fn release_advance<T: Into<String>>(&self, reason: T) {
self.$fname.release_advance(reason);
}
fn allow_one_advance(&self, dur: Duration) {
self.$fname.allow_one_advance(dur);
}
}
impl <$tgens> CoarseTimeProvider for $ttype {
fn now_coarse(&self) -> CoarseInstant {
self.$fname.now_coarse()
}
}
)
#[allow(unused)]
const _: fn() = || {
fn x(_: impl Runtime) { }
fn check_impl_runtime<$tgens>(t: $ttype) { x(t) }
};
}
pub(crate) mod impl_runtime_prelude {
pub(crate) use async_trait::async_trait;
pub(crate) use derive_deftly::Deftly;
pub(crate) use futures::task::{FutureObj, Spawn, SpawnError};
pub(crate) use futures::Future;
pub(crate) use std::io::Result as IoResult;
pub(crate) use std::net::SocketAddr;
pub(crate) use std::time::{Duration, Instant, SystemTime};
pub(crate) use tor_rtcompat::{
unimpl::FakeListener, unimpl::FakeStream, Blocking, CoarseInstant, CoarseTimeProvider,
NetStreamProvider, Runtime, SleepProvider, TlsProvider, ToplevelBlockOn, UdpProvider,
};
}
#[allow(clippy::disallowed_methods)]
pub(crate) fn mpsc_channel<T>(buffer: usize) -> (mpsc::Sender<T>, mpsc::Receiver<T>) {
mpsc::channel(buffer)
}