1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
use async_trait::async_trait;
#[must_use = "transmit do nothing unless polled"]
#[async_trait]
pub trait Transmit {
type Item;
type Error;
async fn transmit(&mut self, item: Self::Item) -> Result<(), Self::Error>
where
Self::Item: 'async_trait;
}
#[allow(dead_code)]
pub(crate) fn assert_transmit<I, E, T>(t: T) -> T
where
T: Transmit<Item = I, Error = E>,
{
t
}
#[cfg(feature = "with-async-channel")]
mod async_channel;
#[cfg(feature = "with-async-channel")]
pub use self::async_channel::*;
#[cfg(feature = "with-tokio")]
mod tokio;
#[cfg(feature = "with-tokio")]
pub use self::tokio::*;
#[cfg(feature = "with-sink")]
mod from_sink;
#[cfg(feature = "with-sink")]
pub use from_sink::FromSink;
mod with;
pub use with::With;
impl<T: ?Sized> TransmitExt for T where T: Transmit {}
#[cfg(feature = "with-sink")]
pub fn from_sink<S, I>(sink: S) -> FromSink<S, I, S::Error>
where
S: futures_sink::Sink<I> + Unpin + Send,
I: Send,
S::Error: Send,
{
assert_transmit::<I, S::Error, _>(from_sink::FromSink::from(sink))
}
pub trait TransmitExt: Transmit {
fn with<F, U>(self, f: F) -> with::With<Self, F, Self::Item, U, Self::Error>
where
Self: Sized + Send,
Self::Item: Send,
Self::Error: Send,
F: FnMut(U) -> Self::Item + Send,
U: Send,
{
assert_transmit::<U, Self::Error, _>(with::With::new(self, f))
}
}
#[cfg(test)]
#[cfg(feature = "with-sink")]
mod sink_tests {
use super::assert_transmit;
use super::*;
use anyhow::Result;
use futures::channel::mpsc;
use futures::prelude::*;
use futures_await_test::async_test;
#[async_test]
async fn transmit_ext_from_sink_is_transmit() -> Result<()> {
let (s, mut r) = mpsc::unbounded::<&'static str>();
let mut t = assert_transmit(from_sink(s));
assert_eq!((), t.transmit("Hello").await?);
assert_eq!((), t.transmit("World").await?);
drop(t);
assert_eq!(r.next().await, Some("Hello"));
assert_eq!(r.next().await, Some("World"));
assert_eq!(r.next().await, None);
Ok(())
}
#[async_test]
async fn transmit_ext_with_is_transmit() -> Result<()> {
let (s, mut r) = mpsc::unbounded::<String>();
let t = assert_transmit(from_sink(s));
let mut t = t.with(|s| format!("!!!{}!!!", s));
assert_eq!((), t.transmit("Hello").await?);
assert_eq!((), t.transmit("World").await?);
drop(t);
assert_eq!(r.next().await, Some("!!!Hello!!!".to_string()));
assert_eq!(r.next().await, Some("!!!World!!!".to_string()));
assert_eq!(r.next().await, None);
Ok(())
}
}