@@ -47,19 +47,22 @@ pub struct SplitTunnel {
47
47
state : State ,
48
48
tunnel_tx : Weak < futures:: channel:: mpsc:: UnboundedSender < TunnelCommand > > ,
49
49
rx : mpsc:: UnboundedReceiver < Message > ,
50
+ shutdown_tx : Option < oneshot:: Sender < ( ) > > ,
50
51
}
51
52
52
53
enum Message {
54
+ /// Return the name of the split tunnel interface
53
55
GetInterface {
54
56
result_tx : oneshot:: Sender < Option < String > > ,
55
57
} ,
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
59
61
SetExcludePaths {
60
62
result_tx : oneshot:: Sender < Result < ( ) , Error > > ,
61
63
paths : HashSet < PathBuf > ,
62
64
} ,
65
+ /// Update VPN tunnel interface
63
66
SetTunnel {
64
67
result_tx : oneshot:: Sender < Result < ( ) , Error > > ,
65
68
new_vpn_interface : Option < VpnInterface > ,
@@ -124,6 +127,7 @@ impl SplitTunnel {
124
127
} ,
125
128
tunnel_tx,
126
129
rx,
130
+ shutdown_tx : None ,
127
131
} ;
128
132
129
133
tokio:: spawn ( Self :: run ( split_tunnel) ) ;
@@ -143,68 +147,79 @@ impl SplitTunnel {
143
147
tokio:: select! {
144
148
// Handle process monitor being stopped
145
149
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) ;
162
151
}
163
152
164
153
// Handle messages
165
154
message = self . rx. recv( ) => {
166
155
let Some ( message) = message else {
167
- // Shut down split tunnel
168
156
break
169
157
} ;
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 ;
197
160
}
198
161
}
199
162
}
200
163
}
201
164
202
165
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
203
218
}
204
219
205
220
/// Shut down split tunnel
206
- async fn shutdown ( self ) {
207
- match self . state {
221
+ async fn shutdown ( & mut self ) {
222
+ match self . state . fail ( ) {
208
223
State :: ProcessMonitorOnly { mut process, .. } => {
209
224
process. shutdown ( ) . await ;
210
225
}
@@ -231,7 +246,6 @@ impl SplitTunnel {
231
246
}
232
247
}
233
248
234
- /// State machine
235
249
enum State {
236
250
/// The initial state: no paths have been provided
237
251
NoExclusions {
0 commit comments