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
use crate::{
Codec, FramedTransport, IntoSplit, MappedListener, Server, ServerExt, WindowsPipeListener,
WindowsPipeServerRef,
};
use async_trait::async_trait;
use serde::{de::DeserializeOwned, Serialize};
use std::{
ffi::{OsStr, OsString},
io,
};
#[async_trait]
pub trait WindowsPipeServerExt {
type Request;
type Response;
async fn start<A, C>(self, addr: A, codec: C) -> io::Result<WindowsPipeServerRef>
where
A: AsRef<OsStr> + Send,
C: Codec + Send + Sync + 'static;
async fn start_local<N, C>(self, name: N, codec: C) -> io::Result<WindowsPipeServerRef>
where
Self: Sized,
N: AsRef<OsStr> + Send,
C: Codec + Send + Sync + 'static,
{
let mut addr = OsString::from(r"\\.\pipe\");
addr.push(name.as_ref());
self.start(addr, codec).await
}
}
#[async_trait]
impl<S, Req, Res, Data> WindowsPipeServerExt for S
where
S: Server<Request = Req, Response = Res, LocalData = Data> + Sync + 'static,
Req: DeserializeOwned + Send + Sync + 'static,
Res: Serialize + Send + 'static,
Data: Default + Send + Sync + 'static,
{
type Request = Req;
type Response = Res;
async fn start<A, C>(self, addr: A, codec: C) -> io::Result<WindowsPipeServerRef>
where
A: AsRef<OsStr> + Send,
C: Codec + Send + Sync + 'static,
{
let a = addr.as_ref();
let listener = WindowsPipeListener::bind(a)?;
let addr = listener.addr().to_os_string();
let listener = MappedListener::new(listener, move |transport| {
let transport = FramedTransport::new(transport, codec.clone());
transport.into_split()
});
let inner = ServerExt::start(self, listener)?;
Ok(WindowsPipeServerRef { addr, inner })
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{Client, PlainCodec, Request, ServerCtx, WindowsPipeClientExt};
pub struct TestServer;
#[async_trait]
impl Server for TestServer {
type Request = String;
type Response = String;
type LocalData = ();
async fn on_request(&self, ctx: ServerCtx<Self::Request, Self::Response, Self::LocalData>) {
ctx.reply
.send(ctx.request.payload.to_string())
.await
.unwrap();
}
}
#[tokio::test]
async fn should_invoke_handler_upon_receiving_a_request() {
let server = WindowsPipeServerExt::start_local(
TestServer,
format!("test_pip_{}", rand::random::<usize>()),
PlainCodec,
)
.await
.expect("Failed to start Windows pipe server");
let mut client: Client<String, String> = Client::connect(server.addr(), PlainCodec)
.await
.expect("Client failed to connect");
let response = client
.send(Request::new("hello".to_string()))
.await
.expect("Failed to send message");
assert_eq!(response.payload, "hello");
}
}