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
|
use super::{Arguments, Error};
use std::collections::HashMap;
use futures::TryFutureExt;
use parity_tokio_ipc::Endpoint as IpcEndpoint;
use tokio::runtime::{self, Runtime};
use tonic::transport::{Endpoint, Uri};
use tower::service_fn;
#[allow(clippy::derive_partial_eq_without_eq)]
mod proto {
tonic::include_proto!("talpid_openvpn_plugin");
}
use proto::openvpn_event_proxy_client::OpenvpnEventProxyClient;
/// Struct processing OpenVPN events and notifies listeners over IPC
pub struct EventProcessor {
ipc_client: OpenvpnEventProxyClient<tonic::transport::Channel>,
runtime: Runtime,
}
impl EventProcessor {
pub fn new(arguments: Arguments) -> Result<EventProcessor, Error> {
log::trace!("Creating EventProcessor");
let runtime = runtime::Builder::new_current_thread()
.enable_all()
.build()
.map_err(Error::CreateRuntime)?;
let ipc_client = runtime
.block_on(Self::spawn_client(arguments.ipc_socket_path))
.map_err(Error::CreateTransport)?;
Ok(EventProcessor {
ipc_client,
runtime,
})
}
async fn spawn_client(
ipc_path: String,
) -> Result<OpenvpnEventProxyClient<tonic::transport::Channel>, tonic::transport::Error> {
// The URI will be ignored
let channel = Endpoint::from_static("lttp://[::]:50051")
.connect_with_connector(service_fn(move |_: Uri| {
IpcEndpoint::connect(ipc_path.clone()).map_ok(hyper_util::rt::tokio::TokioIo::new)
}))
.await?;
Ok(OpenvpnEventProxyClient::new(channel))
}
pub fn process_event(
&mut self,
event: openvpn_plugin::EventType,
env: HashMap<String, String>,
) -> Result<(), Error> {
log::debug!("Processing \"{:?}\" event", event);
let details = proto::EventDetails { env };
let response = match event {
openvpn_plugin::EventType::AuthFailed => {
self.runtime.block_on(self.ipc_client.auth_failed(details))
}
openvpn_plugin::EventType::Up => self.runtime.block_on(self.ipc_client.up(details)),
openvpn_plugin::EventType::RouteUp => {
self.runtime.block_on(self.ipc_client.route_up(details))
}
openvpn_plugin::EventType::RoutePredown => self
.runtime
.block_on(self.ipc_client.route_predown(details)),
other => return Err(Error::UnhandledEvent(other)),
};
match response {
Ok(_) => Ok(()),
Err(e) => Err(Error::SendEvent(Box::new(e))),
}
}
}
|