2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2020-03-08 08:09:22 -04:00
|
|
|
import { File } from "./files.ts";
|
|
|
|
import { close } from "./ops/resources.ts";
|
2019-09-02 17:07:11 -04:00
|
|
|
import { ReadCloser, WriteCloser } from "./io.ts";
|
|
|
|
import { readAll } from "./buffer.ts";
|
2020-03-09 10:18:02 -04:00
|
|
|
import { kill, runStatus as runStatusOp, run as runOp } from "./ops/process.ts";
|
2018-11-15 23:07:40 -05:00
|
|
|
|
2019-02-21 15:52:35 -05:00
|
|
|
/** How to handle subprocess stdio.
|
2018-11-15 23:07:40 -05:00
|
|
|
*
|
|
|
|
* "inherit" The default if unspecified. The child inherits from the
|
|
|
|
* corresponding parent descriptor.
|
|
|
|
*
|
|
|
|
* "piped" A new pipe should be arranged to connect the parent and child
|
|
|
|
* subprocesses.
|
|
|
|
*
|
|
|
|
* "null" This stream will be ignored. This is the equivalent of attaching the
|
|
|
|
* stream to /dev/null.
|
|
|
|
*/
|
|
|
|
export type ProcessStdio = "inherit" | "piped" | "null";
|
|
|
|
|
|
|
|
// TODO Maybe extend VSCode's 'CommandOptions'?
|
|
|
|
// See https://code.visualstudio.com/docs/editor/tasks-appendix#_schema-for-tasksjson
|
|
|
|
export interface RunOptions {
|
|
|
|
args: string[];
|
|
|
|
cwd?: string;
|
2019-02-15 10:37:04 -05:00
|
|
|
env?: { [key: string]: string };
|
2019-06-21 19:00:14 -04:00
|
|
|
stdout?: ProcessStdio | number;
|
|
|
|
stderr?: ProcessStdio | number;
|
|
|
|
stdin?: ProcessStdio | number;
|
2018-11-15 23:07:40 -05:00
|
|
|
}
|
|
|
|
|
2019-08-26 08:50:21 -04:00
|
|
|
async function runStatus(rid: number): Promise<ProcessStatus> {
|
2020-03-09 10:18:02 -04:00
|
|
|
const res = await runStatusOp(rid);
|
2019-03-09 12:30:38 -05:00
|
|
|
|
2019-08-26 08:50:21 -04:00
|
|
|
if (res.gotSignal) {
|
|
|
|
const signal = res.exitSignal;
|
2019-03-09 12:30:38 -05:00
|
|
|
return { signal, success: false };
|
|
|
|
} else {
|
2019-08-26 08:50:21 -04:00
|
|
|
const code = res.exitCode;
|
2019-03-09 12:30:38 -05:00
|
|
|
return { code, success: code === 0 };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-15 23:07:40 -05:00
|
|
|
export class Process {
|
|
|
|
readonly rid: number;
|
|
|
|
readonly pid: number;
|
|
|
|
readonly stdin?: WriteCloser;
|
|
|
|
readonly stdout?: ReadCloser;
|
|
|
|
readonly stderr?: ReadCloser;
|
|
|
|
|
|
|
|
// @internal
|
2019-08-26 08:50:21 -04:00
|
|
|
constructor(res: RunResponse) {
|
|
|
|
this.rid = res.rid;
|
|
|
|
this.pid = res.pid;
|
2018-11-15 23:07:40 -05:00
|
|
|
|
2019-08-26 08:50:21 -04:00
|
|
|
if (res.stdinRid && res.stdinRid > 0) {
|
|
|
|
this.stdin = new File(res.stdinRid);
|
2018-11-15 23:07:40 -05:00
|
|
|
}
|
|
|
|
|
2019-08-26 08:50:21 -04:00
|
|
|
if (res.stdoutRid && res.stdoutRid > 0) {
|
|
|
|
this.stdout = new File(res.stdoutRid);
|
2018-11-15 23:07:40 -05:00
|
|
|
}
|
|
|
|
|
2019-08-26 08:50:21 -04:00
|
|
|
if (res.stderrRid && res.stderrRid > 0) {
|
|
|
|
this.stderr = new File(res.stderrRid);
|
2018-11-15 23:07:40 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async status(): Promise<ProcessStatus> {
|
|
|
|
return await runStatus(this.rid);
|
|
|
|
}
|
|
|
|
|
2018-11-30 13:44:05 -05:00
|
|
|
/** Buffer the stdout and return it as Uint8Array after EOF.
|
2019-03-28 16:09:46 -04:00
|
|
|
* You must set stdout to "piped" when creating the process.
|
2018-11-30 13:44:05 -05:00
|
|
|
* This calls close() on stdout after its done.
|
|
|
|
*/
|
|
|
|
async output(): Promise<Uint8Array> {
|
|
|
|
if (!this.stdout) {
|
|
|
|
throw new Error("Process.output: stdout is undefined");
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
return await readAll(this.stdout);
|
|
|
|
} finally {
|
|
|
|
this.stdout.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-28 16:09:46 -04:00
|
|
|
/** Buffer the stderr and return it as Uint8Array after EOF.
|
|
|
|
* You must set stderr to "piped" when creating the process.
|
|
|
|
* This calls close() on stderr after its done.
|
|
|
|
*/
|
|
|
|
async stderrOutput(): Promise<Uint8Array> {
|
|
|
|
if (!this.stderr) {
|
|
|
|
throw new Error("Process.stderrOutput: stderr is undefined");
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
return await readAll(this.stderr);
|
|
|
|
} finally {
|
|
|
|
this.stderr.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-15 23:07:40 -05:00
|
|
|
close(): void {
|
|
|
|
close(this.rid);
|
|
|
|
}
|
2019-04-21 21:26:56 -04:00
|
|
|
|
|
|
|
kill(signo: number): void {
|
|
|
|
kill(this.pid, signo);
|
|
|
|
}
|
2018-11-15 23:07:40 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface ProcessStatus {
|
|
|
|
success: boolean;
|
|
|
|
code?: number;
|
|
|
|
signal?: number; // TODO: Make this a string, e.g. 'SIGTERM'.
|
|
|
|
}
|
|
|
|
|
2019-06-21 19:00:14 -04:00
|
|
|
function isRid(arg: unknown): arg is number {
|
|
|
|
return !isNaN(arg as number);
|
|
|
|
}
|
|
|
|
|
2019-08-26 08:50:21 -04:00
|
|
|
interface RunResponse {
|
|
|
|
rid: number;
|
|
|
|
pid: number;
|
|
|
|
stdinRid: number | null;
|
|
|
|
stdoutRid: number | null;
|
|
|
|
stderrRid: number | null;
|
|
|
|
}
|
2019-02-21 15:52:35 -05:00
|
|
|
/**
|
|
|
|
* Spawns new subprocess.
|
|
|
|
*
|
|
|
|
* Subprocess uses same working directory as parent process unless `opt.cwd`
|
|
|
|
* is specified.
|
|
|
|
*
|
|
|
|
* Environmental variables for subprocess can be specified using `opt.env`
|
|
|
|
* mapping.
|
|
|
|
*
|
|
|
|
* By default subprocess inherits stdio of parent process. To change that
|
2019-06-21 19:00:14 -04:00
|
|
|
* `opt.stdout`, `opt.stderr` and `opt.stdin` can be specified independently -
|
|
|
|
* they can be set to either `ProcessStdio` or `rid` of open file.
|
2019-02-21 15:52:35 -05:00
|
|
|
*/
|
2020-03-10 12:08:58 -04:00
|
|
|
export function run({
|
|
|
|
args,
|
|
|
|
cwd = undefined,
|
|
|
|
env = {},
|
|
|
|
stdout = "inherit",
|
|
|
|
stderr = "inherit",
|
|
|
|
stdin = "inherit"
|
|
|
|
}: RunOptions): Process {
|
|
|
|
const res = runOp({
|
|
|
|
args: args.map(String),
|
|
|
|
cwd,
|
|
|
|
env: Object.entries(env),
|
|
|
|
stdin: isRid(stdin) ? "" : stdin,
|
|
|
|
stdout: isRid(stdout) ? "" : stdout,
|
|
|
|
stderr: isRid(stderr) ? "" : stderr,
|
|
|
|
stdinRid: isRid(stdin) ? stdin : 0,
|
|
|
|
stdoutRid: isRid(stdout) ? stdout : 0,
|
|
|
|
stderrRid: isRid(stderr) ? stderr : 0
|
|
|
|
}) as RunResponse;
|
2018-11-15 23:07:40 -05:00
|
|
|
return new Process(res);
|
|
|
|
}
|