dd5bcbd0f07cbb448f0ee67b2a4382fa189e9f38.svn-base
4.37 KB
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
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
'use strict'
const Base = require('./base.js')
const assert = require('assert')
const util = require('util')
const ownOr = require('own-or')
const path = require('path')
const cleanYamlObject = require('./clean-yaml-object.js')
const cp = require('child_process')
class Spawn extends Base {
constructor (options) {
// figure out the name before calling super()
options = options || {}
const cwd = ownOr(options, 'cwd', process.cwd())
const command = options.command
if (!command)
throw new TypeError('no command provided')
const args = ownOr(options, 'args', [])
options.name = options.name || Spawn.procName(cwd, command, args)
super(options)
this.command = options.command
this.args = options.args
// stdout must be a pipe
if (options.stdio) {
if (typeof options.stdio === 'string')
this.stdio = [ options.stdio, 'pipe', options.stdio ]
else
this.stdio = options.stdio.slice(0)
} else
this.stdio = [ 0, 'pipe', 2 ]
this.stdio[1] = 'pipe'
options.stdio = this.stdio
if (!options.env)
options.env = process.env
this.env = {
...(options.env),
TAP_CHILD_ID: options.childId
|| options.env.TAP_CHILD_ID
|| /* istanbul ignore next */ 0,
TAP: '1',
TAP_BAIL: this.bail ? '1' : '0',
}
// prune off the extraneous bits so we're not logging the world
this.options.env = Object.keys(this.options.env).reduce((env, k) => {
if (process.env[k] !== this.options.env[k])
env[k] = this.options.env[k]
return env
}, {})
this.cwd = cwd
options.cwd = this.cwd
this.processDB = ownOr(options, 'processDB', null) || {
spawn: (name, ...rest) => cp.spawn(...rest)
}
delete options.processDB
this.proc = null
}
endAll () {
if (this.proc)
this.proc.kill('SIGKILL')
this.parser.abort('test unfinished')
this.callCb()
}
main (cb) {
this.cb = cb
this.setTimeout(this.options.timeout)
this.parser.on('comment', c => {
const tomatch = c.match(/# timeout=([0-9]+)\n$/)
if (tomatch)
this.setTimeout(+tomatch[1])
})
const options = {
...(this.options),
cwd: this.cwd,
env: this.env,
stdio: this.stdio,
}
try {
this.emit('preprocess', options)
const proc = this.proc =
this.processDB.spawn(this.name, this.command, this.args, options)
proc.stdout.pipe(this.parser)
proc.on('close', (code, signal) => this.onprocclose(code, signal))
proc.on('error', er => this.threw(er))
this.emit('process', proc)
if (this.parent)
this.parent.emit('spawn', this)
} catch (er) {
er.tapCaught = 'spawn'
this.threw(er)
}
}
callCb () {
if (this.cb)
this.cb()
this.cb = null
}
threw (er, extra, proxy) {
extra = Base.prototype.threw.call(this, er, extra, proxy)
extra = cleanYamlObject(extra)
// unhook entirely
this.parser.abort(er.message, extra)
if (this.proc) {
this.proc.stdout.removeAllListeners('data')
this.proc.stdout.removeAllListeners('end')
this.proc.removeAllListeners('close')
this.proc.kill('SIGKILL')
}
this.callCb()
}
onprocclose (code, signal) {
this.debug('SPAWN close %j %s', code, signal)
this.options.exitCode = code
if (signal)
this.options.signal = signal
// spawn closing with no tests is treated as a skip.
if (this.results && this.results.plan && this.results.plan.skipAll && !code && !signal)
this.options.skip = this.results.plan.skipReason || true
if (code || signal) {
this.results.ok = false
this.parser.ok = false
}
return this.callCb()
}
timeout (extra) {
if (this.proc) {
this.proc.kill('SIGTERM')
const t = setTimeout(() => {
if (!this.options.signal && this.options.exitCode === undefined) {
Base.prototype.timeout.call(this, extra)
this.proc.kill('SIGKILL')
}
}, 1000)
/* istanbul ignore else */
if (t.unref)
t.unref()
}
}
}
Spawn.procName = (cwd, command, args) => (
(command === process.execPath)
? path.basename(process.execPath) + ' ' + args.map(a =>
a.indexOf(cwd) === 0 ?
'./' + a.substr(cwd.length + 1).replace(/\\/g, '/')
: a).join(' ').trim()
: command + ' ' + args.join(' ')
).replace(/\\/g, '/')
module.exports = Spawn