mirror of
https://github.com/microsoft/playwright.git
synced 2025-01-07 11:46:42 +03:00
fix: collect stdio from forked processes (#27074)
A summary about the attempts we tried in https://github.com/microsoft/playwright/pull/26931 for https://github.com/microsoft/playwright/issues/26859 and https://github.com/microsoft/playwright/issues/24591. 1. attempt: collect all the stdio from the worker host. - this made it hard to capture the live console logs for the live trace 2. attempt: collect all the stdio from the worker host, but keep proxying `process.{stdout,stderr}.write`. - this solved the live console logs problem, since we could still listen on them - we found out about a race that the stdout/stderr streams can still buffer data while the test already had finished - we created a best effort flush implementation but didn't like the complexity it introduced to the project 3. attempt: keep things like they are but also listen to stdio from the worker host - this solves the race of console.log in the test (things stay like they are) - this keeps live trace working (things stay like they are) - a known pitfall is that it could result in logs which don't end up in the reporter onStdOut/onStdErr, but it is already a significant improvement over how it was before. - before: they were not getting displayed - after: they are getting displayed and there is a low likelihood that they don't end up in the reporter API if the write happens slightly before a test finished. Closes #26931 Fixes #26859 Fixes #24591
This commit is contained in:
parent
57e5183809
commit
4431a8366d
@ -14,6 +14,7 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import util from 'util';
|
||||||
import { serializeCompilationCache } from '../transform/compilationCache';
|
import { serializeCompilationCache } from '../transform/compilationCache';
|
||||||
import type { FullConfigInternal } from './config';
|
import type { FullConfigInternal } from './config';
|
||||||
import type { ReporterDescription, TestInfoError, TestStatus } from '../../types/test';
|
import type { ReporterDescription, TestInfoError, TestStatus } from '../../types/test';
|
||||||
@ -141,3 +142,11 @@ export function serializeConfig(config: FullConfigInternal): SerializedConfig {
|
|||||||
};
|
};
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function stdioChunkToParams(chunk: Uint8Array | string): TestOutputPayload {
|
||||||
|
if (chunk instanceof Uint8Array)
|
||||||
|
return { buffer: Buffer.from(chunk).toString('base64') };
|
||||||
|
if (typeof chunk !== 'string')
|
||||||
|
return { text: util.inspect(chunk) };
|
||||||
|
return { text: chunk };
|
||||||
|
}
|
||||||
|
@ -58,7 +58,7 @@ export class OutOfProcessLoaderHost {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async start() {
|
async start() {
|
||||||
await this._processHost.startRunner(serializeConfig(this._config), true);
|
await this._processHost.startRunner(serializeConfig(this._config));
|
||||||
}
|
}
|
||||||
|
|
||||||
async loadTestFile(file: string, testErrors: TestError[]): Promise<Suite> {
|
async loadTestFile(file: string, testErrors: TestError[]): Promise<Suite> {
|
||||||
|
@ -46,11 +46,16 @@ export class ProcessHost extends EventEmitter {
|
|||||||
this._extraEnv = env;
|
this._extraEnv = env;
|
||||||
}
|
}
|
||||||
|
|
||||||
async startRunner(runnerParams: any, inheritStdio: boolean) {
|
async startRunner(runnerParams: any, options: { onStdOut?: (chunk: Buffer | string) => void, onStdErr?: (chunk: Buffer | string) => void } = {}) {
|
||||||
this.process = child_process.fork(require.resolve('../common/process'), {
|
this.process = child_process.fork(require.resolve('../common/process'), {
|
||||||
detached: false,
|
detached: false,
|
||||||
env: { ...process.env, ...this._extraEnv },
|
env: { ...process.env, ...this._extraEnv },
|
||||||
stdio: inheritStdio ? ['ignore', 'inherit', 'inherit', 'ipc'] : ['ignore', 'ignore', process.env.PW_RUNNER_DEBUG ? 'inherit' : 'ignore', 'ipc'],
|
stdio: [
|
||||||
|
'ignore',
|
||||||
|
options.onStdOut ? 'pipe' : 'inherit',
|
||||||
|
(options.onStdErr && !process.env.PW_RUNNER_DEBUG) ? 'pipe' : 'inherit',
|
||||||
|
'ipc',
|
||||||
|
],
|
||||||
...(process.env.PW_TS_ESM_ON ? { execArgv: execArgvWithExperimentalLoaderOptions() } : {}),
|
...(process.env.PW_TS_ESM_ON ? { execArgv: execArgvWithExperimentalLoaderOptions() } : {}),
|
||||||
});
|
});
|
||||||
this.process.on('exit', (code, signal) => {
|
this.process.on('exit', (code, signal) => {
|
||||||
@ -84,6 +89,11 @@ export class ProcessHost extends EventEmitter {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (options.onStdOut)
|
||||||
|
this.process.stdout?.on('data', options.onStdOut);
|
||||||
|
if (options.onStdErr)
|
||||||
|
this.process.stderr?.on('data', options.onStdErr);
|
||||||
|
|
||||||
await new Promise<void>((resolve, reject) => {
|
await new Promise<void>((resolve, reject) => {
|
||||||
this.process!.once('exit', (code, signal) => reject(new Error(`process exited with code "${code}" and signal "${signal}" before it became ready`)));
|
this.process!.once('exit', (code, signal) => reject(new Error(`process exited with code "${code}" and signal "${signal}" before it became ready`)));
|
||||||
this.once('ready', () => resolve());
|
this.once('ready', () => resolve());
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
import fs from 'fs';
|
import fs from 'fs';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import type { TestGroup } from './testGroups';
|
import type { TestGroup } from './testGroups';
|
||||||
|
import { stdioChunkToParams } from '../common/ipc';
|
||||||
import type { RunPayload, SerializedConfig, WorkerInitParams } from '../common/ipc';
|
import type { RunPayload, SerializedConfig, WorkerInitParams } from '../common/ipc';
|
||||||
import { ProcessHost } from './processHost';
|
import { ProcessHost } from './processHost';
|
||||||
import { artifactsFolderName } from '../isomorphic/folders';
|
import { artifactsFolderName } from '../isomorphic/folders';
|
||||||
@ -54,7 +55,10 @@ export class WorkerHost extends ProcessHost {
|
|||||||
|
|
||||||
async start() {
|
async start() {
|
||||||
await fs.promises.mkdir(this._params.artifactsDir, { recursive: true });
|
await fs.promises.mkdir(this._params.artifactsDir, { recursive: true });
|
||||||
await this.startRunner(this._params, false);
|
await this.startRunner(this._params, {
|
||||||
|
onStdOut: chunk => this.emit('stdOut', stdioChunkToParams(chunk)),
|
||||||
|
onStdErr: chunk => this.emit('stdErr', stdioChunkToParams(chunk)),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
override async stop(didFail?: boolean) {
|
override async stop(didFail?: boolean) {
|
||||||
|
@ -15,9 +15,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { colors, rimraf } from 'playwright-core/lib/utilsBundle';
|
import { colors, rimraf } from 'playwright-core/lib/utilsBundle';
|
||||||
import util from 'util';
|
|
||||||
import { debugTest, formatLocation, relativeFilePath, serializeError } from '../util';
|
import { debugTest, formatLocation, relativeFilePath, serializeError } from '../util';
|
||||||
import type { TestBeginPayload, TestEndPayload, RunPayload, DonePayload, WorkerInitParams, TeardownErrorsPayload, TestOutputPayload } from '../common/ipc';
|
import { type TestBeginPayload, type TestEndPayload, type RunPayload, type DonePayload, type WorkerInitParams, type TeardownErrorsPayload, stdioChunkToParams } from '../common/ipc';
|
||||||
import { setCurrentTestInfo, setIsWorkerProcess } from '../common/globals';
|
import { setCurrentTestInfo, setIsWorkerProcess } from '../common/globals';
|
||||||
import { ConfigLoader } from '../common/configLoader';
|
import { ConfigLoader } from '../common/configLoader';
|
||||||
import type { Suite, TestCase } from '../common/test';
|
import type { Suite, TestCase } from '../common/test';
|
||||||
@ -76,20 +75,14 @@ export class WorkerMain extends ProcessRunner {
|
|||||||
process.on('unhandledRejection', reason => this.unhandledError(reason));
|
process.on('unhandledRejection', reason => this.unhandledError(reason));
|
||||||
process.on('uncaughtException', error => this.unhandledError(error));
|
process.on('uncaughtException', error => this.unhandledError(error));
|
||||||
process.stdout.write = (chunk: string | Buffer) => {
|
process.stdout.write = (chunk: string | Buffer) => {
|
||||||
const outPayload: TestOutputPayload = {
|
this.dispatchEvent('stdOut', stdioChunkToParams(chunk));
|
||||||
...chunkToParams(chunk)
|
|
||||||
};
|
|
||||||
this.dispatchEvent('stdOut', outPayload);
|
|
||||||
this._currentTest?._tracing.appendStdioToTrace('stdout', chunk);
|
this._currentTest?._tracing.appendStdioToTrace('stdout', chunk);
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!process.env.PW_RUNNER_DEBUG) {
|
if (!process.env.PW_RUNNER_DEBUG) {
|
||||||
process.stderr.write = (chunk: string | Buffer) => {
|
process.stderr.write = (chunk: string | Buffer) => {
|
||||||
const outPayload: TestOutputPayload = {
|
this.dispatchEvent('stdErr', stdioChunkToParams(chunk));
|
||||||
...chunkToParams(chunk)
|
|
||||||
};
|
|
||||||
this.dispatchEvent('stdErr', outPayload);
|
|
||||||
this._currentTest?._tracing.appendStdioToTrace('stderr', chunk);
|
this._currentTest?._tracing.appendStdioToTrace('stderr', chunk);
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
@ -652,12 +645,4 @@ function formatTestTitle(test: TestCase, projectName: string) {
|
|||||||
return `${projectTitle}${location} › ${titles.join(' › ')}`;
|
return `${projectTitle}${location} › ${titles.join(' › ')}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
function chunkToParams(chunk: Uint8Array | string, encoding?: BufferEncoding): { text?: string, buffer?: string } {
|
|
||||||
if (chunk instanceof Uint8Array)
|
|
||||||
return { buffer: Buffer.from(chunk).toString('base64') };
|
|
||||||
if (typeof chunk !== 'string')
|
|
||||||
return { text: util.inspect(chunk) };
|
|
||||||
return { text: chunk };
|
|
||||||
}
|
|
||||||
|
|
||||||
export const create = (params: WorkerInitParams) => new WorkerMain(params);
|
export const create = (params: WorkerInitParams) => new WorkerMain(params);
|
||||||
|
@ -40,6 +40,30 @@ test('should collect stdio', async ({ runInlineTest }) => {
|
|||||||
expect(stderr).toEqual([{ text: 'stderr text' }, { buffer: Buffer.from('stderr buffer').toString('base64') }]);
|
expect(stderr).toEqual([{ text: 'stderr text' }, { buffer: Buffer.from('stderr buffer').toString('base64') }]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('should collect stdio from forked process', async ({ runInlineTest }) => {
|
||||||
|
const { exitCode, report } = await runInlineTest({
|
||||||
|
'stdio.spec.js': `
|
||||||
|
import { test } from '@playwright/test';
|
||||||
|
import { fork } from 'child_process';
|
||||||
|
test('stdio', async () => {
|
||||||
|
const child = fork('fork.js');
|
||||||
|
await new Promise((resolve) => child.on('exit', (code) => resolve(code)));
|
||||||
|
});
|
||||||
|
`,
|
||||||
|
'fork.js': `
|
||||||
|
process.stdout.write('stdout text');
|
||||||
|
process.stdout.write(Buffer.from('stdout buffer'));
|
||||||
|
process.stderr.write('stderr text');
|
||||||
|
process.stderr.write(Buffer.from('stderr buffer'));
|
||||||
|
`
|
||||||
|
});
|
||||||
|
expect(exitCode).toBe(0);
|
||||||
|
const testResult = report.suites[0].specs[0].tests[0].results[0];
|
||||||
|
const { stdout, stderr } = testResult;
|
||||||
|
expect(stdout.map(e => Buffer.from((e as any).buffer, 'base64').toString()).join('')).toEqual('stdout textstdout buffer');
|
||||||
|
expect(stderr.map(e => Buffer.from((e as any).buffer, 'base64').toString()).join('')).toEqual('stderr textstderr buffer');
|
||||||
|
});
|
||||||
|
|
||||||
test('should work with not defined errors', async ({ runInlineTest }) => {
|
test('should work with not defined errors', async ({ runInlineTest }) => {
|
||||||
const result = await runInlineTest({
|
const result = await runInlineTest({
|
||||||
'is-not-defined-error.spec.ts': `
|
'is-not-defined-error.spec.ts': `
|
||||||
|
Loading…
Reference in New Issue
Block a user