feat: correct non blocking test for process.create

This commit is contained in:
Erica Marigold 2024-06-24 18:25:09 +05:30
parent aa10e88495
commit bd71b5e4b8
No known key found for this signature in database
GPG key ID: 2768CC0C23D245D1
3 changed files with 40 additions and 72 deletions

View file

@ -79,7 +79,7 @@ pub fn module(lua: &Lua) -> LuaResult<LuaTable> {
.with_value("env", env_tab)?
.with_value("exit", process_exit)?
.with_async_function("exec", process_exec)?
.with_async_function("create", process_spawn)?
.with_function("create", process_spawn)?
.build_readonly()
}
@ -153,7 +153,7 @@ async fn process_exec(
) -> LuaResult<LuaTable> {
let res = lua
.spawn(async move {
let cmd = spawn_command(program, args, options.clone()).await?;
let cmd = spawn_command_with_stdin(program, args, options.clone()).await?;
wait_for_child(cmd, options.stdio.stdout, options.stdio.stderr).await
})
.await?;
@ -180,7 +180,7 @@ async fn process_exec(
}
#[allow(clippy::await_holding_refcell_ref)]
async fn process_spawn(
fn process_spawn(
lua: &Lua,
(program, args, options): (String, Option<Vec<String>>, ProcessSpawnOptions),
) -> LuaResult<LuaTable> {
@ -189,26 +189,15 @@ async fn process_spawn(
let mut spawn_options = options.clone();
spawn_options.stdio = ProcessSpawnOptionsStdio::default();
let (stdin_tx, stdin_rx) = tokio::sync::oneshot::channel();
let (stdout_tx, stdout_rx) = tokio::sync::oneshot::channel();
let (stderr_tx, stderr_rx) = tokio::sync::oneshot::channel();
let (code_tx, code_rx) = tokio::sync::broadcast::channel(4);
let code_rx_rc = Rc::new(RefCell::new(code_rx));
tokio::spawn(async move {
let mut child = spawn_command(program, args, spawn_options)
.await
.expect("Could not spawn child process");
stdin_tx
.send(child.stdin.take())
.expect("Stdin receiver was unexpectedly dropped");
stdout_tx
.send(child.stdout.take())
.expect("Stdout receiver was unexpectedly dropped");
stderr_tx
.send(child.stderr.take())
.expect("Stderr receiver was unexpectedly dropped");
let mut child = spawn_command(program, args, spawn_options)?;
let stdin = child.stdin.take().unwrap();
let stdout = child.stdout.take().unwrap();
let stderr = child.stderr.take().unwrap();
tokio::spawn(async move {
let res = child
.wait_with_output()
.await
@ -225,33 +214,9 @@ async fn process_spawn(
});
TableBuilder::new(lua)?
.with_value(
"stdout",
ChildProcessReader(
stdout_rx
.await
.expect("Stdout sender unexpectedly dropped")
.unwrap(),
),
)?
.with_value(
"stderr",
ChildProcessReader(
stderr_rx
.await
.expect("Stderr sender unexpectedly dropped")
.unwrap(),
),
)?
.with_value(
"stdin",
ChildProcessWriter(
stdin_rx
.await
.expect("Stdin sender unexpectedly dropped")
.unwrap(),
),
)?
.with_value("stdout", ChildProcessReader(stdout))?
.with_value("stderr", ChildProcessReader(stderr))?
.with_value("stdin", ChildProcessWriter(stdin))?
.with_async_function("status", move |lua, ()| {
let code_rx_rc_clone = Rc::clone(&code_rx_rc);
async move {
@ -270,21 +235,14 @@ async fn process_spawn(
.build_readonly()
}
async fn spawn_command(
async fn spawn_command_with_stdin(
program: String,
args: Option<Vec<String>>,
mut options: ProcessSpawnOptions,
) -> LuaResult<Child> {
let stdout = options.stdio.stdout;
let stderr = options.stdio.stderr;
let stdin = options.stdio.stdin.take();
let mut child = options
.into_command(program, args)
.stdin(Stdio::piped())
.stdout(stdout.as_stdio())
.stderr(stderr.as_stdio())
.spawn()?;
let mut child = spawn_command(program, args, options)?;
if let Some(stdin) = stdin {
let mut child_stdin = child.stdin.take().unwrap();
@ -293,3 +251,21 @@ async fn spawn_command(
Ok(child)
}
fn spawn_command(
program: String,
args: Option<Vec<String>>,
options: ProcessSpawnOptions,
) -> LuaResult<Child> {
let stdout = options.stdio.stdout;
let stderr = options.stdio.stderr;
let child = options
.into_command(program, args)
.stdin(Stdio::piped())
.stdout(stdout.as_stdio())
.stderr(stderr.as_stdio())
.spawn()?;
Ok(child)
}

View file

@ -1,18 +1,13 @@
local process = require("@lune/process")
-- Spawning a child process should not block the main thread
-- Spawning a child process should not block the thread
local SAMPLES = 400
local childThread = coroutine.create(process.create)
for _ = 1, SAMPLES do
local start = os.time()
local child = process.create("echo", { "hello, world" })
local ok, err = coroutine.resume(childThread, "echo", { "hello, world" })
assert(ok, err)
assert(child ~= nil, "Failed to spawn child process")
local delta = os.time() - start
assert(
delta <= 1,
`Spawning a child process should not block the main thread, process.spawn took {delta}s to return when it should return immediately`
)
end
assert(
coroutine.status(childThread) == "dead",
"Child process should not block the thread it is running on"
)

View file

@ -15,7 +15,4 @@ local echoChild = if process.os == "windows"
then process.create("/c", { "echo", msg, "1>&2" }, { shell = "cmd" })
else process.create("echo", { msg, ">>/dev/stderr" }, { shell = true })
assert(
msg == echoChild.stderr:read(#msg),
"Failed to read from stderr of child process"
)
assert(msg == echoChild.stderr:read(#msg), "Failed to read from stderr of child process")