-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathrun.ts
153 lines (133 loc) · 5.08 KB
/
run.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
import install, { Logger } from "../plumbing/install.ts"
import useShellEnv from '../hooks/useShellEnv.ts'
import usePantry from '../hooks/usePantry.ts'
import hydrate from "../plumbing/hydrate.ts"
import resolve from "../plumbing/resolve.ts"
import { PkgxError } from "../utils/error.ts"
import { spawn } from "node:child_process"
import useSync from "../hooks/useSync.ts"
import which from "../plumbing/which.ts"
import link from "../plumbing/link.ts"
import { is_what } from "../deps.ts"
import Path from "../utils/Path.ts"
const { isArray } = is_what
interface OptsEx {
env?: Record<string, string | undefined>
logger?: Logger
}
type Options = {
stdout?: boolean
stderr?: boolean
status?: boolean
} & OptsEx
type Cmd = string | (string | Path)[]
/// if you pass a single string we call that string via /bin/sh
/// if you don’t want that pass an array of args
export default async function run(cmd: Cmd, opts?: OptsEx): Promise<void>;
export default async function run(cmd: Cmd, opts: {stdout: true} & OptsEx): Promise<{ stdout: string }>;
export default async function run(cmd: Cmd, opts: {stderr: true} & OptsEx): Promise<{ stderr: string }>;
export default async function run(cmd: Cmd, opts: {status: true} & OptsEx): Promise<{ status: number }>;
export default async function run(cmd: Cmd, opts: {stdout: true, stderr: true} & OptsEx): Promise<{ stdout: string, stderr: string }>;
export default async function run(cmd: Cmd, opts: {stdout: true, status: true} & OptsEx): Promise<{ stdout: string, status: number }>;
export default async function run(cmd: Cmd, opts: {stderr: true, status: true} & OptsEx): Promise<{ stderr: string, status: number }>;
export default async function run(cmd: Cmd, opts: {stdout: true, stderr: true, status: true } & OptsEx): Promise<{ stdout: string, stderr: string, status: number }>;
export default async function run(cmd: Cmd, opts?: Options): Promise<void|{ stdout?: string|undefined; stderr?: string|undefined; status?: number|undefined; }> {
const { usesh, arg0: whom } = (() => {
if (!isArray(cmd)) {
const s = cmd.trim()
const i = s.indexOf(' ')
if (i == -1) {
cmd = []
return { usesh: false, arg0: s }
} else if (Deno.build.os == 'windows') {
cmd = cmd.split(/\s+/)
const arg0 = cmd.shift()! as string
return { usesh: false, arg0 }
} else {
const arg0 = s.slice(0, i)
cmd = s.slice(i + 1)
return { usesh: true, arg0 }
}
} else if (cmd.length == 0) {
throw new RunError('EUSAGE', `\`cmd\` evaluated empty: ${cmd}`)
} else {
return {
usesh: false,
arg0: cmd.shift()!.toString().trim()
}
}
})()
const { env, shebang } = await setup(whom, opts?.env ?? Deno.env.toObject(), opts?.logger)
const arg0 = usesh ? '/bin/sh' : shebang.shift()!
const args = usesh
? ['-c', `${shebang.join(' ')} ${cmd}`]
: [...shebang, ...(cmd as (string | Path)[]).map(x => x.toString())]
return new Promise((resolve, reject) => {
const proc = spawn(arg0, args, {
env,
stdio: [
"pipe",
opts?.stdout ? 'pipe' : 'inherit',
opts?.stderr ? 'pipe' : 'inherit'
],
/// on windows .bat files are not executable unless invoked via a shell
/// our provides database deliberately excludes `.bat` so that the same
/// filename is used for all platforms, this works since provided we use
/// a shell to execute, we don’t need to know the extension
shell: Deno.build.os == 'windows'
})
let stdout = '', stderr = ''
proc.stdout?.on('data', data => stdout += data)
proc.stderr?.on('data', data => stderr += data)
proc.on('close', status => {
if (status && !opts?.status) {
const err = new RunError('EIO', `${cmd} exited with: ${status}`)
err.cause = status
reject(err)
} else {
const fulfill = resolve as ({}) => void
fulfill({ stdout, stderr, status })
}
})
})
}
async function setup(cmd: string, env: Record<string, string | undefined>, logger: Logger | undefined) {
const pantry = usePantry()
const sh = useShellEnv()
const { install, link } = _internals
if (pantry.missing()) {
await useSync()
}
const wut = await which(cmd)
if (!wut) throw new RunError('ENOENT', `No project in pantry provides ${cmd}`)
const { pkgs } = await hydrate(wut)
const { pending, installed } = await resolve(pkgs)
for (const pkg of pending) {
const installation = await install(pkg, logger)
await link(installation)
installed.push(installation)
}
const pkgenv = await sh.map({ installations: installed })
for (const [key, value] of Object.entries(env)) {
if (!value) {
continue
} else if (pkgenv[key]) {
pkgenv[key].push(value)
} else {
pkgenv[key] = [value]
}
}
return { env: sh.flatten(pkgenv), shebang: wut.shebang }
}
type RunErrorCode = 'ENOENT' | 'EUSAGE' | 'EIO'
export class RunError extends PkgxError {
code: RunErrorCode
constructor(code: RunErrorCode, message: string) {
super(message)
this.code = code
}
}
const _internals = {
install,
link
}