-
-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathtest.js
309 lines (245 loc) · 9.83 KB
/
test.js
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
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
'use strict'
const { test } = require('node:test')
const tty = require('node:tty')
const Hook = require('./')
const ttySupportColor = tty.isatty(process.stdout.fd)
const proxyquire = require('proxyquire')
test('pre-commit', async (t) => {
await t.test('is exported as a function', (t) => {
t.assert.strictEqual(typeof Hook, 'function', 'Hook should be a function')
})
await t.test('can be initialized without a `new` keyword', (t) => {
const hook = Hook(function () {}, {
ignorestatus: true
})
t.assert.ok(hook instanceof Hook, 'hook should be an instance of Hook')
t.assert.strictEqual(typeof hook.parse, 'function', 'hook.parse should be a function')
})
await t.test('#parse', async (t) => {
let hook
t.beforeEach(() => {
hook = new Hook(function () {}, {
ignorestatus: true
})
})
await t.test('extracts configuration values from precommit.<flag>', (t) => {
hook.json = {
'precommit.silent': true
}
t.assert.strictEqual(hook.silent, false, 'hook.silent should initially be false')
hook.parse()
t.assert.strictEqual(hook.config.silent, true, 'hook.config.silent should be true after parsing')
t.assert.strictEqual(hook.silent, true, 'hook.silent should be true after parsing')
})
await t.test('extracts configuration values from pre-commit.<flag>', (t) => {
hook.json = {
'pre-commit.silent': true,
'pre-commit.colors': false
}
t.assert.strictEqual(hook.silent, false, 'hook.silent should initially be false')
t.assert.strictEqual(hook.colors, ttySupportColor, 'hook.colors should initially match ttySupportColor')
hook.parse()
t.assert.strictEqual(hook.config.silent, true, 'hook.config.silent should be true after parsing')
t.assert.strictEqual(hook.silent, true, 'hook.silent should be true after parsing')
t.assert.strictEqual(hook.colors, false, 'hook.colors should be false after parsing')
})
await t.test('normalizes the `pre-commit` to an array', (t) => {
hook.json = {
'pre-commit': 'test, cows, moo'
}
hook.parse()
t.assert.strictEqual(hook.config.run.length, 3, 'hook.config.run should have 3 items')
t.assert.deepStrictEqual(hook.config.run, ['test', 'cows', 'moo'], 'hook.config.run should contain the correct items')
})
await t.test('normalizes the `precommit` to an array', (t) => {
hook.json = {
precommit: 'test, cows, moo'
}
hook.parse()
t.assert.strictEqual(hook.config.run.length, 3, 'hook.config.run should have 3 items')
t.assert.deepStrictEqual(hook.config.run, ['test', 'cows', 'moo'], 'hook.config.run should contain the correct items')
})
await t.test('allows `pre-commit` object based syntax', (t) => {
hook.json = {
'pre-commit': {
run: 'test scripts go here',
silent: true,
colors: false
}
}
hook.parse()
t.assert.strictEqual(hook.config.run.length, 4, 'hook.config.run should have 4 items')
t.assert.deepStrictEqual(hook.config.run, ['test', 'scripts', 'go', 'here'], 'hook.config.run should contain the correct items')
t.assert.strictEqual(hook.silent, true, 'hook.silent should be true')
t.assert.strictEqual(hook.colors, false, 'hook.colors should be false')
})
await t.test('defaults to `test` if nothing is specified', (t) => {
hook.json = {
scripts: {
test: 'mocha test.js'
}
}
hook.parse()
t.assert.strictEqual(hook.config.run.length, 1, 'hook.config.run should have 1 item')
t.assert.deepStrictEqual(hook.config.run, ['test'], 'hook.config.run should contain the correct item')
})
await t.test('ignores the default npm.script.test placeholder', (t) => {
hook.json = {
scripts: {
test: 'echo "Error: no test specified" && exit 1'
}
}
hook.parse()
t.assert.strictEqual(typeof hook.config.run, 'undefined', 'hook.config.run should be undefined')
})
await t.test('overrides the `pre-commit` config property in package.json with the config inside `.pre-commit.json` if it exists', (t) => {
const Hook = proxyquire('.', {
'node:fs': {
existsSync () {
return true
},
readFileSync () {
const rawText = JSON.stringify({ run: ['lint', 'bench'] })
return Buffer.from(rawText)
}
}
})
hook = new Hook(function () {}, { ignorestatus: true })
t.assert.deepStrictEqual(hook.config.run, ['lint', 'bench'], 'hook.config.run should contain the correct items')
})
await t.test('should properly handle errors while trying to read and parse the contents of `.pre-commit.json`', (t) => {
let Hook = proxyquire('.', {
'node:fs': {
existsSync () {
return true
},
readFileSync () {
throw new Error()
}
}
})
hook = new Hook(exit)
Hook = proxyquire('.', {
'node:fs': {
existsSync () { return true },
readFileSync () {
return Buffer.from('{ "bad": [json }')
}
}
})
hook = new Hook(exit)
function exit (code, lines) {
t.assert.ok(lines.length !== 0, 'lines should not be empty')
t.assert.strictEqual(code, 1, 'exit code should be 1')
}
})
})
await t.test('#log', async (t) => {
await t.test('prefixes the logs with `pre-commit`', (t) => {
const hook = new Hook(function (code, lines) {
t.assert.strictEqual(code, 1, 'exit code should be 1')
t.assert.ok(Array.isArray(lines), 'lines should be an array')
t.assert.ok(lines[0].includes('pre-commit: '), 'first line should start with `pre-commit: `')
t.assert.ok(lines[1].includes('pre-commit: foo'), 'second line should start with `pre-commit: foo`')
t.assert.ok(lines[2].includes('pre-commit: '), 'third line should start with `pre-commit: `')
t.assert.strictEqual(lines.length, 3, 'total lines should be 3')
// color prefix check
lines.forEach(function (line) {
t.assert.strictEqual(line.includes('\u001b'), ttySupportColor)
})
}, { ignorestatus: true })
hook.config.silent = true
hook.log(['foo'])
})
await t.test('allows for a custom error code', (t) => {
const hook = new Hook(function (code) {
t.assert.strictEqual(code, 0, 'exit code should be 0')
}, { ignorestatus: true })
hook.config.silent = true
hook.log(['foo'], 0)
})
await t.test('allows strings to be split \\n', (t) => {
const hook = new Hook(function (code, lines) {
t.assert.strictEqual(code, 0, 'exit code should be 0')
t.assert.strictEqual(lines.length, 4, 'total lines should be 4')
t.assert.ok(lines[1].includes('pre-commit: foo'), 'second line should start with `pre-commit: foo`')
t.assert.ok(lines[2].includes('pre-commit: bar'), 'third line should start with `pre-commit: bar`')
}, { ignorestatus: true })
hook.config.silent = true
hook.log('foo\nbar', 0)
})
await t.test('does not output colors when configured to do so', (t) => {
const hook = new Hook(function (code, lines) {
t.assert.strictEqual(code, 0, 'exit code should be 0')
lines.forEach(function (line) {
t.assert.strictEqual(line.includes('\u001b'), false, 'line should not include color prefix')
})
}, { ignorestatus: true })
hook.config.silent = true
hook.config.colors = false
hook.log('foo\nbar', 0)
})
await t.test('output lines to stderr if error code 1', (t) => {
const err = console.error
const hook = new Hook(function () {
console.error = err
}, { ignorestatus: true })
console.error = function (line) {
t.assert.ok(line.includes('pre-commit: '), true)
}
hook.config.colors = false
hook.log('foo\nbar', 1)
})
await t.test('output lines to stderr if error code 0', (t) => {
const log = console.log
const hook = new Hook(function () {
console.log = log
}, { ignorestatus: true })
console.log = function (line) {
t.assert.ok(line.includes('pre-commit: '), true)
}
hook.config.colors = false
hook.log('foo\nbar', 0)
})
})
await t.test('#run', async (t) => {
await t.test('runs the specified scripts and exit with 0 on no error', (t) => {
const hook = new Hook(function (code, lines) {
t.assert.strictEqual(code, 0, 'exit code should be 0')
t.assert.strictEqual(typeof lines, 'undefined', 'lines should be undefined')
}, { ignorestatus: true })
hook.config.run = ['example-pass']
hook.run()
})
await t.test('runs the specified test and exits with 1 on error', (t) => {
const hook = new Hook(function (code, lines) {
t.assert.strictEqual(code, 1, 'exit code should be 1')
t.assert.ok(Array.isArray(lines), 'lines should be an array')
t.assert.ok(lines[1].includes('`example-fail`'), true)
t.assert.ok(lines[2].includes('code (1)'), true)
}, { ignorestatus: true })
hook.config.run = ['example-fail']
hook.run()
})
})
await t.test('skips pre-commit if no staged changes', (t) => {
const Hook = proxyquire('.', {
'node:child_process': {
execSync (command) {
if (command === 'git diff --cached --quiet') {
throw new Error('No staged changes')
}
}
}
})
const hook = new Hook(
function (code, lines) {
t.assert.strictEqual(code, 0, 'exit code should be 0')
t.assert.strictEqual(lines, undefined, 'lines should be undefined')
},
{ ignorestatus: true }
)
hook.config.run = ['example-pass']
hook.run()
})
})