2020-09-29 13:48:24 -07:00
|
|
|
/**
|
|
|
|
* Copyright (c) Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2020-12-28 15:44:24 -08:00
|
|
|
/* eslint-disable no-console */
|
2020-09-29 13:48:24 -07:00
|
|
|
|
2021-02-11 06:36:15 -08:00
|
|
|
import fs from 'fs';
|
2021-03-04 10:32:06 -08:00
|
|
|
import * as playwright from '../..';
|
2022-04-06 13:57:14 -08:00
|
|
|
import type { BrowserType } from '../client/browserType';
|
|
|
|
import type { LaunchServerOptions } from '../client/types';
|
2022-04-06 21:21:27 -08:00
|
|
|
import { createPlaywright, DispatcherConnection, Root, PlaywrightDispatcher } from '../server';
|
2022-08-05 19:34:57 -07:00
|
|
|
import type { Playwright } from '../server';
|
2022-02-02 21:26:45 -08:00
|
|
|
import { IpcTransport, PipeTransport } from '../protocol/transport';
|
2021-08-19 15:16:46 -07:00
|
|
|
import { PlaywrightServer } from '../remote/playwrightServer';
|
2021-07-07 21:14:16 +02:00
|
|
|
import { gracefullyCloseAll } from '../utils/processLauncher';
|
2022-08-05 19:34:57 -07:00
|
|
|
import { Recorder } from '../server/recorder';
|
|
|
|
import { EmptyRecorderApp } from '../server/recorder/recorderApp';
|
|
|
|
import type { BrowserContext } from '../server/browserContext';
|
2022-08-08 10:39:54 -07:00
|
|
|
import { serverSideCallMetadata } from '../server/instrumentation';
|
2022-08-08 17:16:13 -07:00
|
|
|
import type { Mode } from '../server/recorder/recorderTypes';
|
2020-09-29 13:48:24 -07:00
|
|
|
|
2020-12-28 15:44:24 -08:00
|
|
|
export function printApiJson() {
|
2021-06-02 14:01:05 -07:00
|
|
|
// Note: this file is generated by build-playwright-driver.sh
|
2020-12-28 15:44:24 -08:00
|
|
|
console.log(JSON.stringify(require('../../api.json')));
|
2020-09-29 13:48:24 -07:00
|
|
|
}
|
|
|
|
|
2021-04-12 11:14:54 -07:00
|
|
|
export function runDriver() {
|
2020-09-29 13:48:24 -07:00
|
|
|
const dispatcherConnection = new DispatcherConnection();
|
2021-08-20 21:32:21 +02:00
|
|
|
new Root(dispatcherConnection, async (rootScope, { sdkLanguage }) => {
|
|
|
|
const playwright = createPlaywright(sdkLanguage);
|
2021-08-19 17:31:14 +02:00
|
|
|
return new PlaywrightDispatcher(rootScope, playwright);
|
|
|
|
});
|
2022-02-02 21:26:45 -08:00
|
|
|
const transport = process.send ? new IpcTransport(process) : new PipeTransport(process.stdout, process.stdin);
|
2020-12-16 14:21:59 -08:00
|
|
|
transport.onmessage = message => dispatcherConnection.dispatch(JSON.parse(message));
|
|
|
|
dispatcherConnection.onmessage = message => transport.send(JSON.stringify(message));
|
2022-08-02 21:09:39 -07:00
|
|
|
transport.onclose = () => {
|
2020-12-16 14:21:59 -08:00
|
|
|
// Drop any messages during shutdown on the floor.
|
|
|
|
dispatcherConnection.onmessage = () => {};
|
2022-08-02 21:09:39 -07:00
|
|
|
selfDestruct();
|
2020-09-29 13:48:24 -07:00
|
|
|
};
|
|
|
|
}
|
2020-09-29 18:00:56 -07:00
|
|
|
|
2022-07-31 14:31:17 -07:00
|
|
|
export async function runServer(port: number | undefined, path = '/', maxClients = Infinity, enableSocksProxy = true, reuseBrowser = false) {
|
2022-08-03 19:37:06 -07:00
|
|
|
const maxIncomingConnections = maxClients;
|
|
|
|
const maxConcurrentConnections = reuseBrowser ? 1 : maxClients;
|
|
|
|
const server = new PlaywrightServer(reuseBrowser ? 'reuse-browser' : 'auto', { path, maxIncomingConnections, maxConcurrentConnections, enableSocksProxy });
|
2021-07-19 07:53:12 +02:00
|
|
|
const wsEndpoint = await server.listen(port);
|
|
|
|
process.on('exit', () => server.close().catch(console.error));
|
2021-04-12 11:14:54 -07:00
|
|
|
console.log('Listening on ' + wsEndpoint); // eslint-disable-line no-console
|
2022-08-02 21:09:39 -07:00
|
|
|
process.stdin.on('close', () => selfDestruct());
|
2022-08-05 19:34:57 -07:00
|
|
|
if (process.send && server.preLaunchedPlaywright())
|
|
|
|
wireController(server.preLaunchedPlaywright()!, wsEndpoint);
|
2021-04-12 11:14:54 -07:00
|
|
|
}
|
|
|
|
|
2021-03-04 10:32:06 -08:00
|
|
|
export async function launchBrowserServer(browserName: string, configFile?: string) {
|
|
|
|
let options: LaunchServerOptions = {};
|
|
|
|
if (configFile)
|
|
|
|
options = JSON.parse(fs.readFileSync(configFile).toString());
|
|
|
|
const browserType = (playwright as any)[browserName] as BrowserType;
|
|
|
|
const server = await browserType.launchServer(options);
|
|
|
|
console.log(server.wsEndpoint());
|
|
|
|
}
|
2022-08-02 21:09:39 -07:00
|
|
|
|
|
|
|
function selfDestruct() {
|
|
|
|
// Force exit after 30 seconds.
|
|
|
|
setTimeout(() => process.exit(0), 30000);
|
|
|
|
// Meanwhile, try to gracefully close all browsers.
|
|
|
|
gracefullyCloseAll().then(() => {
|
|
|
|
process.exit(0);
|
|
|
|
});
|
|
|
|
}
|
2022-08-05 19:34:57 -07:00
|
|
|
|
2022-08-08 10:39:54 -07:00
|
|
|
const internalMetadata = serverSideCallMetadata();
|
|
|
|
|
2022-08-08 17:16:13 -07:00
|
|
|
class ProtocolHandler {
|
|
|
|
private _playwright: Playwright;
|
|
|
|
private _autoCloseTimer: NodeJS.Timeout | undefined;
|
2022-08-05 19:34:57 -07:00
|
|
|
|
2022-08-08 17:16:13 -07:00
|
|
|
constructor(playwright: Playwright) {
|
|
|
|
this._playwright = playwright;
|
2022-08-11 13:42:16 -07:00
|
|
|
playwright.instrumentation.addListener({
|
|
|
|
onPageOpen: () => this._sendSnapshot(),
|
|
|
|
onPageNavigated: () => this._sendSnapshot(),
|
|
|
|
onPageClose: () => this._sendSnapshot(),
|
|
|
|
}, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _sendSnapshot() {
|
|
|
|
const browsers = [];
|
|
|
|
for (const browser of this._playwright.allBrowsers()) {
|
|
|
|
const b = {
|
|
|
|
name: browser.options.name,
|
|
|
|
guid: browser.guid,
|
|
|
|
contexts: [] as any[]
|
|
|
|
};
|
|
|
|
browsers.push(b);
|
|
|
|
for (const context of browser.contexts()) {
|
|
|
|
const c = {
|
|
|
|
guid: context.guid,
|
|
|
|
pages: [] as any[]
|
|
|
|
};
|
|
|
|
b.contexts.push(c);
|
|
|
|
for (const page of context.pages()) {
|
|
|
|
const p = {
|
|
|
|
guid: page.guid,
|
|
|
|
url: page.mainFrame().url()
|
|
|
|
};
|
|
|
|
c.pages.push(p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
process.send!({ method: 'browsersChanged', params: { browsers } });
|
2022-08-08 17:16:13 -07:00
|
|
|
}
|
2022-08-05 19:34:57 -07:00
|
|
|
|
2022-08-10 10:57:28 -07:00
|
|
|
async resetForReuse() {
|
|
|
|
const contexts = new Set<BrowserContext>();
|
|
|
|
for (const page of this._playwright.allPages())
|
|
|
|
contexts.add(page.context());
|
|
|
|
for (const context of contexts)
|
|
|
|
await context.resetForReuse(internalMetadata, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
async navigate(params: { url: string }) {
|
|
|
|
for (const p of this._playwright.allPages())
|
|
|
|
await p.mainFrame().goto(internalMetadata, params.url);
|
|
|
|
}
|
|
|
|
|
2022-08-08 17:16:13 -07:00
|
|
|
async setMode(params: { mode: Mode, language?: string, file?: string }) {
|
|
|
|
await gc(this._playwright);
|
|
|
|
|
|
|
|
if (params.mode === 'none') {
|
|
|
|
for (const recorder of await allRecorders(this._playwright)) {
|
|
|
|
recorder.setHighlightedSelector('');
|
|
|
|
recorder.setMode('none');
|
2022-08-05 19:34:57 -07:00
|
|
|
}
|
2022-08-08 17:16:13 -07:00
|
|
|
this.setAutoClose({ enabled: true });
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const browsers = this._playwright.allBrowsers();
|
|
|
|
if (!browsers.length)
|
|
|
|
await this._playwright.chromium.launch(internalMetadata, { headless: false });
|
|
|
|
// Create page if none.
|
|
|
|
const pages = this._playwright.allPages();
|
|
|
|
if (!pages.length) {
|
|
|
|
const [browser] = this._playwright.allBrowsers();
|
|
|
|
const { context } = await browser.newContextForReuse({}, internalMetadata);
|
|
|
|
await context.newPage(internalMetadata);
|
|
|
|
}
|
|
|
|
// Toggle the mode.
|
|
|
|
for (const recorder of await allRecorders(this._playwright)) {
|
|
|
|
recorder.setHighlightedSelector('');
|
|
|
|
if (params.mode === 'recording')
|
|
|
|
recorder.setOutput(params.language!, params.file);
|
|
|
|
recorder.setMode(params.mode);
|
|
|
|
}
|
|
|
|
this.setAutoClose({ enabled: true });
|
|
|
|
}
|
2022-08-05 19:34:57 -07:00
|
|
|
|
2022-08-08 17:16:13 -07:00
|
|
|
async setAutoClose(params: { enabled: boolean }) {
|
|
|
|
if (this._autoCloseTimer)
|
|
|
|
clearTimeout(this._autoCloseTimer);
|
|
|
|
if (!params.enabled)
|
|
|
|
return;
|
|
|
|
const heartBeat = () => {
|
|
|
|
if (!this._playwright.allPages().length)
|
|
|
|
selfDestruct();
|
|
|
|
else
|
|
|
|
this._autoCloseTimer = setTimeout(heartBeat, 5000);
|
|
|
|
};
|
|
|
|
this._autoCloseTimer = setTimeout(heartBeat, 30000);
|
|
|
|
}
|
|
|
|
|
|
|
|
async highlight(params: { selector: string }) {
|
|
|
|
for (const recorder of await allRecorders(this._playwright))
|
|
|
|
recorder.setHighlightedSelector(params.selector);
|
|
|
|
}
|
|
|
|
|
2022-08-09 16:42:55 -07:00
|
|
|
async hideHighlight() {
|
|
|
|
await this._playwright.hideHighlight();
|
|
|
|
}
|
|
|
|
|
2022-08-08 17:16:13 -07:00
|
|
|
async kill() {
|
|
|
|
selfDestruct();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function wireController(playwright: Playwright, wsEndpoint: string) {
|
|
|
|
process.send!({ method: 'ready', params: { wsEndpoint } });
|
|
|
|
const handler = new ProtocolHandler(playwright);
|
|
|
|
process.on('message', async message => {
|
|
|
|
try {
|
|
|
|
const result = await (handler as any)[message.method](message.params);
|
|
|
|
process.send!({ id: message.id, result });
|
2022-08-05 19:34:57 -07:00
|
|
|
} catch (e) {
|
2022-08-08 17:16:13 -07:00
|
|
|
process.send!({ id: message.id, error: e.toString() });
|
2022-08-05 19:34:57 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-08-08 17:16:13 -07:00
|
|
|
async function gc(playwright: Playwright) {
|
|
|
|
for (const browser of playwright.allBrowsers()) {
|
|
|
|
for (const context of browser.contexts()) {
|
|
|
|
if (!context.pages().length)
|
|
|
|
await context.close(serverSideCallMetadata());
|
|
|
|
}
|
|
|
|
if (!browser.contexts())
|
|
|
|
await browser.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-05 19:34:57 -07:00
|
|
|
async function allRecorders(playwright: Playwright): Promise<Recorder[]> {
|
|
|
|
const contexts = new Set<BrowserContext>();
|
|
|
|
for (const page of playwright.allPages())
|
|
|
|
contexts.add(page.context());
|
2022-08-09 16:42:55 -07:00
|
|
|
const result = await Promise.all([...contexts].map(c => Recorder.show(c, { omitCallTracking: true }, () => Promise.resolve(new InspectingRecorderApp()))));
|
2022-08-05 19:34:57 -07:00
|
|
|
return result.filter(Boolean) as Recorder[];
|
|
|
|
}
|
|
|
|
|
|
|
|
class InspectingRecorderApp extends EmptyRecorderApp {
|
|
|
|
override async setSelector(selector: string): Promise<void> {
|
|
|
|
process.send!({ method: 'inspectRequested', params: { selector } });
|
|
|
|
}
|
|
|
|
}
|