Skip to content

Commit 52b2a92

Browse files
author
MrDenkoV
committed
Fix exec_replace
1 parent d5c0df8 commit 52b2a92

File tree

9 files changed

+109
-47
lines changed

9 files changed

+109
-47
lines changed

Cargo.lock

+2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,7 @@ typed-builder = "0.15.1"
9393
url = { version = "2.4.1", features = ["serde"] }
9494
walkdir = "2.4.0"
9595
which = "4.4.0"
96+
windows-sys = "0.48.0"
9697
xshell = "0.2.5"
9798
xxhash-rust = { version = "0.8.7", features = ["xxh3"] }
9899
zip = { version = "0.6.6", default-features = false, features = ["deflate"] }

scarb/Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,7 @@ typed-builder.workspace = true
6767
url.workspace = true
6868
walkdir.workspace = true
6969
which.workspace = true
70+
windows-sys.workspace = true
7071
xxhash-rust.workspace = true
7172
zip.workspace = true
7273

scarb/src/ops/subcommands.rs

-2
Original file line numberDiff line numberDiff line change
@@ -29,8 +29,6 @@ pub fn execute_external_subcommand(
2929
};
3030

3131
// TODO(mkaput): Jobserver.
32-
// TODO(#129): Write a test that CTRL+C kills everything, like Cargo's death,
33-
// but perhaps use an external bash script? Use Job Objects or smth else to fix it.
3432

3533
let mut cmd = Command::new(cmd);
3634
cmd.args(args);

scarb/src/process.rs

+52-16
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,6 @@ use scarb_ui::components::{Spinner, Status};
1212

1313
use crate::core::Config;
1414

15-
// TODO(#125): Do what is documented here, take a look at what cargo-util does.
1615
/// Replaces the current process with the target process.
1716
///
1817
/// On Unix, this executes the process using the Unix syscall `execvp`, which will block this
@@ -30,21 +29,58 @@ use crate::core::Config;
3029
/// itself later exits.
3130
#[tracing::instrument(level = "debug")]
3231
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+
}
4884
}
4985
}
5086

scarb/tests/subcommand.rs

+8-29
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
11
use std::io::Read;
22
use std::net::TcpListener;
3-
use std::process::{Child, Stdio};
3+
use std::process::{Child, Command};
44
use std::{env, io};
55

66
use assert_fs::TempDir;
7-
use indoc::{formatdoc, indoc};
7+
#[cfg(unix)]
8+
use indoc::indoc;
9+
use scarb_test_support::cargo::cargo_bin;
810

911
use scarb_test_support::command::Scarb;
1012
use scarb_test_support::filesystem::{path_with_temp_dir, write_script, write_simple_hello_script};
@@ -128,38 +130,15 @@ fn env_scarb_log_is_passed_verbatim() {
128130
.success();
129131
}
130132

131-
// TODO(#129): Fix this test.
132133
#[test]
133-
#[ignore] // something doesn't work here
134134
fn ctrl_c_kills_everyone() {
135-
let t = assert_fs::TempDir::new().unwrap();
136135
let listener = TcpListener::bind("127.0.0.1:0").unwrap();
136+
let addr = listener.local_addr().unwrap().to_string();
137137

138-
write_script(
139-
"hang-on-tcp",
140-
&{
141-
let addr = listener.local_addr().unwrap();
142-
let ip = addr.ip();
143-
let port = addr.port();
144-
formatdoc!(
145-
r#"
146-
#!/usr/bin/env python3
147-
import socket
148-
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
149-
sock.connect(("{ip}", {port}))
150-
sock.recv(10)
151-
raise Exception("recv should never return")
152-
"#
153-
)
154-
},
155-
&t,
156-
);
157-
158-
let mut child = Scarb::new()
159-
.std()
138+
let mut child = Command::new(cargo_bin("scarb-test-support"))
160139
.arg("hang-on-tcp")
161-
.env("PATH", path_with_temp_dir(&t))
162-
.stdin(Stdio::piped())
140+
.arg("--address")
141+
.arg(addr)
163142
.spawn()
164143
.unwrap();
165144

utils/scarb-test-support/Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ publish = false
88
anyhow.workspace = true
99
assert_fs.workspace = true
1010
camino.workspace = true
11+
clap.workspace = true
1112
dunce.workspace = true
1213
indoc.workspace = true
1314
itertools.workspace = true

utils/scarb-test-support/src/main.rs

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
use anyhow::Result;
2+
use clap::{Parser, Subcommand};
3+
4+
#[derive(Parser, Clone, Debug)]
5+
struct Args {
6+
/// Subcommand and its arguments.
7+
#[command(subcommand)]
8+
pub command: Command,
9+
}
10+
11+
#[derive(Subcommand, Clone, Debug)]
12+
pub enum Command {
13+
HangOnTcp(HangOnTcpArgs),
14+
}
15+
16+
#[derive(Parser, Clone, Debug)]
17+
pub struct HangOnTcpArgs {
18+
#[arg(short, long)]
19+
address: String,
20+
}
21+
22+
fn main() -> Result<()> {
23+
let args: Args = Args::parse();
24+
match args.command {
25+
Command::HangOnTcp(args) => hang_on_tcp(args),
26+
}
27+
}
28+
29+
fn hang_on_tcp(args: HangOnTcpArgs) -> Result<()> {
30+
use std::io::Read;
31+
use std::net::TcpStream;
32+
33+
let address: &str = args.address.as_ref();
34+
35+
let mut socket = TcpStream::connect(address).unwrap();
36+
let _ = socket.read(&mut [0; 10]);
37+
unreachable!("that read should never return");
38+
}
+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
// HACK: We need integration tests, for cargo test to generate the binary "scarb-test-support",
2+
// necessary for cargo_bin("scarb-test-support") to work correctly
3+
// (used in ctrl_c_kills_everyone test in scarb/tests/subcommand.rs).
4+
5+
#[test]
6+
fn binary_dependencies_hack() {}

0 commit comments

Comments
 (0)