chore: prepare to reuse test server from ui mode (2) (#29966)

This commit is contained in:
Pavel Feldman 2024-03-18 09:59:02 -07:00 committed by GitHub
parent 6faadf5160
commit ef4438ee99
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 132 additions and 92 deletions

View File

@ -14,12 +14,24 @@
* limitations under the License.
*/
import type { TestError } from '../../types/testReporter';
import type * as reporterTypes from '../../types/testReporter';
export interface TestServerInterface {
listFiles(params: {
configFile: string;
}): Promise<{
ping(): Promise<void>;
watch(params: {
fileNames: string[];
}): Promise<void>;
open(params: { location: reporterTypes.Location }): Promise<void>;
resizeTerminal(params: { cols: number, rows: number }): Promise<void>;
checkBrowsers(): Promise<{ hasBrowsers: boolean }>;
installBrowsers(): Promise<void>;
listFiles(): Promise<{
projects: {
name: string;
testDir: string;
@ -27,41 +39,40 @@ export interface TestServerInterface {
files: string[];
}[];
cliEntryPoint?: string;
error?: TestError;
error?: reporterTypes.TestError;
}>;
listTests(params: {
configFile: string;
locations: string[];
reporter: string;
reporter?: string;
fileNames?: string[];
}): Promise<void>;
test(params: {
configFile: string;
locations: string[];
reporter: string;
runTests(params: {
reporter?: string;
locations?: string[];
grep?: string;
testIds?: string[];
headed?: boolean;
oneWorker?: boolean;
trace?: 'on' | 'off';
projects?: string[];
grep?: string;
reuseContext?: boolean;
connectWsEndpoint?: string;
}): Promise<void>;
findRelatedTestFiles(params: {
configFile: string;
files: string[];
}): Promise<{ testFiles: string[]; errors?: TestError[]; }>;
}): Promise<{ testFiles: string[]; errors?: reporterTypes.TestError[]; }>;
stop(params: {
configFile: string;
}): Promise<void>;
stop(): Promise<void>;
closeGracefully(): Promise<void>;
}
export interface TestServerEvents {
on(event: 'report', listener: (params: any) => void): void;
on(event: 'listReport', listener: (params: any) => void): void;
on(event: 'testReport', listener: (params: any) => void): void;
on(event: 'stdio', listener: (params: { type: 'stdout' | 'stderr', text?: string, buffer?: string }) => void): void;
on(event: 'listChanged', listener: () => void): void;
on(event: 'testFilesChanged', listener: (testFileNames: string[]) => void): void;
}

View File

@ -15,9 +15,9 @@
*/
import { registry, startTraceViewerServer } from 'playwright-core/lib/server';
import { ManualPromise } from 'playwright-core/lib/utils';
import { ManualPromise, gracefullyProcessExitDoNotHang } from 'playwright-core/lib/utils';
import type { Transport, HttpServer } from 'playwright-core/lib/utils';
import type { FullResult } from '../../types/testReporter';
import type { FullResult, Location, TestError } from '../../types/testReporter';
import { collectAffectedTestFiles, dependenciesForTestFile } from '../transform/compilationCache';
import type { FullConfigInternal } from '../common/config';
import { InternalReporter } from '../reporters/internalReporter';
@ -29,14 +29,15 @@ import ListReporter from '../reporters/list';
import { Multiplexer } from '../reporters/multiplexer';
import { SigIntWatcher } from './sigIntWatcher';
import { Watcher } from '../fsWatcher';
import type { TestServerInterface } from './testServerInterface';
import { Runner } from './runner';
import { serializeError } from '../util';
import { prepareErrorStack } from '../reporters/base';
class TestServer {
private _config: FullConfigInternal;
private _transport: Transport | undefined;
private _testRun: { run: Promise<FullResult['status']>, stop: ManualPromise<void> } | undefined;
private _dispatcher: TestServerDispatcher | undefined;
globalCleanup: (() => Promise<FullResult['status']>) | undefined;
private _globalWatcher: Watcher;
private _testWatcher: Watcher;
private _originalStdoutWrite: NodeJS.WriteStream['write'];
private _originalStderrWrite: NodeJS.WriteStream['write'];
@ -56,12 +57,6 @@ class TestServer {
this._originalStdoutWrite = process.stdout.write;
this._originalStderrWrite = process.stderr.write;
this._globalWatcher = new Watcher('deep', () => this._dispatchEvent('listChanged', {}));
this._testWatcher = new Watcher('flat', events => {
const collector = new Set<string>();
events.forEach(f => collectAffectedTestFiles(f.file, collector));
this._dispatchEvent('testFilesChanged', { testFileNames: [...collector] });
});
}
async runGlobalSetup(): Promise<FullResult['status']> {
@ -81,56 +76,18 @@ class TestServer {
}
async start(options: { host?: string, port?: number }): Promise<HttpServer> {
let queue = Promise.resolve();
const transport: Transport = {
dispatch: async (method, params) => {
if (method === 'ping')
return;
if (method === 'watch') {
this._watchFiles(params.fileNames);
return;
}
if (method === 'open' && params.location) {
open('vscode://file/' + params.location).catch(e => this._originalStderrWrite.call(process.stderr, String(e)));
return;
}
if (method === 'resizeTerminal') {
process.stdout.columns = params.cols;
process.stdout.rows = params.rows;
process.stderr.columns = params.cols;
process.stderr.columns = params.rows;
return;
}
if (method === 'stop') {
void this._stopTests();
return;
}
if (method === 'checkBrowsers')
return { hasBrowsers: hasSomeBrowsers() };
if (method === 'installBrowsers') {
await installBrowsers();
return;
}
queue = queue.then(() => this._queueListOrRun(method, params));
await queue;
},
onclose: () => {},
};
this._transport = transport;
return await startTraceViewerServer({ ...options, transport });
this._dispatcher = new TestServerDispatcher(this._config);
return await startTraceViewerServer({ ...options, transport: this._dispatcher.transport });
}
wireStdIO() {
if (!process.env.PWTEST_DEBUG) {
process.stdout.write = (chunk: string | Buffer) => {
this._dispatchEvent('stdio', chunkToPayload('stdout', chunk));
this._dispatcher?._dispatchEvent('stdio', chunkToPayload('stdout', chunk));
return true;
};
process.stderr.write = (chunk: string | Buffer) => {
this._dispatchEvent('stdio', chunkToPayload('stderr', chunk));
this._dispatcher?._dispatchEvent('stdio', chunkToPayload('stderr', chunk));
return true;
};
}
@ -142,20 +99,71 @@ class TestServer {
process.stderr.write = this._originalStderrWrite;
}
}
private async _queueListOrRun(method: string, params: any) {
if (method === 'list')
await this._listTests();
if (method === 'run')
await this._runTests(params.testIds, params.projects);
}
private _dispatchEvent(method: string, params?: any) {
this._transport?.sendEvent?.(method, params);
class TestServerDispatcher implements TestServerInterface {
private _config: FullConfigInternal;
private _globalWatcher: Watcher;
private _testWatcher: Watcher;
private _testRun: { run: Promise<FullResult['status']>, stop: ManualPromise<void> } | undefined;
readonly transport: Transport;
private _queue = Promise.resolve();
constructor(config: FullConfigInternal) {
this._config = config;
this.transport = {
dispatch: (method, params) => (this as any)[method](params),
onclose: () => {},
};
this._globalWatcher = new Watcher('deep', () => this._dispatchEvent('listChanged', {}));
this._testWatcher = new Watcher('flat', events => {
const collector = new Set<string>();
events.forEach(f => collectAffectedTestFiles(f.file, collector));
this._dispatchEvent('testFilesChanged', { testFileNames: [...collector] });
});
}
private async _listTests() {
async ping() {}
async open(params: { location: Location }) {
// eslint-disable-next-line no-console
open('vscode://file/' + params.location.file + ':' + params.location.column).catch(e => console.error(e));
}
async resizeTerminal(params: { cols: number; rows: number; }) {
process.stdout.columns = params.cols;
process.stdout.rows = params.rows;
process.stderr.columns = params.cols;
process.stderr.columns = params.rows;
}
async checkBrowsers(): Promise<{ hasBrowsers: boolean; }> {
return { hasBrowsers: hasSomeBrowsers() };
}
async installBrowsers() {
await installBrowsers();
}
async listFiles() {
try {
const runner = new Runner(this._config);
return runner.listTestFiles();
} catch (e) {
const error: TestError = serializeError(e);
error.location = prepareErrorStack(e.stack).location;
return { projects: [], error };
}
}
async listTests(params: { reporter?: string; fileNames: string[]; }) {
this._queue = this._queue.then(() => this._innerListTests(params));
await this._queue;
}
private async _innerListTests(params: { reporter?: string; fileNames?: string[]; }) {
const reporter = new InternalReporter(new TeleReporterEmitter(e => this._dispatchEvent('listReport', e), { omitBuffers: true }));
this._config.cliArgs = params.fileNames || [];
this._config.cliListOnly = true;
this._config.testIdMatcher = undefined;
const taskRunner = createTaskRunnerForList(this._config, reporter, 'out-of-process', { failOnLoadErrors: false });
@ -174,12 +182,20 @@ class TestServer {
this._globalWatcher.update([...projectDirs], [...projectOutputs], false);
}
private async _runTests(testIds: string[], projects: string[]) {
await this._stopTests();
async runTests(params: { reporter?: string; locations?: string[] | undefined; grep?: string | undefined; testIds?: string[] | undefined; headed?: boolean | undefined; oneWorker?: boolean | undefined; trace?: 'off' | 'on' | undefined; projects?: string[] | undefined; reuseContext?: boolean | undefined; connectWsEndpoint?: string | undefined; }) {
this._queue = this._queue.then(() => this._innerRunTests(params));
await this._queue;
}
private async _innerRunTests(params: { reporter?: string; locations?: string[] | undefined; grep?: string | undefined; testIds?: string[] | undefined; headed?: boolean | undefined; oneWorker?: boolean | undefined; trace?: 'off' | 'on' | undefined; projects?: string[] | undefined; reuseContext?: boolean | undefined; connectWsEndpoint?: string | undefined; }) {
await this.stop();
const { testIds, projects, locations, grep } = params;
const testIdSet = testIds ? new Set<string>(testIds) : null;
this._config.cliArgs = locations ? locations : [];
this._config.cliGrep = grep;
this._config.cliListOnly = false;
this._config.cliProjectFilter = projects.length ? projects : undefined;
this._config.cliProjectFilter = projects?.length ? projects : undefined;
this._config.testIdMatcher = id => !testIdSet || testIdSet.has(id);
const reporters = await createReporters(this._config, 'ui');
@ -200,19 +216,32 @@ class TestServer {
await run;
}
private _watchFiles(fileNames: string[]) {
async watch(params: { fileNames: string[]; }) {
const files = new Set<string>();
for (const fileName of fileNames) {
for (const fileName of params.fileNames) {
files.add(fileName);
dependenciesForTestFile(fileName).forEach(file => files.add(file));
}
this._testWatcher.update([...files], [], true);
}
private async _stopTests() {
findRelatedTestFiles(params: { files: string[]; }): Promise<{ testFiles: string[]; errors?: TestError[] | undefined; }> {
const runner = new Runner(this._config);
return runner.findRelatedTestFiles('out-of-process', params.files);
}
async stop() {
this._testRun?.stop?.resolve();
await this._testRun?.run;
}
async closeGracefully() {
gracefullyProcessExitDoNotHang(0);
}
_dispatchEvent(method: string, params?: any) {
this.transport.sendEvent?.(method, params);
}
}
export async function runTestServer(config: FullConfigInternal, options: { host?: string, port?: number }, openUI: (server: HttpServer, cancelPromise: ManualPromise<void>) => Promise<void>): Promise<FullResult['status']> {

View File

@ -166,7 +166,7 @@ export const UIModeView: React.FC<{}> = ({
setProgress({ total: 0, passed: 0, failed: 0, skipped: 0 });
setRunningState({ testIds });
await sendMessage('run', { testIds: [...testIds], projects: [...projectFilters].filter(([_, v]) => v).map(([p]) => p) });
await sendMessage('runTests', { testIds: [...testIds], projects: [...projectFilters].filter(([_, v]) => v).map(([p]) => p) });
// Clear pending tests in case of interrupt.
for (const test of testModel.rootSuite?.allTests() || []) {
if (test.results[0]?.duration === -1)
@ -625,7 +625,7 @@ const refreshRootSuite = (): Promise<void> => {
},
pathSeparator,
});
return sendMessage('list', {});
return sendMessage('listTests', {});
};
const sendMessageNoReply = (method: string, params?: any) => {
@ -641,7 +641,7 @@ const sendMessageNoReply = (method: string, params?: any) => {
const dispatchEvent = (method: string, params?: any) => {
if (method === 'listChanged') {
sendMessage('list', {}).catch(() => {});
sendMessage('listTests', {}).catch(() => {});
return;
}

View File

@ -286,6 +286,6 @@ test('should not watch output', async ({ runUITest }) => {
await page.getByTitle('Run all').click();
await expect(page.getByTestId('status-line')).toHaveText('1/1 passed (100%)');
expect(commands).toContain('run');
expect(commands).not.toContain('list');
expect(commands).toContain('runTests');
expect(commands).not.toContain('listTests');
});