Skip to content

Commit 097acb3

Browse files
committed
test: make test-worker-prof more robust
Use a JS workload instead of repeating FS operations to make it less flaky on machines with little resources.
1 parent 484ad83 commit 097acb3

File tree

1 file changed

+46
-58
lines changed

1 file changed

+46
-58
lines changed

test/sequential/test-worker-prof.js

+46-58
Original file line numberDiff line numberDiff line change
@@ -1,80 +1,68 @@
11
'use strict';
22
const common = require('../common');
33
const tmpdir = require('../common/tmpdir');
4+
const fixtures = require('../common/fixtures');
45
const fs = require('fs');
56
const assert = require('assert');
6-
const util = require('util');
7-
const { join } = require('path');
8-
const { spawnSync } = require('child_process');
7+
const { spawnSyncAndExitWithoutError } = require('../common/child_process');
98

109
// Test that --prof also tracks Worker threads.
1110
// Refs: https://github.com/nodejs/node/issues/24016
1211

1312
if (process.argv[2] === 'child') {
14-
const fs = require('fs');
15-
let files = fs.readdirSync(tmpdir.path);
16-
const plog = files.filter((name) => /\.log$/.test(name))[0];
17-
if (plog === undefined) {
18-
console.error('`--prof` did not produce a profile log for parent thread!');
19-
process.exit(1);
20-
}
21-
const pingpong = `
22-
let counter = 0;
23-
const fs = require('fs');
24-
const { Worker, parentPort } = require('worker_threads');
25-
parentPort.on('message', (m) => {
26-
if (counter++ === 1024)
27-
process.exit(0);
28-
parentPort.postMessage(
29-
fs.readFileSync(m.toString()).slice(0, 1024 * 1024));
30-
});
31-
`;
13+
let files = fs.readdirSync(process.cwd());
14+
console.log('files found in working directory before worker starts', files);
15+
const parentProf = files.find((name) => name.endsWith('.log'));
16+
assert(parentProf, '--prof did not produce a profile log for parent thread');
17+
console.log('parent prof file:', parentProf);
3218

3319
const { Worker } = require('worker_threads');
34-
const w = new Worker(pingpong, { eval: true });
35-
w.on('message', (m) => {
36-
w.postMessage(__filename);
20+
const w = new Worker(fixtures.path('workload', 'fibonacci.js'), {
21+
env: { FIB: 20, }
3722
});
3823

3924
w.on('exit', common.mustCall(() => {
40-
files = fs.readdirSync(tmpdir.path);
41-
const wlog = files.filter((name) => /\.log$/.test(name) && name !== plog)[0];
42-
if (wlog === undefined) {
43-
console.error('`--prof` did not produce a profile log' +
44-
' for worker thread!');
45-
process.exit(1);
46-
}
47-
process.exit(0);
25+
files = fs.readdirSync(process.cwd());
26+
console.log('files found in working directory before worker exits', files);
27+
const workerProf = files.find((name) => name !== parentProf && name.endsWith('.log'));
28+
assert(workerProf, '--prof did not produce a profile log for worker thread');
29+
console.log('worker prof file:', workerProf);
4830
}));
49-
w.postMessage(__filename);
5031
} else {
5132
tmpdir.refresh();
52-
const timeout = common.platformTimeout(30_000);
53-
const spawnResult = spawnSync(
33+
34+
const workerProfRegexp = /worker prof file: (.+\.log)/;
35+
const parentProfRegexp = /parent prof file: (.+\.log)/;
36+
const { stdout } = spawnSyncAndExitWithoutError(
5437
process.execPath, ['--prof', __filename, 'child'],
55-
{ cwd: tmpdir.path, encoding: 'utf8', timeout });
56-
assert.strictEqual(spawnResult.stderr.toString(), '',
57-
`child exited with an error: \
58-
${util.inspect(spawnResult)}`);
59-
assert.strictEqual(spawnResult.signal, null,
60-
`child exited with signal: ${util.inspect(spawnResult)}`);
61-
assert.strictEqual(spawnResult.status, 0,
62-
`child exited with non-zero status: \
63-
${util.inspect(spawnResult)}`);
64-
const files = fs.readdirSync(tmpdir.path);
65-
const logfiles = files.filter((name) => /\.log$/.test(name));
66-
assert.strictEqual(logfiles.length, 2); // Parent thread + child thread.
38+
{ cwd: tmpdir.path, encoding: 'utf8'
39+
}, {
40+
stdout(output) {
41+
assert.match(output, workerProfRegexp);
42+
assert.match(output, parentProfRegexp);
43+
return true;
44+
}
45+
});
46+
const workerProf = stdout.match(workerProfRegexp)[1];
47+
const parentProf = stdout.match(parentProfRegexp)[1];
48+
49+
const logfiles = fs.readdirSync(tmpdir.path).filter((name) => name.endsWith('.log'));
50+
assert.deepStrictEqual(new Set(logfiles), new Set([workerProf, parentProf]));
51+
52+
const workerLines = fs.readFileSync(tmpdir.resolve(workerProf), 'utf8').split('\n');
53+
const parentLines = fs.readFileSync(tmpdir.resolve(parentProf), 'utf8').split('\n');
6754

68-
for (const logfile of logfiles) {
69-
const lines = fs.readFileSync(
70-
join(tmpdir.path, logfile), 'utf8').split('\n');
71-
const ticks = lines.filter((line) => /^tick,/.test(line)).length;
55+
const workerTicks = workerLines.filter((line) => line.startsWith('tick'));
56+
const parentTicks = parentLines.filter((line) => line.startsWith('tick'));
7257

73-
// Test that at least 15 ticks have been recorded for both parent and child
74-
// threads. When not tracking Worker threads, only 1 or 2 ticks would
75-
// have been recorded.
76-
// When running locally, this number is usually around 200
77-
// for both threads, so 15 seems like a very safe threshold.
78-
assert(ticks >= 15, `${ticks} >= 15`);
79-
}
58+
// Test that at least 15 ticks have been recorded for both parent and child
59+
// threads. When not tracking Worker threads, only 1 or 2 ticks would
60+
// have been recorded.
61+
// When running locally, this number is usually around 100
62+
// for both threads, so 15 seems like a very safe threshold.
63+
// TODO(joyeecheung): if this still flakes, we could log the time spent on the
64+
// workload in the worker and the parent, and divide that with v8's
65+
// prof_sampling_interval to get a better expectation.
66+
assert(workerTicks.length > 15, `worker ticks ${workerTicks} <= 15`);
67+
assert(parentTicks.length > 15, `parent ticks ${parentTicks} <= 15`);
8068
}

0 commit comments

Comments
 (0)