mirror of
https://github.com/anomalyco/opencode.git
synced 2026-04-25 07:15:19 +00:00
151 lines
4.0 KiB
TypeScript
151 lines
4.0 KiB
TypeScript
import { spawn as launch, type ChildProcess } from "child_process"
|
|
import { buffer } from "node:stream/consumers"
|
|
|
|
export namespace Process {
|
|
export type Stdio = "inherit" | "pipe" | "ignore"
|
|
|
|
export interface Options {
|
|
cwd?: string
|
|
env?: NodeJS.ProcessEnv | null
|
|
stdin?: Stdio
|
|
stdout?: Stdio
|
|
stderr?: Stdio
|
|
abort?: AbortSignal
|
|
kill?: NodeJS.Signals | number
|
|
timeout?: number
|
|
}
|
|
|
|
export interface RunOptions extends Omit<Options, "stdout" | "stderr"> {
|
|
nothrow?: boolean
|
|
}
|
|
|
|
export interface Result {
|
|
code: number
|
|
stdout: Buffer
|
|
stderr: Buffer
|
|
}
|
|
|
|
export interface TextResult extends Result {
|
|
text: string
|
|
}
|
|
|
|
export class RunFailedError extends Error {
|
|
readonly cmd: string[]
|
|
readonly code: number
|
|
readonly stdout: Buffer
|
|
readonly stderr: Buffer
|
|
|
|
constructor(cmd: string[], code: number, stdout: Buffer, stderr: Buffer) {
|
|
const text = stderr.toString().trim()
|
|
super(
|
|
text
|
|
? `Command failed with code ${code}: ${cmd.join(" ")}\n${text}`
|
|
: `Command failed with code ${code}: ${cmd.join(" ")}`,
|
|
)
|
|
this.name = "ProcessRunFailedError"
|
|
this.cmd = [...cmd]
|
|
this.code = code
|
|
this.stdout = stdout
|
|
this.stderr = stderr
|
|
}
|
|
}
|
|
|
|
export type Child = ChildProcess & { exited: Promise<number> }
|
|
|
|
export function spawn(cmd: string[], opts: Options = {}): Child {
|
|
if (cmd.length === 0) throw new Error("Command is required")
|
|
opts.abort?.throwIfAborted()
|
|
|
|
const proc = launch(cmd[0], cmd.slice(1), {
|
|
cwd: opts.cwd,
|
|
env: opts.env === null ? {} : opts.env ? { ...process.env, ...opts.env } : undefined,
|
|
stdio: [opts.stdin ?? "ignore", opts.stdout ?? "ignore", opts.stderr ?? "ignore"],
|
|
})
|
|
|
|
let closed = false
|
|
let timer: ReturnType<typeof setTimeout> | undefined
|
|
|
|
const abort = () => {
|
|
if (closed) return
|
|
if (proc.exitCode !== null || proc.signalCode !== null) return
|
|
closed = true
|
|
|
|
proc.kill(opts.kill ?? "SIGTERM")
|
|
|
|
const ms = opts.timeout ?? 5_000
|
|
if (ms <= 0) return
|
|
timer = setTimeout(() => proc.kill("SIGKILL"), ms)
|
|
}
|
|
|
|
const exited = new Promise<number>((resolve, reject) => {
|
|
const done = () => {
|
|
opts.abort?.removeEventListener("abort", abort)
|
|
if (timer) clearTimeout(timer)
|
|
}
|
|
|
|
proc.once("exit", (code, signal) => {
|
|
done()
|
|
resolve(code ?? (signal ? 1 : 0))
|
|
})
|
|
|
|
proc.once("error", (error) => {
|
|
done()
|
|
reject(error)
|
|
})
|
|
})
|
|
|
|
if (opts.abort) {
|
|
opts.abort.addEventListener("abort", abort, { once: true })
|
|
if (opts.abort.aborted) abort()
|
|
}
|
|
|
|
const child = proc as Child
|
|
child.exited = exited
|
|
return child
|
|
}
|
|
|
|
export async function run(cmd: string[], opts: RunOptions = {}): Promise<Result> {
|
|
const proc = spawn(cmd, {
|
|
cwd: opts.cwd,
|
|
env: opts.env,
|
|
stdin: opts.stdin,
|
|
abort: opts.abort,
|
|
kill: opts.kill,
|
|
timeout: opts.timeout,
|
|
stdout: "pipe",
|
|
stderr: "pipe",
|
|
})
|
|
|
|
if (!proc.stdout || !proc.stderr) throw new Error("Process output not available")
|
|
|
|
const out = await Promise.all([proc.exited, buffer(proc.stdout), buffer(proc.stderr)])
|
|
.then(([code, stdout, stderr]) => ({
|
|
code,
|
|
stdout,
|
|
stderr,
|
|
}))
|
|
.catch((err: unknown) => {
|
|
if (!opts.nothrow) throw err
|
|
return {
|
|
code: 1,
|
|
stdout: Buffer.alloc(0),
|
|
stderr: Buffer.from(err instanceof Error ? err.message : String(err)),
|
|
}
|
|
})
|
|
if (out.code === 0 || opts.nothrow) return out
|
|
throw new RunFailedError(cmd, out.code, out.stdout, out.stderr)
|
|
}
|
|
|
|
export async function text(cmd: string[], opts: RunOptions = {}): Promise<TextResult> {
|
|
const out = await run(cmd, opts)
|
|
return {
|
|
...out,
|
|
text: out.stdout.toString(),
|
|
}
|
|
}
|
|
|
|
export async function lines(cmd: string[], opts: RunOptions = {}): Promise<string[]> {
|
|
return (await text(cmd, opts)).text.split(/\r?\n/).filter(Boolean)
|
|
}
|
|
}
|