@@ -12,7 +12,6 @@ use scarb_ui::components::{Spinner, Status};
12
12
13
13
use crate :: core:: Config ;
14
14
15
- // TODO(#125): Do what is documented here, take a look at what cargo-util does.
16
15
/// Replaces the current process with the target process.
17
16
///
18
17
/// On Unix, this executes the process using the Unix syscall `execvp`, which will block this
@@ -30,21 +29,58 @@ use crate::core::Config;
30
29
/// itself later exits.
31
30
#[ tracing:: instrument( level = "debug" ) ]
32
31
pub fn exec_replace ( cmd : & mut Command ) -> Result < ( ) > {
33
- let exit_status = cmd
34
- . spawn ( )
35
- . with_context ( || format ! ( "failed to spawn: {}" , cmd. get_program( ) . to_string_lossy( ) ) ) ?
36
- . wait ( )
37
- . with_context ( || {
38
- format ! (
39
- "failed to wait for process to finish: {}" ,
40
- cmd. get_program( ) . to_string_lossy( )
41
- )
42
- } ) ?;
43
-
44
- if exit_status. success ( ) {
45
- Ok ( ( ) )
46
- } else {
47
- bail ! ( "process did not exit successfully: {exit_status}" ) ;
32
+ imp:: exec_replace ( cmd)
33
+ }
34
+
35
+ #[ cfg( unix) ]
36
+ mod imp {
37
+ use anyhow:: { bail, Result } ;
38
+ use std:: os:: unix:: process:: CommandExt ;
39
+ use std:: process:: Command ;
40
+
41
+ pub fn exec_replace ( cmd : & mut Command ) -> Result < ( ) > {
42
+ let err = cmd. exec ( ) ;
43
+ bail ! ( "process did not exit successfully: {err}" )
44
+ }
45
+ }
46
+
47
+ #[ cfg( windows) ]
48
+ mod imp {
49
+ use anyhow:: { bail, Context , Result } ;
50
+ use std:: process:: Command ;
51
+ use windows_sys:: Win32 :: Foundation :: { BOOL , FALSE , TRUE } ;
52
+ use windows_sys:: Win32 :: System :: Console :: SetConsoleCtrlHandler ;
53
+
54
+ unsafe extern "system" fn ctrlc_handler ( _: u32 ) -> BOOL {
55
+ // Do nothing; let the child process handle it.
56
+ TRUE
57
+ }
58
+
59
+ pub fn exec_replace ( cmd : & mut Command ) -> Result < ( ) > {
60
+ unsafe {
61
+ if SetConsoleCtrlHandler ( Some ( ctrlc_handler) , TRUE ) == FALSE {
62
+ panic ! ( "Could not set Ctrl-C handler." ) ;
63
+ }
64
+ }
65
+
66
+ // Just execute the process as normal.
67
+
68
+ let exit_status = cmd
69
+ . spawn ( )
70
+ . with_context ( || format ! ( "failed to spawn: {}" , cmd. get_program( ) . to_string_lossy( ) ) ) ?
71
+ . wait ( )
72
+ . with_context ( || {
73
+ format ! (
74
+ "failed to wait for process to finish: {}" ,
75
+ cmd. get_program( ) . to_string_lossy( )
76
+ )
77
+ } ) ?;
78
+
79
+ if exit_status. success ( ) {
80
+ Ok ( ( ) )
81
+ } else {
82
+ bail ! ( "process did not exit successfully: {exit_status}" ) ;
83
+ }
48
84
}
49
85
}
50
86
0 commit comments