2024-01-01 14:58:21 -05:00
|
|
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
2023-02-16 08:30:14 -05:00
|
|
|
|
|
|
|
import CP from "node:child_process";
|
|
|
|
import { Buffer } from "node:buffer";
|
|
|
|
import {
|
|
|
|
assert,
|
|
|
|
assertEquals,
|
|
|
|
assertExists,
|
|
|
|
assertNotStrictEquals,
|
|
|
|
assertStrictEquals,
|
|
|
|
assertStringIncludes,
|
2023-12-01 21:20:06 -05:00
|
|
|
} from "../../../test_util/std/assert/mod.ts";
|
2023-02-16 08:30:14 -05:00
|
|
|
import * as path from "../../../test_util/std/path/mod.ts";
|
|
|
|
|
2023-09-05 06:42:35 -04:00
|
|
|
const { spawn, spawnSync, execFile, execFileSync, ChildProcess } = CP;
|
2023-02-16 08:30:14 -05:00
|
|
|
|
2023-11-22 06:11:20 -05:00
|
|
|
function withTimeout<T>(
|
|
|
|
timeoutInMS = 10_000,
|
|
|
|
): ReturnType<typeof Promise.withResolvers<T>> {
|
|
|
|
const deferred = Promise.withResolvers<T>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const timer = setTimeout(() => {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.reject("Timeout");
|
2023-02-16 08:30:14 -05:00
|
|
|
}, timeoutInMS);
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.promise.then(() => {
|
2023-02-16 08:30:14 -05:00
|
|
|
clearTimeout(timer);
|
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
return deferred;
|
2023-02-16 08:30:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(uki00a): Once Node.js's `parallel/test-child-process-spawn-error.js` works, this test case should be removed.
|
|
|
|
Deno.test("[node/child_process spawn] The 'error' event is emitted when no binary is found", async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const childProcess = spawn("no-such-cmd");
|
|
|
|
childProcess.on("error", (_err: Error) => {
|
|
|
|
// TODO(@bartlomieju) Assert an error message.
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test("[node/child_process spawn] The 'exit' event is emitted with an exit code after the child process ends", async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const childProcess = spawn(Deno.execPath(), ["--help"], {
|
|
|
|
env: { NO_COLOR: "true" },
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
let exitCode = null;
|
|
|
|
childProcess.on("exit", (code: number) => {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
exitCode = code;
|
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
assertStrictEquals(exitCode, 0);
|
|
|
|
assertStrictEquals(childProcess.exitCode, exitCode);
|
|
|
|
} finally {
|
|
|
|
childProcess.kill();
|
|
|
|
childProcess.stdout?.destroy();
|
|
|
|
childProcess.stderr?.destroy();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test("[node/child_process disconnect] the method exists", async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const childProcess = spawn(Deno.execPath(), ["--help"], {
|
|
|
|
env: { NO_COLOR: "true" },
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
childProcess.disconnect();
|
|
|
|
childProcess.on("exit", () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
} finally {
|
|
|
|
childProcess.kill();
|
|
|
|
childProcess.stdout?.destroy();
|
|
|
|
childProcess.stderr?.destroy();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
name: "[node/child_process spawn] Verify that stdin and stdout work",
|
|
|
|
fn: async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const childProcess = spawn(Deno.execPath(), ["fmt", "-"], {
|
|
|
|
env: { NO_COLOR: "true" },
|
|
|
|
stdio: ["pipe", "pipe"],
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
assert(childProcess.stdin, "stdin should be defined");
|
|
|
|
assert(childProcess.stdout, "stdout should be defined");
|
|
|
|
let data = "";
|
|
|
|
childProcess.stdout.on("data", (chunk) => {
|
|
|
|
data += chunk;
|
|
|
|
});
|
|
|
|
childProcess.stdin.write(" console.log('hello')", "utf-8");
|
|
|
|
childProcess.stdin.end();
|
|
|
|
childProcess.on("close", () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
assertStrictEquals(data, `console.log("hello");\n`);
|
|
|
|
} finally {
|
|
|
|
childProcess.kill();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
name: "[node/child_process spawn] stdin and stdout with binary data",
|
|
|
|
fn: async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const p = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"./testdata/binary_stdio.js",
|
|
|
|
);
|
|
|
|
const childProcess = spawn(Deno.execPath(), ["run", p], {
|
|
|
|
env: { NO_COLOR: "true" },
|
|
|
|
stdio: ["pipe", "pipe"],
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
assert(childProcess.stdin, "stdin should be defined");
|
|
|
|
assert(childProcess.stdout, "stdout should be defined");
|
|
|
|
let data: Buffer;
|
|
|
|
childProcess.stdout.on("data", (chunk) => {
|
|
|
|
data = chunk;
|
|
|
|
});
|
|
|
|
const buffer = new Uint8Array([0, 1, 2, 3, 4, 5, 6, 7, 8, 9]);
|
|
|
|
childProcess.stdin.write(buffer);
|
|
|
|
childProcess.stdin.end();
|
|
|
|
childProcess.on("close", () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
assertEquals(new Uint8Array(data!), buffer);
|
|
|
|
} finally {
|
|
|
|
childProcess.kill();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
async function spawnAndGetEnvValue(
|
|
|
|
inputValue: string | number | boolean,
|
|
|
|
): Promise<string> {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<string>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const env = spawn(
|
|
|
|
`"${Deno.execPath()}" eval -p "Deno.env.toObject().BAZ"`,
|
|
|
|
{
|
|
|
|
env: { BAZ: String(inputValue), NO_COLOR: "true" },
|
|
|
|
shell: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
try {
|
|
|
|
let envOutput = "";
|
|
|
|
|
|
|
|
assert(env.stdout);
|
2023-11-22 06:11:20 -05:00
|
|
|
env.on("error", (err: Error) => deferred.reject(err));
|
2023-02-16 08:30:14 -05:00
|
|
|
env.stdout.on("data", (data) => {
|
|
|
|
envOutput += data;
|
|
|
|
});
|
|
|
|
env.on("close", () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve(envOutput.trim());
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
return await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
} finally {
|
|
|
|
env.kill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
ignore: Deno.build.os === "windows",
|
|
|
|
name:
|
|
|
|
"[node/child_process spawn] Verify that environment values can be numbers",
|
|
|
|
async fn() {
|
|
|
|
const envOutputValue = await spawnAndGetEnvValue(42);
|
|
|
|
assertStrictEquals(envOutputValue, "42");
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
ignore: Deno.build.os === "windows",
|
|
|
|
name:
|
|
|
|
"[node/child_process spawn] Verify that environment values can be booleans",
|
|
|
|
async fn() {
|
|
|
|
const envOutputValue = await spawnAndGetEnvValue(false);
|
|
|
|
assertStrictEquals(envOutputValue, "false");
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2023-04-11 16:00:05 -04:00
|
|
|
/* Start of ported part */
|
2023-02-16 08:30:14 -05:00
|
|
|
// Copyright Joyent and Node contributors. All rights reserved. MIT license.
|
|
|
|
// Ported from Node 15.5.1
|
|
|
|
|
|
|
|
// TODO(uki00a): Remove this case once Node's `parallel/test-child-process-spawn-event.js` works.
|
|
|
|
Deno.test("[child_process spawn] 'spawn' event", async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const timeout = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const subprocess = spawn(Deno.execPath(), ["eval", "console.log('ok')"]);
|
|
|
|
|
|
|
|
let didSpawn = false;
|
|
|
|
subprocess.on("spawn", function () {
|
|
|
|
didSpawn = true;
|
|
|
|
});
|
|
|
|
|
|
|
|
function mustNotBeCalled() {
|
|
|
|
timeout.reject(new Error("function should not have been called"));
|
|
|
|
}
|
|
|
|
|
|
|
|
const promises = [] as Promise<void>[];
|
|
|
|
function mustBeCalledAfterSpawn() {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = Promise.withResolvers<void>();
|
|
|
|
promises.push(deferred.promise);
|
2023-02-16 08:30:14 -05:00
|
|
|
return () => {
|
|
|
|
if (didSpawn) {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
} else {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.reject(
|
2023-02-16 08:30:14 -05:00
|
|
|
new Error("function should be called after the 'spawn' event"),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
subprocess.on("error", mustNotBeCalled);
|
|
|
|
subprocess.stdout!.on("data", mustBeCalledAfterSpawn());
|
|
|
|
subprocess.stdout!.on("end", mustBeCalledAfterSpawn());
|
|
|
|
subprocess.stdout!.on("close", mustBeCalledAfterSpawn());
|
|
|
|
subprocess.stderr!.on("data", mustNotBeCalled);
|
|
|
|
subprocess.stderr!.on("end", mustBeCalledAfterSpawn());
|
|
|
|
subprocess.stderr!.on("close", mustBeCalledAfterSpawn());
|
|
|
|
subprocess.on("exit", mustBeCalledAfterSpawn());
|
|
|
|
subprocess.on("close", mustBeCalledAfterSpawn());
|
|
|
|
|
|
|
|
try {
|
2023-11-22 06:11:20 -05:00
|
|
|
await Promise.race([Promise.all(promises), timeout.promise]);
|
2023-02-16 08:30:14 -05:00
|
|
|
timeout.resolve();
|
|
|
|
} finally {
|
|
|
|
subprocess.kill();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// TODO(uki00a): Remove this case once Node's `parallel/test-child-process-spawn-shell.js` works.
|
|
|
|
Deno.test("[child_process spawn] Verify that a shell is executed", async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const doesNotExist = spawn("does-not-exist", { shell: true });
|
|
|
|
try {
|
|
|
|
assertNotStrictEquals(doesNotExist.spawnfile, "does-not-exist");
|
|
|
|
doesNotExist.on("error", () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.reject("The 'error' event must not be emitted.");
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
|
|
|
doesNotExist.on("exit", (code: number, signal: null) => {
|
|
|
|
assertStrictEquals(signal, null);
|
|
|
|
|
|
|
|
if (Deno.build.os === "windows") {
|
|
|
|
assertStrictEquals(code, 1); // Exit code of cmd.exe
|
|
|
|
} else {
|
|
|
|
assertStrictEquals(code, 127); // Exit code of /bin/sh });
|
|
|
|
}
|
|
|
|
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
} finally {
|
|
|
|
doesNotExist.kill();
|
|
|
|
doesNotExist.stdout?.destroy();
|
|
|
|
doesNotExist.stderr?.destroy();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// TODO(uki00a): Remove this case once Node's `parallel/test-child-process-spawn-shell.js` works.
|
|
|
|
Deno.test({
|
|
|
|
ignore: Deno.build.os === "windows",
|
|
|
|
name: "[node/child_process spawn] Verify that passing arguments works",
|
|
|
|
async fn() {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const echo = spawn("echo", ["foo"], {
|
|
|
|
shell: true,
|
|
|
|
});
|
|
|
|
let echoOutput = "";
|
|
|
|
|
|
|
|
try {
|
|
|
|
assertStrictEquals(
|
|
|
|
echo.spawnargs[echo.spawnargs.length - 1].replace(/"/g, ""),
|
|
|
|
"echo foo",
|
|
|
|
);
|
|
|
|
assert(echo.stdout);
|
|
|
|
echo.stdout.on("data", (data) => {
|
|
|
|
echoOutput += data;
|
|
|
|
});
|
|
|
|
echo.on("close", () => {
|
|
|
|
assertStrictEquals(echoOutput.trim(), "foo");
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
} finally {
|
|
|
|
echo.kill();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
// TODO(uki00a): Remove this case once Node's `parallel/test-child-process-spawn-shell.js` works.
|
|
|
|
Deno.test({
|
|
|
|
ignore: Deno.build.os === "windows",
|
|
|
|
name: "[node/child_process spawn] Verity that shell features can be used",
|
|
|
|
async fn() {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const cmd = "echo bar | cat";
|
|
|
|
const command = spawn(cmd, {
|
|
|
|
shell: true,
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
let commandOutput = "";
|
|
|
|
|
|
|
|
assert(command.stdout);
|
|
|
|
command.stdout.on("data", (data) => {
|
|
|
|
commandOutput += data;
|
|
|
|
});
|
|
|
|
|
|
|
|
command.on("close", () => {
|
|
|
|
assertStrictEquals(commandOutput.trim(), "bar");
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
|
|
|
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
} finally {
|
|
|
|
command.kill();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
// TODO(uki00a): Remove this case once Node's `parallel/test-child-process-spawn-shell.js` works.
|
|
|
|
Deno.test({
|
|
|
|
ignore: Deno.build.os === "windows",
|
|
|
|
name:
|
|
|
|
"[node/child_process spawn] Verity that environment is properly inherited",
|
|
|
|
async fn() {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const env = spawn(
|
|
|
|
`"${Deno.execPath()}" eval -p "Deno.env.toObject().BAZ"`,
|
|
|
|
{
|
|
|
|
env: { BAZ: "buzz", NO_COLOR: "true" },
|
|
|
|
shell: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
try {
|
|
|
|
let envOutput = "";
|
|
|
|
|
|
|
|
assert(env.stdout);
|
2023-11-22 06:11:20 -05:00
|
|
|
env.on("error", (err: Error) => deferred.reject(err));
|
2023-02-16 08:30:14 -05:00
|
|
|
env.stdout.on("data", (data) => {
|
|
|
|
envOutput += data;
|
|
|
|
});
|
|
|
|
env.on("close", () => {
|
|
|
|
assertStrictEquals(envOutput.trim(), "buzz");
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve();
|
2023-02-16 08:30:14 -05:00
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
} finally {
|
|
|
|
env.kill();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
/* End of ported part */
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
name: "[node/child_process execFile] Get stdout as a string",
|
|
|
|
async fn() {
|
|
|
|
let child: unknown;
|
|
|
|
const script = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"./testdata/exec_file_text_output.js",
|
|
|
|
);
|
|
|
|
const promise = new Promise<string | null>((resolve, reject) => {
|
|
|
|
child = execFile(Deno.execPath(), ["run", script], (err, stdout) => {
|
|
|
|
if (err) reject(err);
|
|
|
|
else if (stdout) resolve(stdout as string);
|
|
|
|
else resolve(null);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
const stdout = await promise;
|
|
|
|
assertEquals(stdout, "Hello World!\n");
|
|
|
|
} finally {
|
|
|
|
if (child instanceof ChildProcess) {
|
|
|
|
child.kill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
name: "[node/child_process execFile] Get stdout as a buffer",
|
|
|
|
async fn() {
|
|
|
|
let child: unknown;
|
|
|
|
const script = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"./testdata/exec_file_text_output.js",
|
|
|
|
);
|
|
|
|
const promise = new Promise<Buffer | null>((resolve, reject) => {
|
|
|
|
child = execFile(
|
|
|
|
Deno.execPath(),
|
|
|
|
["run", script],
|
|
|
|
{ encoding: "buffer" },
|
|
|
|
(err, stdout) => {
|
|
|
|
if (err) reject(err);
|
|
|
|
else if (stdout) resolve(stdout as Buffer);
|
|
|
|
else resolve(null);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
const stdout = await promise;
|
|
|
|
assert(Buffer.isBuffer(stdout));
|
|
|
|
assertEquals(stdout.toString("utf8"), "Hello World!\n");
|
|
|
|
} finally {
|
|
|
|
if (child instanceof ChildProcess) {
|
|
|
|
child.kill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
name: "[node/child_process execFile] Get stderr",
|
|
|
|
async fn() {
|
|
|
|
let child: unknown;
|
|
|
|
const script = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"./testdata/exec_file_text_error.js",
|
|
|
|
);
|
|
|
|
const promise = new Promise<
|
|
|
|
{ err: Error | null; stderr?: string | Buffer }
|
|
|
|
>((resolve) => {
|
|
|
|
child = execFile(Deno.execPath(), ["run", script], (err, _, stderr) => {
|
|
|
|
resolve({ err, stderr });
|
|
|
|
});
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
const { err, stderr } = await promise;
|
|
|
|
if (child instanceof ChildProcess) {
|
|
|
|
assertEquals(child.exitCode, 1);
|
|
|
|
assertEquals(stderr, "yikes!\n");
|
|
|
|
} else {
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
if (child instanceof ChildProcess) {
|
|
|
|
child.kill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
name: "[node/child_process execFile] Exceed given maxBuffer limit",
|
|
|
|
async fn() {
|
|
|
|
let child: unknown;
|
|
|
|
const script = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"./testdata/exec_file_text_error.js",
|
|
|
|
);
|
|
|
|
const promise = new Promise<
|
|
|
|
{ err: Error | null; stderr?: string | Buffer }
|
|
|
|
>((resolve) => {
|
|
|
|
child = execFile(Deno.execPath(), ["run", script], {
|
|
|
|
encoding: "buffer",
|
|
|
|
maxBuffer: 3,
|
|
|
|
}, (err, _, stderr) => {
|
|
|
|
resolve({ err, stderr });
|
|
|
|
});
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
const { err, stderr } = await promise;
|
|
|
|
if (child instanceof ChildProcess) {
|
|
|
|
assert(err);
|
|
|
|
assertEquals(
|
|
|
|
// deno-lint-ignore no-explicit-any
|
|
|
|
(err as any).code,
|
|
|
|
"ERR_CHILD_PROCESS_STDIO_MAXBUFFER",
|
|
|
|
);
|
|
|
|
assertEquals(err.message, "stderr maxBuffer length exceeded");
|
|
|
|
assertEquals((stderr as Buffer).toString("utf8"), "yik");
|
|
|
|
} else {
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
if (child instanceof ChildProcess) {
|
|
|
|
child.kill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
|
|
|
name: "[node/child_process] ChildProcess.kill()",
|
|
|
|
async fn() {
|
|
|
|
const script = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"./testdata/infinite_loop.js",
|
|
|
|
);
|
|
|
|
const childProcess = spawn(Deno.execPath(), ["run", script]);
|
2023-11-22 06:11:20 -05:00
|
|
|
const p = withTimeout<void>();
|
|
|
|
const pStdout = withTimeout<void>();
|
|
|
|
const pStderr = withTimeout<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
childProcess.on("exit", () => p.resolve());
|
2023-09-16 01:48:31 -04:00
|
|
|
childProcess.stdout.on("close", () => pStdout.resolve());
|
|
|
|
childProcess.stderr.on("close", () => pStderr.resolve());
|
2023-02-16 08:30:14 -05:00
|
|
|
childProcess.kill("SIGKILL");
|
2023-11-22 06:11:20 -05:00
|
|
|
await p.promise;
|
|
|
|
await pStdout.promise;
|
|
|
|
await pStderr.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
assert(childProcess.killed);
|
|
|
|
assertEquals(childProcess.signalCode, "SIGKILL");
|
|
|
|
assertExists(childProcess.exitCode);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
2023-03-22 14:39:13 -04:00
|
|
|
ignore: true,
|
2023-02-16 08:30:14 -05:00
|
|
|
name: "[node/child_process] ChildProcess.unref()",
|
|
|
|
async fn() {
|
|
|
|
const script = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"testdata",
|
|
|
|
"child_process_unref.js",
|
|
|
|
);
|
|
|
|
const childProcess = spawn(Deno.execPath(), [
|
|
|
|
"run",
|
|
|
|
"-A",
|
|
|
|
"--unstable",
|
|
|
|
script,
|
|
|
|
]);
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = Promise.withResolvers<void>();
|
|
|
|
childProcess.on("exit", () => deferred.resolve());
|
|
|
|
await deferred.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test({
|
2023-03-22 14:39:13 -04:00
|
|
|
ignore: true,
|
2023-02-16 08:30:14 -05:00
|
|
|
name: "[node/child_process] child_process.fork",
|
|
|
|
async fn() {
|
|
|
|
const testdataDir = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"testdata",
|
|
|
|
);
|
|
|
|
const script = path.join(
|
|
|
|
testdataDir,
|
|
|
|
"node_modules",
|
|
|
|
"foo",
|
|
|
|
"index.js",
|
|
|
|
);
|
2023-11-22 06:11:20 -05:00
|
|
|
const p = Promise.withResolvers<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
const cp = CP.fork(script, [], { cwd: testdataDir, stdio: "pipe" });
|
|
|
|
let output = "";
|
|
|
|
cp.on("close", () => p.resolve());
|
|
|
|
cp.stdout?.on("data", (data) => {
|
|
|
|
output += data;
|
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await p.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
assertEquals(output, "foo\ntrue\ntrue\ntrue\n");
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test("[node/child_process execFileSync] 'inherit' stdout and stderr", () => {
|
|
|
|
execFileSync(Deno.execPath(), ["--help"], { stdio: "inherit" });
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/child_process spawn] supports windowsVerbatimArguments option",
|
|
|
|
{ ignore: Deno.build.os !== "windows" },
|
|
|
|
async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const cmdFinished = Promise.withResolvers<void>();
|
2023-02-16 08:30:14 -05:00
|
|
|
let output = "";
|
|
|
|
const cp = spawn("cmd", ["/d", "/s", "/c", '"deno ^"--version^""'], {
|
|
|
|
stdio: "pipe",
|
|
|
|
windowsVerbatimArguments: true,
|
|
|
|
});
|
|
|
|
cp.on("close", () => cmdFinished.resolve());
|
|
|
|
cp.stdout?.on("data", (data) => {
|
|
|
|
output += data;
|
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await cmdFinished.promise;
|
2023-02-16 08:30:14 -05:00
|
|
|
assertStringIncludes(output, "deno");
|
|
|
|
assertStringIncludes(output, "v8");
|
|
|
|
assertStringIncludes(output, "typescript");
|
|
|
|
},
|
|
|
|
);
|
2023-05-18 08:02:14 -04:00
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/child_process spawn] supports stdio array option",
|
|
|
|
async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const cmdFinished = Promise.withResolvers<void>();
|
2023-05-18 08:02:14 -04:00
|
|
|
let output = "";
|
|
|
|
const script = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"testdata",
|
|
|
|
"child_process_stdio.js",
|
|
|
|
);
|
|
|
|
const cp = spawn(Deno.execPath(), ["run", "-A", script]);
|
|
|
|
cp.stdout?.on("data", (data) => {
|
|
|
|
output += data;
|
|
|
|
});
|
|
|
|
cp.on("close", () => cmdFinished.resolve());
|
2023-11-22 06:11:20 -05:00
|
|
|
await cmdFinished.promise;
|
2023-05-18 08:02:14 -04:00
|
|
|
|
|
|
|
assertStringIncludes(output, "foo");
|
|
|
|
assertStringIncludes(output, "close");
|
|
|
|
},
|
|
|
|
);
|
2023-05-31 18:39:01 -04:00
|
|
|
|
fix(node): map stdio [0, 1, 2] to "inherit" (#19352)
<!--
Before submitting a PR, please read https://deno.com/manual/contributing
1. Give the PR a descriptive title.
Examples of good title:
- fix(std/http): Fix race condition in server
- docs(console): Update docstrings
- feat(doc): Handle nested reexports
Examples of bad title:
- fix #7123
- update docs
- fix bugs
2. Ensure there is a related issue and it is referenced in the PR text.
3. Ensure there are tests that cover the changes.
4. Ensure `cargo test` passes.
5. Ensure `./tools/format.js` passes without changing files.
6. Ensure `./tools/lint.js` passes.
7. Open as a draft PR if your work is still in progress. The CI won't
run
all steps, but you can add '[ci]' to a commit message to force it to.
8. If you would like to run the benchmarks on the CI, add the 'ci-bench'
label.
-->
Internally, `node-tap` spawns a child process with `stdio: [0, 1, 2]`.
Whilst we don't support passing fd numbers as an argument so far, it
turns out that `[0, 1, 2]` is equivalent to `"inherit"` which we already
support. See: https://nodejs.org/api/child_process.html#optionsstdio
Mapping it to `"inherit"` is fine for us and gets us one step closer in
getting `node-tap` working. I'm now at the stage where already the
coverage table is shown 🎉
2023-06-02 11:46:50 -04:00
|
|
|
Deno.test(
|
|
|
|
"[node/child_process spawn] supports stdio [0, 1, 2] option",
|
|
|
|
async () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
const cmdFinished = Promise.withResolvers<void>();
|
fix(node): map stdio [0, 1, 2] to "inherit" (#19352)
<!--
Before submitting a PR, please read https://deno.com/manual/contributing
1. Give the PR a descriptive title.
Examples of good title:
- fix(std/http): Fix race condition in server
- docs(console): Update docstrings
- feat(doc): Handle nested reexports
Examples of bad title:
- fix #7123
- update docs
- fix bugs
2. Ensure there is a related issue and it is referenced in the PR text.
3. Ensure there are tests that cover the changes.
4. Ensure `cargo test` passes.
5. Ensure `./tools/format.js` passes without changing files.
6. Ensure `./tools/lint.js` passes.
7. Open as a draft PR if your work is still in progress. The CI won't
run
all steps, but you can add '[ci]' to a commit message to force it to.
8. If you would like to run the benchmarks on the CI, add the 'ci-bench'
label.
-->
Internally, `node-tap` spawns a child process with `stdio: [0, 1, 2]`.
Whilst we don't support passing fd numbers as an argument so far, it
turns out that `[0, 1, 2]` is equivalent to `"inherit"` which we already
support. See: https://nodejs.org/api/child_process.html#optionsstdio
Mapping it to `"inherit"` is fine for us and gets us one step closer in
getting `node-tap` working. I'm now at the stage where already the
coverage table is shown 🎉
2023-06-02 11:46:50 -04:00
|
|
|
let output = "";
|
|
|
|
const script = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"testdata",
|
|
|
|
"child_process_stdio_012.js",
|
|
|
|
);
|
|
|
|
const cp = spawn(Deno.execPath(), ["run", "-A", script]);
|
|
|
|
cp.stdout?.on("data", (data) => {
|
|
|
|
output += data;
|
|
|
|
});
|
|
|
|
cp.on("close", () => cmdFinished.resolve());
|
2023-11-22 06:11:20 -05:00
|
|
|
await cmdFinished.promise;
|
fix(node): map stdio [0, 1, 2] to "inherit" (#19352)
<!--
Before submitting a PR, please read https://deno.com/manual/contributing
1. Give the PR a descriptive title.
Examples of good title:
- fix(std/http): Fix race condition in server
- docs(console): Update docstrings
- feat(doc): Handle nested reexports
Examples of bad title:
- fix #7123
- update docs
- fix bugs
2. Ensure there is a related issue and it is referenced in the PR text.
3. Ensure there are tests that cover the changes.
4. Ensure `cargo test` passes.
5. Ensure `./tools/format.js` passes without changing files.
6. Ensure `./tools/lint.js` passes.
7. Open as a draft PR if your work is still in progress. The CI won't
run
all steps, but you can add '[ci]' to a commit message to force it to.
8. If you would like to run the benchmarks on the CI, add the 'ci-bench'
label.
-->
Internally, `node-tap` spawns a child process with `stdio: [0, 1, 2]`.
Whilst we don't support passing fd numbers as an argument so far, it
turns out that `[0, 1, 2]` is equivalent to `"inherit"` which we already
support. See: https://nodejs.org/api/child_process.html#optionsstdio
Mapping it to `"inherit"` is fine for us and gets us one step closer in
getting `node-tap` working. I'm now at the stage where already the
coverage table is shown 🎉
2023-06-02 11:46:50 -04:00
|
|
|
|
|
|
|
assertStringIncludes(output, "foo");
|
|
|
|
assertStringIncludes(output, "close");
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2023-05-31 18:39:01 -04:00
|
|
|
Deno.test({
|
|
|
|
name: "[node/child_process spawn] supports SIGIOT signal",
|
|
|
|
ignore: Deno.build.os === "windows",
|
|
|
|
async fn() {
|
2023-07-31 09:29:00 -04:00
|
|
|
// Note: attempting to kill Deno with SIGABRT causes the process to zombify on certain OSX builds
|
|
|
|
// eg: 22.5.0 Darwin Kernel Version 22.5.0: Mon Apr 24 20:53:19 PDT 2023; root:xnu-8796.121.2~5/RELEASE_ARM64_T6020 arm64
|
|
|
|
// M2 Pro running Ventura 13.4
|
|
|
|
|
|
|
|
// Spawn an infinite cat
|
|
|
|
const cp = spawn("cat", ["-"]);
|
2023-11-22 06:11:20 -05:00
|
|
|
const p = withTimeout<void>();
|
|
|
|
const pStdout = withTimeout<void>();
|
|
|
|
const pStderr = withTimeout<void>();
|
2023-05-31 18:39:01 -04:00
|
|
|
cp.on("exit", () => p.resolve());
|
2023-09-16 01:48:31 -04:00
|
|
|
cp.stdout.on("close", () => pStdout.resolve());
|
|
|
|
cp.stderr.on("close", () => pStderr.resolve());
|
2023-05-31 18:39:01 -04:00
|
|
|
cp.kill("SIGIOT");
|
2023-11-22 06:11:20 -05:00
|
|
|
await p.promise;
|
|
|
|
await pStdout.promise;
|
|
|
|
await pStderr.promise;
|
2023-05-31 18:39:01 -04:00
|
|
|
assert(cp.killed);
|
|
|
|
assertEquals(cp.signalCode, "SIGIOT");
|
|
|
|
},
|
|
|
|
});
|
2023-09-05 06:42:35 -04:00
|
|
|
|
|
|
|
// Regression test for https://github.com/denoland/deno/issues/20373
|
|
|
|
Deno.test(async function undefinedValueInEnvVar() {
|
2023-11-22 06:11:20 -05:00
|
|
|
const deferred = withTimeout<string>();
|
2023-09-05 06:42:35 -04:00
|
|
|
const env = spawn(
|
|
|
|
`"${Deno.execPath()}" eval -p "Deno.env.toObject().BAZ"`,
|
|
|
|
{
|
|
|
|
env: {
|
|
|
|
BAZ: "BAZ",
|
|
|
|
NO_COLOR: "true",
|
|
|
|
UNDEFINED_ENV: undefined,
|
|
|
|
// deno-lint-ignore no-explicit-any
|
|
|
|
NULL_ENV: null as any,
|
|
|
|
},
|
|
|
|
shell: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
try {
|
|
|
|
let envOutput = "";
|
|
|
|
|
|
|
|
assert(env.stdout);
|
2023-11-22 06:11:20 -05:00
|
|
|
env.on("error", (err: Error) => deferred.reject(err));
|
2023-09-05 06:42:35 -04:00
|
|
|
env.stdout.on("data", (data) => {
|
|
|
|
envOutput += data;
|
|
|
|
});
|
|
|
|
env.on("close", () => {
|
2023-11-22 06:11:20 -05:00
|
|
|
deferred.resolve(envOutput.trim());
|
2023-09-05 06:42:35 -04:00
|
|
|
});
|
2023-11-22 06:11:20 -05:00
|
|
|
await deferred.promise;
|
2023-09-05 06:42:35 -04:00
|
|
|
} finally {
|
|
|
|
env.kill();
|
|
|
|
}
|
2023-11-22 06:11:20 -05:00
|
|
|
const value = await deferred.promise;
|
2023-09-05 06:42:35 -04:00
|
|
|
assertEquals(value, "BAZ");
|
|
|
|
});
|
|
|
|
|
|
|
|
// Regression test for https://github.com/denoland/deno/issues/20373
|
|
|
|
Deno.test(function spawnSyncUndefinedValueInEnvVar() {
|
|
|
|
const ret = spawnSync(
|
|
|
|
`"${Deno.execPath()}" eval -p "Deno.env.toObject().BAZ"`,
|
|
|
|
{
|
|
|
|
env: {
|
|
|
|
BAZ: "BAZ",
|
|
|
|
NO_COLOR: "true",
|
|
|
|
UNDEFINED_ENV: undefined,
|
|
|
|
// deno-lint-ignore no-explicit-any
|
|
|
|
NULL_ENV: null as any,
|
|
|
|
},
|
|
|
|
shell: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
assertEquals(ret.status, 0);
|
|
|
|
assertEquals(ret.stdout.toString("utf-8").trim(), "BAZ");
|
|
|
|
});
|
2023-11-10 00:59:39 -05:00
|
|
|
|
|
|
|
Deno.test(function spawnSyncStdioUndefined() {
|
|
|
|
const ret = spawnSync(
|
|
|
|
`"${Deno.execPath()}" eval "console.log('hello');console.error('world')"`,
|
|
|
|
{
|
|
|
|
stdio: [undefined, undefined, undefined],
|
|
|
|
shell: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
assertEquals(ret.status, 0);
|
|
|
|
assertEquals(ret.stdout.toString("utf-8").trim(), "hello");
|
|
|
|
assertEquals(ret.stderr.toString("utf-8").trim(), "world");
|
|
|
|
});
|
2023-11-27 19:54:01 -05:00
|
|
|
|
|
|
|
Deno.test(function spawnSyncExitNonZero() {
|
|
|
|
const ret = spawnSync(
|
|
|
|
`"${Deno.execPath()}" eval "Deno.exit(22)"`,
|
|
|
|
{ shell: true },
|
|
|
|
);
|
|
|
|
|
|
|
|
assertEquals(ret.status, 22);
|
|
|
|
});
|
2023-12-19 21:25:09 -05:00
|
|
|
|
|
|
|
// https://github.com/denoland/deno/issues/21630
|
|
|
|
Deno.test(async function forkIpcKillDoesNotHang() {
|
|
|
|
const testdataDir = path.join(
|
|
|
|
path.dirname(path.fromFileUrl(import.meta.url)),
|
|
|
|
"testdata",
|
|
|
|
);
|
|
|
|
const script = path.join(
|
|
|
|
testdataDir,
|
|
|
|
"node_modules",
|
|
|
|
"foo",
|
|
|
|
"index.js",
|
|
|
|
);
|
|
|
|
const p = Promise.withResolvers<void>();
|
|
|
|
const cp = CP.fork(script, [], {
|
|
|
|
cwd: testdataDir,
|
|
|
|
stdio: ["inherit", "inherit", "inherit", "ipc"],
|
|
|
|
});
|
|
|
|
cp.on("close", () => p.resolve());
|
|
|
|
cp.kill();
|
|
|
|
|
|
|
|
await p.promise;
|
|
|
|
});
|