Skip to content

Commit e3572d0

Browse files
authored
Do not write verbose logs synchronously (#1167)
1 parent ab886ff commit e3572d0

File tree

2 files changed

+25
-6
lines changed

2 files changed

+25
-6
lines changed

lib/verbose/log.js

+11-6
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,23 @@
1-
import {writeFileSync} from 'node:fs';
21
import {inspect} from 'node:util';
32
import {escapeLines} from '../arguments/escape.js';
43
import {defaultVerboseFunction} from './default.js';
54
import {applyVerboseOnLines} from './custom.js';
65

7-
// Write synchronously to ensure lines are properly ordered and not interleaved with `stdout`
6+
// This prints on stderr.
7+
// If the subprocess prints on stdout and is using `stdout: 'inherit'`,
8+
// there is a chance both writes will compete (introducing a race condition).
9+
// This means their respective order is not deterministic.
10+
// In particular, this means the verbose command lines might be after the start of the subprocess output.
11+
// Using synchronous I/O does not solve this problem.
12+
// However, this only seems to happen when the stdout/stderr target
13+
// (e.g. a terminal) is being written to by many subprocesses at once, which is unlikely in real scenarios.
814
export const verboseLog = ({type, verboseMessage, fdNumber, verboseInfo, result}) => {
915
const verboseObject = getVerboseObject({type, result, verboseInfo});
1016
const printedLines = getPrintedLines(verboseMessage, verboseObject);
1117
const finalLines = applyVerboseOnLines(printedLines, verboseInfo, fdNumber);
12-
writeFileSync(STDERR_FD, finalLines);
18+
if (finalLines !== '') {
19+
console.warn(finalLines.slice(0, -1));
20+
}
1321
};
1422

1523
const getVerboseObject = ({
@@ -35,9 +43,6 @@ const getPrintedLine = verboseObject => {
3543
return {verboseLine, verboseObject};
3644
};
3745

38-
// Unless a `verbose` function is used, print all logs on `stderr`
39-
const STDERR_FD = 2;
40-
4146
// Serialize any type to a line string, for logging
4247
export const serializeVerboseMessage = message => {
4348
const messageString = typeof message === 'string' ? message : inspect(message);

test/verbose/log.js

+14
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,8 @@ import test from 'ava';
33
import {setFixtureDirectory} from '../helpers/fixtures-directory.js';
44
import {foobarString} from '../helpers/input.js';
55
import {nestedSubprocess} from '../helpers/nested.js';
6+
import {getNormalizedLines, getCommandLine, getCompletionLine} from '../helpers/verbose.js';
7+
import {PARALLEL_COUNT} from '../helpers/parallel.js';
68

79
setFixtureDirectory();
810

@@ -25,3 +27,15 @@ const testColor = async (t, expectedResult, forceColor) => {
2527

2628
test('Prints with colors if supported', testColor, true, '1');
2729
test('Prints without colors if not supported', testColor, false, '0');
30+
31+
test.serial('Prints lines in order when interleaved with subprocess stderr', async t => {
32+
const results = await Promise.all(Array.from({length: PARALLEL_COUNT}, () =>
33+
nestedSubprocess('noop-fd.js', ['2', `${foobarString}\n`], {verbose: 'full', stderr: 'inherit'}, {all: true}),
34+
));
35+
for (const {all} of results) {
36+
t.deepEqual(
37+
getNormalizedLines(all),
38+
[getCommandLine(all), foobarString, getCompletionLine(all)],
39+
);
40+
}
41+
});

0 commit comments

Comments
 (0)