Skip to content

Commit 82cd769

Browse files
committed
Refactor state handling
1 parent d2cc168 commit 82cd769

File tree

1 file changed

+64
-50
lines changed
  • talpid-core/src/split_tunnel/macos

1 file changed

+64
-50
lines changed

talpid-core/src/split_tunnel/macos/mod.rs

+64-50
Original file line numberDiff line numberDiff line change
@@ -47,19 +47,22 @@ pub struct SplitTunnel {
4747
state: State,
4848
tunnel_tx: Weak<futures::channel::mpsc::UnboundedSender<TunnelCommand>>,
4949
rx: mpsc::UnboundedReceiver<Message>,
50+
shutdown_tx: Option<oneshot::Sender<()>>,
5051
}
5152

5253
enum Message {
54+
/// Return the name of the split tunnel interface
5355
GetInterface {
5456
result_tx: oneshot::Sender<Option<String>>,
5557
},
56-
Shutdown {
57-
result_tx: oneshot::Sender<()>,
58-
},
58+
/// Shut down split tunnel service
59+
Shutdown { result_tx: oneshot::Sender<()> },
60+
/// Set paths to exclude from the VPN tunnel
5961
SetExcludePaths {
6062
result_tx: oneshot::Sender<Result<(), Error>>,
6163
paths: HashSet<PathBuf>,
6264
},
65+
/// Update VPN tunnel interface
6366
SetTunnel {
6467
result_tx: oneshot::Sender<Result<(), Error>>,
6568
new_vpn_interface: Option<VpnInterface>,
@@ -124,6 +127,7 @@ impl SplitTunnel {
124127
},
125128
tunnel_tx,
126129
rx,
130+
shutdown_tx: None,
127131
};
128132

129133
tokio::spawn(Self::run(split_tunnel));
@@ -143,68 +147,79 @@ impl SplitTunnel {
143147
tokio::select! {
144148
// Handle process monitor being stopped
145149
result = process_monitor_stopped => {
146-
match result {
147-
Ok(()) => log::error!("Process monitor stopped unexpectedly with no error"),
148-
Err(error) => {
149-
log::error!("{}", error.display_chain_with_msg("Process monitor stopped unexpectedly"));
150-
}
151-
}
152-
153-
// Enter the error state if split tunneling is active. Otherwise, we might make incorrect
154-
// decisions for new processes
155-
if self.state.active() {
156-
if let Some(tunnel_tx) = self.tunnel_tx.upgrade() {
157-
let _ = tunnel_tx.unbounded_send(TunnelCommand::Block(ErrorStateCause::SplitTunnelError));
158-
}
159-
}
160-
161-
self.state.fail();
150+
self.handle_process_monitor_shutdown(result);
162151
}
163152

164153
// Handle messages
165154
message = self.rx.recv() => {
166155
let Some(message) = message else {
167-
// Shut down split tunnel
168156
break
169157
};
170-
171-
match message {
172-
Message::GetInterface {
173-
result_tx,
174-
} => {
175-
let _ = result_tx.send(self.interface().map(str::to_owned));
176-
}
177-
Message::Shutdown {
178-
result_tx,
179-
} => {
180-
// Shut down; early exit
181-
self.shutdown().await;
182-
let _ = result_tx.send(());
183-
return;
184-
}
185-
Message::SetExcludePaths {
186-
result_tx,
187-
paths,
188-
} => {
189-
let _ = result_tx.send(self.state.set_exclude_paths(paths).await);
190-
}
191-
Message::SetTunnel {
192-
result_tx,
193-
new_vpn_interface,
194-
} => {
195-
let _ = result_tx.send(self.state.set_tunnel(new_vpn_interface).await);
196-
}
158+
if !self.handle_message(message).await {
159+
break;
197160
}
198161
}
199162
}
200163
}
201164

202165
self.shutdown().await;
166+
167+
if let Some(tx) = self.shutdown_tx.take() {
168+
let _ = tx.send(());
169+
}
170+
}
171+
172+
/// Handle process monitor unexpectedly stopping
173+
fn handle_process_monitor_shutdown(&mut self, result: Result<(), process::Error>) {
174+
match result {
175+
Ok(()) => log::error!("Process monitor stopped unexpectedly with no error"),
176+
Err(error) => {
177+
log::error!(
178+
"{}",
179+
error.display_chain_with_msg("Process monitor stopped unexpectedly")
180+
);
181+
}
182+
}
183+
184+
// Enter the error state if split tunneling is active. Otherwise, we might make incorrect
185+
// decisions for new processes
186+
if self.state.active() {
187+
if let Some(tunnel_tx) = self.tunnel_tx.upgrade() {
188+
let _ = tunnel_tx
189+
.unbounded_send(TunnelCommand::Block(ErrorStateCause::SplitTunnelError));
190+
}
191+
}
192+
193+
self.state.fail();
194+
}
195+
196+
/// Handle an incoming message
197+
/// Return whether the actor should continue running
198+
async fn handle_message(&mut self, message: Message) -> bool {
199+
match message {
200+
Message::GetInterface { result_tx } => {
201+
let _ = result_tx.send(self.interface().map(str::to_owned));
202+
}
203+
Message::Shutdown { result_tx } => {
204+
self.shutdown_tx = Some(result_tx);
205+
return false;
206+
}
207+
Message::SetExcludePaths { result_tx, paths } => {
208+
let _ = result_tx.send(self.state.set_exclude_paths(paths).await);
209+
}
210+
Message::SetTunnel {
211+
result_tx,
212+
new_vpn_interface,
213+
} => {
214+
let _ = result_tx.send(self.state.set_tunnel(new_vpn_interface).await);
215+
}
216+
}
217+
true
203218
}
204219

205220
/// Shut down split tunnel
206-
async fn shutdown(self) {
207-
match self.state {
221+
async fn shutdown(&mut self) {
222+
match self.state.fail() {
208223
State::ProcessMonitorOnly { mut process, .. } => {
209224
process.shutdown().await;
210225
}
@@ -231,7 +246,6 @@ impl SplitTunnel {
231246
}
232247
}
233248

234-
/// State machine
235249
enum State {
236250
/// The initial state: no paths have been provided
237251
NoExclusions {

0 commit comments

Comments
 (0)