@@ -38,6 +38,11 @@ pub(crate) struct AppHandle {
38
38
pub ( crate ) server_stdout : Option < Lines < BufReader < ChildStdout > > > ,
39
39
pub ( crate ) server_stderr : Option < Lines < BufReader < ChildStderr > > > ,
40
40
41
+ /// The executables but with some extra entropy in their name so we can run two instances of the
42
+ /// same app without causing collisions on the filesystem.
43
+ pub ( crate ) entropy_app_exe : Option < PathBuf > ,
44
+ pub ( crate ) entropy_server_exe : Option < PathBuf > ,
45
+
41
46
/// The virtual directory that assets will be served from
42
47
/// Used mostly for apk/ipa builds since they live in simulator
43
48
pub ( crate ) runtime_asst_dir : Option < PathBuf > ,
@@ -54,6 +59,8 @@ impl AppHandle {
54
59
server_child : None ,
55
60
server_stdout : None ,
56
61
server_stderr : None ,
62
+ entropy_app_exe : None ,
63
+ entropy_server_exe : None ,
57
64
} )
58
65
}
59
66
@@ -79,6 +86,15 @@ impl AppHandle {
79
86
// unset the cargo dirs in the event we're running `dx` locally
80
87
// since the child process will inherit the env vars, we don't want to confuse the downstream process
81
88
( "CARGO_MANIFEST_DIR" , "" . to_string( ) ) ,
89
+ (
90
+ dioxus_cli_config:: SESSION_CACHE_DIR ,
91
+ self . app
92
+ . build
93
+ . krate
94
+ . session_cache_dir( )
95
+ . display( )
96
+ . to_string( ) ,
97
+ ) ,
82
98
] ;
83
99
84
100
if let Some ( base_path) = & self . app . build . krate . config . web . app . base_path {
@@ -87,7 +103,7 @@ impl AppHandle {
87
103
88
104
// Launch the server if we were given an address to start it on, and the build includes a server. After we
89
105
// start the server, consume its stdout/stderr.
90
- if let ( Some ( addr) , Some ( server) ) = ( start_fullstack_on_address, self . app . server_exe ( ) ) {
106
+ if let ( Some ( addr) , Some ( server) ) = ( start_fullstack_on_address, self . server_exe ( ) ) {
91
107
tracing:: debug!( "Proxying fullstack server from port {:?}" , addr) ;
92
108
envs. push ( ( dioxus_cli_config:: SERVER_IP_ENV , addr. ip ( ) . to_string ( ) ) ) ;
93
109
envs. push ( ( dioxus_cli_config:: SERVER_PORT_ENV , addr. port ( ) . to_string ( ) ) ) ;
@@ -147,6 +163,70 @@ impl AppHandle {
147
163
Ok ( ( ) )
148
164
}
149
165
166
+ /// Gracefully kill the process and all of its children
167
+ ///
168
+ /// Uses the `SIGTERM` signal on unix and `taskkill` on windows.
169
+ /// This complex logic is necessary for things like window state preservation to work properly.
170
+ ///
171
+ /// Also wipes away the entropy executables if they exist.
172
+ pub ( crate ) async fn cleanup ( & mut self ) {
173
+ tracing:: debug!( "Cleaning up process" ) ;
174
+
175
+ // Soft-kill the process by sending a sigkill, allowing the process to clean up
176
+ self . soft_kill ( ) . await ;
177
+
178
+ // Wipe out the entropy executables if they exist
179
+ if let Some ( entropy_app_exe) = self . entropy_app_exe . take ( ) {
180
+ _ = std:: fs:: remove_file ( entropy_app_exe) ;
181
+ }
182
+
183
+ if let Some ( entropy_server_exe) = self . entropy_server_exe . take ( ) {
184
+ _ = std:: fs:: remove_file ( entropy_server_exe) ;
185
+ }
186
+ }
187
+
188
+ /// Kill the app and server exes
189
+ pub ( crate ) async fn soft_kill ( & mut self ) {
190
+ use futures_util:: FutureExt ;
191
+
192
+ // Kill any running executables on Windows
193
+ let server_process = self . server_child . take ( ) ;
194
+ let client_process = self . app_child . take ( ) ;
195
+ let processes = [ server_process, client_process]
196
+ . into_iter ( )
197
+ . flatten ( )
198
+ . collect :: < Vec < _ > > ( ) ;
199
+
200
+ for mut process in processes {
201
+ let Some ( pid) = process. id ( ) else {
202
+ _ = process. kill ( ) . await ;
203
+ continue ;
204
+ } ;
205
+
206
+ // on unix, we can send a signal to the process to shut down
207
+ #[ cfg( unix) ]
208
+ {
209
+ _ = Command :: new ( "kill" )
210
+ . args ( [ "-s" , "TERM" , & pid. to_string ( ) ] )
211
+ . spawn ( ) ;
212
+ }
213
+
214
+ // on windows, use the `taskkill` command
215
+ #[ cfg( windows) ]
216
+ {
217
+ _ = Command :: new ( "taskkill" )
218
+ . args ( [ "/F" , "/PID" , & pid. to_string ( ) ] )
219
+ . spawn ( ) ;
220
+ }
221
+
222
+ // join the wait with a 100ms timeout
223
+ futures_util:: select! {
224
+ _ = process. wait( ) . fuse( ) => { }
225
+ _ = tokio:: time:: sleep( std:: time:: Duration :: from_millis( 1000 ) ) . fuse( ) => { }
226
+ } ;
227
+ }
228
+ }
229
+
150
230
/// Hotreload an asset in the running app.
151
231
///
152
232
/// This will modify the build dir in place! Be careful! We generally assume you want all bundles
@@ -236,12 +316,15 @@ impl AppHandle {
236
316
///
237
317
/// Server/liveview/desktop are all basically the same, though
238
318
fn open_with_main_exe ( & mut self , envs : Vec < ( & str , String ) > ) -> Result < Child > {
239
- let child = Command :: new ( self . app . main_exe ( ) )
319
+ // Create a new entropy app exe if we need to
320
+ let main_exe = self . app_exe ( ) ;
321
+ let child = Command :: new ( main_exe)
240
322
. envs ( envs)
241
323
. stderr ( Stdio :: piped ( ) )
242
324
. stdout ( Stdio :: piped ( ) )
243
325
. kill_on_drop ( true )
244
326
. spawn ( ) ?;
327
+
245
328
Ok ( child)
246
329
}
247
330
@@ -650,4 +733,72 @@ We checked the folder: {}
650
733
} ;
651
734
} ) ;
652
735
}
736
+
737
+ fn make_entropy_path ( exe : & PathBuf ) -> PathBuf {
738
+ let id = uuid:: Uuid :: new_v4 ( ) ;
739
+ let name = id. to_string ( ) ;
740
+ let some_entropy = name. split ( '-' ) . next ( ) . unwrap ( ) ;
741
+
742
+ // Make a copy of the server exe with a new name
743
+ let entropy_server_exe = exe. with_file_name ( format ! (
744
+ "{}-{}" ,
745
+ exe. file_name( ) . unwrap( ) . to_str( ) . unwrap( ) ,
746
+ some_entropy
747
+ ) ) ;
748
+
749
+ std:: fs:: copy ( exe, & entropy_server_exe) . unwrap ( ) ;
750
+
751
+ entropy_server_exe
752
+ }
753
+
754
+ fn server_exe ( & mut self ) -> Option < PathBuf > {
755
+ let mut server = self . app . server_exe ( ) ?;
756
+
757
+ // Create a new entropy server exe if we need to
758
+ if cfg ! ( target_os = "windows" ) || cfg ! ( target_os = "linux" ) {
759
+ // If we already have an entropy server exe, return it - this is useful for re-opening the same app
760
+ if let Some ( existing_server) = self . entropy_server_exe . clone ( ) {
761
+ return Some ( existing_server) ;
762
+ }
763
+
764
+ // Otherwise, create a new entropy server exe and save it for re-opning
765
+ let entropy_server_exe = Self :: make_entropy_path ( & server) ;
766
+ self . entropy_server_exe = Some ( entropy_server_exe. clone ( ) ) ;
767
+ server = entropy_server_exe;
768
+ }
769
+
770
+ Some ( server)
771
+ }
772
+
773
+ fn app_exe ( & mut self ) -> PathBuf {
774
+ let mut main_exe = self . app . main_exe ( ) ;
775
+
776
+ // The requirement here is based on the platform, not necessarily our current architecture.
777
+ let requires_entropy = match self . app . build . build . platform ( ) {
778
+ // When running "bundled", we don't need entropy
779
+ Platform :: Web => false ,
780
+ Platform :: MacOS => false ,
781
+ Platform :: Ios => false ,
782
+ Platform :: Android => false ,
783
+
784
+ // But on platforms that aren't running as "bundled", we do.
785
+ Platform :: Windows => true ,
786
+ Platform :: Linux => true ,
787
+ Platform :: Server => true ,
788
+ Platform :: Liveview => true ,
789
+ } ;
790
+
791
+ if requires_entropy || std:: env:: var ( "DIOXUS_ENTROPY" ) . is_ok ( ) {
792
+ // If we already have an entropy app exe, return it - this is useful for re-opening the same app
793
+ if let Some ( existing_app_exe) = self . entropy_app_exe . clone ( ) {
794
+ return existing_app_exe;
795
+ }
796
+
797
+ let entropy_app_exe = Self :: make_entropy_path ( & main_exe) ;
798
+ self . entropy_app_exe = Some ( entropy_app_exe. clone ( ) ) ;
799
+ main_exe = entropy_app_exe;
800
+ }
801
+
802
+ main_exe
803
+ }
653
804
}
0 commit comments