2021-06-06 17:09:53 -07:00
|
|
|
/**
|
|
|
|
* Copyright Microsoft Corporation. All rights reserved.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { Console } from 'console';
|
|
|
|
import * as util from 'util';
|
2022-02-23 12:32:12 -08:00
|
|
|
import { RunPayload, TeardownErrorsPayload, TestOutputPayload, WorkerInitParams } from './ipc';
|
2021-06-21 14:49:43 -07:00
|
|
|
import { startProfiling, stopProfiling } from './profiler';
|
2021-06-06 17:09:53 -07:00
|
|
|
import { serializeError } from './util';
|
|
|
|
import { WorkerRunner } from './workerRunner';
|
|
|
|
|
|
|
|
let closed = false;
|
|
|
|
|
|
|
|
sendMessageToParent('ready');
|
|
|
|
|
|
|
|
global.console = new Console({
|
|
|
|
stdout: process.stdout,
|
|
|
|
stderr: process.stderr,
|
|
|
|
colorMode: process.env.FORCE_COLOR === '1',
|
|
|
|
});
|
|
|
|
|
|
|
|
process.stdout.write = (chunk: string | Buffer) => {
|
|
|
|
const outPayload: TestOutputPayload = {
|
2022-01-28 17:39:42 -08:00
|
|
|
testId: workerRunner?._currentTest?._test._id,
|
2021-06-06 17:09:53 -07:00
|
|
|
...chunkToParams(chunk)
|
|
|
|
};
|
|
|
|
sendMessageToParent('stdOut', outPayload);
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!process.env.PW_RUNNER_DEBUG) {
|
|
|
|
process.stderr.write = (chunk: string | Buffer) => {
|
|
|
|
const outPayload: TestOutputPayload = {
|
2022-01-28 17:39:42 -08:00
|
|
|
testId: workerRunner?._currentTest?._test._id,
|
2021-06-06 17:09:53 -07:00
|
|
|
...chunkToParams(chunk)
|
|
|
|
};
|
|
|
|
sendMessageToParent('stdErr', outPayload);
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
process.on('disconnect', gracefullyCloseAndExit);
|
|
|
|
process.on('SIGINT',() => {});
|
|
|
|
process.on('SIGTERM',() => {});
|
|
|
|
|
|
|
|
let workerRunner: WorkerRunner;
|
2021-06-21 14:49:43 -07:00
|
|
|
let workerIndex: number | undefined;
|
2021-06-06 17:09:53 -07:00
|
|
|
|
|
|
|
process.on('unhandledRejection', (reason, promise) => {
|
|
|
|
if (workerRunner)
|
|
|
|
workerRunner.unhandledError(reason);
|
|
|
|
});
|
|
|
|
|
|
|
|
process.on('uncaughtException', error => {
|
|
|
|
if (workerRunner)
|
|
|
|
workerRunner.unhandledError(error);
|
|
|
|
});
|
|
|
|
|
|
|
|
process.on('message', async message => {
|
|
|
|
if (message.method === 'init') {
|
|
|
|
const initParams = message.params as WorkerInitParams;
|
2021-06-21 14:49:43 -07:00
|
|
|
workerIndex = initParams.workerIndex;
|
|
|
|
startProfiling();
|
2021-06-06 17:09:53 -07:00
|
|
|
workerRunner = new WorkerRunner(initParams);
|
2022-02-23 12:32:12 -08:00
|
|
|
for (const event of ['testBegin', 'testEnd', 'stepBegin', 'stepEnd', 'done', 'teardownErrors'])
|
2021-06-06 17:09:53 -07:00
|
|
|
workerRunner.on(event, sendMessageToParent.bind(null, event));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (message.method === 'stop') {
|
|
|
|
await gracefullyCloseAndExit();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (message.method === 'run') {
|
|
|
|
const runPayload = message.params as RunPayload;
|
2022-03-01 09:11:17 -08:00
|
|
|
await workerRunner!.run(runPayload);
|
2021-06-06 17:09:53 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
async function gracefullyCloseAndExit() {
|
|
|
|
if (closed)
|
|
|
|
return;
|
|
|
|
closed = true;
|
|
|
|
// Force exit after 30 seconds.
|
|
|
|
setTimeout(() => process.exit(0), 30000);
|
|
|
|
// Meanwhile, try to gracefully shutdown.
|
|
|
|
try {
|
|
|
|
if (workerRunner) {
|
2021-08-10 10:54:05 -07:00
|
|
|
await workerRunner.stop();
|
2021-06-06 17:09:53 -07:00
|
|
|
await workerRunner.cleanup();
|
|
|
|
}
|
2021-06-21 14:49:43 -07:00
|
|
|
if (workerIndex !== undefined)
|
|
|
|
await stopProfiling(workerIndex);
|
2021-06-06 17:09:53 -07:00
|
|
|
} catch (e) {
|
2022-02-23 12:32:12 -08:00
|
|
|
const payload: TeardownErrorsPayload = { fatalErrors: [serializeError(e)] };
|
|
|
|
process.send!({ method: 'teardownErrors', params: payload });
|
2021-06-06 17:09:53 -07:00
|
|
|
}
|
|
|
|
process.exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
function sendMessageToParent(method: string, params = {}) {
|
|
|
|
try {
|
|
|
|
process.send!({ method, params });
|
|
|
|
} catch (e) {
|
|
|
|
// Can throw when closing.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function chunkToParams(chunk: Buffer | string): { text?: string, buffer?: string } {
|
|
|
|
if (chunk instanceof Buffer)
|
|
|
|
return { buffer: chunk.toString('base64') };
|
|
|
|
if (typeof chunk !== 'string')
|
|
|
|
return { text: util.inspect(chunk) };
|
|
|
|
return { text: chunk };
|
|
|
|
}
|