2020-01-24 14:49:47 -08: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-05-22 07:03:42 -07:00
|
|
|
import * as fs from 'fs';
|
|
|
|
import * as os from 'os';
|
|
|
|
import * as path from 'path';
|
|
|
|
import * as util from 'util';
|
2020-05-21 15:13:16 -07:00
|
|
|
import { BrowserContext, PersistentContextOptions, validatePersistentContextOptions } from '../browserContext';
|
2020-05-22 07:03:42 -07:00
|
|
|
import { BrowserServer, WebSocketWrapper } from './browserServer';
|
2020-04-28 17:06:01 -07:00
|
|
|
import * as browserPaths from '../install/browserPaths';
|
2020-05-22 07:03:42 -07:00
|
|
|
import { Logger, RootLogger, InnerLogger } from '../logger';
|
2020-05-20 16:30:04 -07:00
|
|
|
import { ConnectionTransport, WebSocketTransport } from '../transport';
|
|
|
|
import { BrowserBase, BrowserOptions, Browser } from '../browser';
|
2020-05-21 09:43:10 -07:00
|
|
|
import { assert, helper } from '../helper';
|
|
|
|
import { TimeoutSettings } from '../timeoutSettings';
|
2020-05-22 07:03:42 -07:00
|
|
|
import { launchProcess, Env, waitForLine } from './processLauncher';
|
|
|
|
import { Events } from '../events';
|
|
|
|
import { TimeoutError } from '../errors';
|
|
|
|
import { PipeTransport } from './pipeTransport';
|
2020-01-24 14:49:47 -08:00
|
|
|
|
|
|
|
export type BrowserArgOptions = {
|
|
|
|
headless?: boolean,
|
|
|
|
args?: string[],
|
|
|
|
devtools?: boolean,
|
|
|
|
};
|
|
|
|
|
2020-05-20 16:30:04 -07:00
|
|
|
type LaunchOptionsBase = BrowserArgOptions & {
|
2020-01-24 14:49:47 -08:00
|
|
|
executablePath?: string,
|
|
|
|
ignoreDefaultArgs?: boolean | string[],
|
|
|
|
handleSIGINT?: boolean,
|
|
|
|
handleSIGTERM?: boolean,
|
|
|
|
handleSIGHUP?: boolean,
|
|
|
|
timeout?: number,
|
2020-04-20 23:24:53 -07:00
|
|
|
logger?: Logger,
|
2020-05-22 07:03:42 -07:00
|
|
|
env?: Env,
|
2020-01-24 14:49:47 -08:00
|
|
|
};
|
|
|
|
|
2020-05-14 13:22:33 -07:00
|
|
|
export function processBrowserArgOptions(options: LaunchOptionsBase): { devtools: boolean, headless: boolean } {
|
|
|
|
const { devtools = false, headless = !devtools } = options;
|
|
|
|
return { devtools, headless };
|
|
|
|
}
|
|
|
|
|
2020-05-22 07:03:42 -07:00
|
|
|
type ConnectOptions = {
|
2020-03-31 16:34:59 -07:00
|
|
|
wsEndpoint: string,
|
2020-04-20 07:52:26 -07:00
|
|
|
slowMo?: number,
|
2020-04-20 23:24:53 -07:00
|
|
|
logger?: Logger,
|
2020-05-21 09:43:10 -07:00
|
|
|
timeout?: number,
|
2020-03-31 16:34:59 -07:00
|
|
|
};
|
2020-05-20 16:30:04 -07:00
|
|
|
export type LaunchType = 'local' | 'server' | 'persistent';
|
2020-03-31 16:34:59 -07:00
|
|
|
export type LaunchOptions = LaunchOptionsBase & { slowMo?: number };
|
2020-05-22 07:03:42 -07:00
|
|
|
type LaunchServerOptions = LaunchOptionsBase & { port?: number };
|
2020-05-20 16:30:04 -07:00
|
|
|
|
|
|
|
export interface BrowserType {
|
2020-01-24 14:49:47 -08:00
|
|
|
executablePath(): string;
|
2020-01-28 18:09:07 -08:00
|
|
|
name(): string;
|
2020-03-31 16:34:59 -07:00
|
|
|
launch(options?: LaunchOptions): Promise<Browser>;
|
|
|
|
launchServer(options?: LaunchServerOptions): Promise<BrowserServer>;
|
2020-05-21 15:13:16 -07:00
|
|
|
launchPersistentContext(userDataDir: string, options?: LaunchOptions & PersistentContextOptions): Promise<BrowserContext>;
|
2020-02-04 19:41:38 -08:00
|
|
|
connect(options: ConnectOptions): Promise<Browser>;
|
2020-01-24 14:49:47 -08:00
|
|
|
}
|
2020-04-28 17:06:01 -07:00
|
|
|
|
2020-05-22 07:03:42 -07:00
|
|
|
const mkdtempAsync = util.promisify(fs.mkdtemp);
|
|
|
|
|
2020-05-20 16:30:04 -07:00
|
|
|
export abstract class BrowserTypeBase implements BrowserType {
|
2020-04-28 17:06:01 -07:00
|
|
|
private _name: string;
|
2020-04-29 17:19:21 -07:00
|
|
|
private _executablePath: string | undefined;
|
2020-05-22 07:03:42 -07:00
|
|
|
private _webSocketRegexNotPipe: RegExp | null;
|
2020-05-19 14:55:11 -07:00
|
|
|
readonly _browserPath: string;
|
2020-04-28 17:06:01 -07:00
|
|
|
|
2020-05-22 07:03:42 -07:00
|
|
|
constructor(packagePath: string, browser: browserPaths.BrowserDescriptor, webSocketRegexNotPipe: RegExp | null) {
|
2020-04-28 17:06:01 -07:00
|
|
|
this._name = browser.name;
|
2020-04-29 17:19:21 -07:00
|
|
|
const browsersPath = browserPaths.browsersPath(packagePath);
|
2020-05-19 14:55:11 -07:00
|
|
|
this._browserPath = browserPaths.browserDirectory(browsersPath, browser);
|
|
|
|
this._executablePath = browserPaths.executablePath(this._browserPath, browser);
|
2020-05-22 07:03:42 -07:00
|
|
|
this._webSocketRegexNotPipe = webSocketRegexNotPipe;
|
2020-04-28 17:06:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
executablePath(): string {
|
2020-04-29 17:19:21 -07:00
|
|
|
if (!this._executablePath)
|
|
|
|
throw new Error('Browser is not supported on current platform');
|
2020-04-28 17:06:01 -07:00
|
|
|
return this._executablePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
name(): string {
|
|
|
|
return this._name;
|
|
|
|
}
|
|
|
|
|
2020-05-20 16:30:04 -07:00
|
|
|
async launch(options: LaunchOptions = {}): Promise<Browser> {
|
|
|
|
assert(!(options as any).userDataDir, 'userDataDir option is not supported in `browserType.launch`. Use `browserType.launchPersistentContext` instead');
|
2020-05-21 15:13:16 -07:00
|
|
|
return this._innerLaunch('local', options, undefined);
|
2020-05-20 16:30:04 -07:00
|
|
|
}
|
|
|
|
|
2020-05-21 15:13:16 -07:00
|
|
|
async launchPersistentContext(userDataDir: string, options: LaunchOptions & PersistentContextOptions = {}): Promise<BrowserContext> {
|
|
|
|
const persistent = validatePersistentContextOptions(options);
|
|
|
|
const browser = await this._innerLaunch('persistent', options, persistent, userDataDir);
|
2020-05-21 09:43:10 -07:00
|
|
|
return browser._defaultContext!;
|
|
|
|
}
|
|
|
|
|
2020-05-21 15:13:16 -07:00
|
|
|
async _innerLaunch(launchType: LaunchType, options: LaunchOptions, persistent: PersistentContextOptions | undefined, userDataDir?: string): Promise<BrowserBase> {
|
2020-05-21 09:43:10 -07:00
|
|
|
const deadline = TimeoutSettings.computeDeadline(options.timeout, 30000);
|
|
|
|
const logger = new RootLogger(options.logger);
|
|
|
|
logger.startLaunchRecording();
|
|
|
|
|
|
|
|
let browserServer: BrowserServer | undefined;
|
|
|
|
try {
|
|
|
|
browserServer = await this._launchServer(options, launchType, logger, deadline, userDataDir);
|
2020-05-21 15:13:16 -07:00
|
|
|
const promise = this._innerLaunchPromise(browserServer, options, persistent);
|
2020-05-21 09:43:10 -07:00
|
|
|
const browser = await helper.waitWithDeadline(promise, 'the browser to launch', deadline, 'pw:browser*');
|
|
|
|
return browser;
|
|
|
|
} catch (e) {
|
|
|
|
e.message += '\n=============== Process output during launch: ===============\n' +
|
|
|
|
logger.launchRecording() +
|
|
|
|
'\n=============================================================';
|
|
|
|
if (browserServer)
|
|
|
|
await browserServer._closeOrKill(deadline);
|
|
|
|
throw e;
|
|
|
|
} finally {
|
|
|
|
logger.stopLaunchRecording();
|
|
|
|
}
|
|
|
|
}
|
2020-05-20 16:30:04 -07:00
|
|
|
|
2020-05-21 15:13:16 -07:00
|
|
|
async _innerLaunchPromise(browserServer: BrowserServer, options: LaunchOptions, persistent: PersistentContextOptions | undefined): Promise<BrowserBase> {
|
2020-05-21 09:43:10 -07:00
|
|
|
if ((options as any).__testHookBeforeCreateBrowser)
|
|
|
|
await (options as any).__testHookBeforeCreateBrowser();
|
|
|
|
|
2020-05-21 19:16:13 -07:00
|
|
|
const browserOptions: BrowserOptions = {
|
|
|
|
slowMo: options.slowMo,
|
|
|
|
persistent,
|
|
|
|
headful: browserServer._headful,
|
|
|
|
logger: browserServer._logger,
|
|
|
|
downloadsPath: browserServer._downloadsPath,
|
|
|
|
ownedServer: browserServer,
|
|
|
|
};
|
|
|
|
for (const [key, value] of Object.entries(options)) {
|
|
|
|
if (key.startsWith('__testHook'))
|
|
|
|
(browserOptions as any)[key] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
const browser = await this._connectToTransport(browserServer._transport, browserOptions);
|
2020-05-21 15:13:16 -07:00
|
|
|
if (persistent && (!options.ignoreDefaultArgs || Array.isArray(options.ignoreDefaultArgs))) {
|
2020-05-20 16:30:04 -07:00
|
|
|
const context = browser._defaultContext!;
|
2020-05-21 09:43:10 -07:00
|
|
|
await context._loadDefaultContext();
|
|
|
|
}
|
|
|
|
return browser;
|
2020-05-20 16:30:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
async launchServer(options: LaunchServerOptions = {}): Promise<BrowserServer> {
|
2020-05-21 09:43:10 -07:00
|
|
|
const logger = new RootLogger(options.logger);
|
|
|
|
return this._launchServer(options, 'server', logger, TimeoutSettings.computeDeadline(options.timeout, 30000));
|
2020-05-20 16:30:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
async connect(options: ConnectOptions): Promise<Browser> {
|
2020-05-21 09:43:10 -07:00
|
|
|
const deadline = TimeoutSettings.computeDeadline(options.timeout, 30000);
|
2020-05-20 16:30:04 -07:00
|
|
|
const logger = new RootLogger(options.logger);
|
2020-05-21 09:43:10 -07:00
|
|
|
logger.startLaunchRecording();
|
|
|
|
|
|
|
|
let transport: ConnectionTransport | undefined;
|
|
|
|
try {
|
|
|
|
transport = await WebSocketTransport.connect(options.wsEndpoint, logger, deadline);
|
|
|
|
const promise = this._innerConnectPromise(transport, options, logger);
|
|
|
|
const browser = await helper.waitWithDeadline(promise, 'connect to browser', deadline, 'pw:browser*');
|
|
|
|
logger.stopLaunchRecording();
|
|
|
|
return browser;
|
|
|
|
} catch (e) {
|
|
|
|
e.message += '\n=============== Process output during connect: ===============\n' +
|
|
|
|
logger.launchRecording() +
|
|
|
|
'\n=============================================================';
|
|
|
|
try {
|
|
|
|
if (transport)
|
|
|
|
transport.close();
|
|
|
|
} catch (e) {
|
|
|
|
}
|
|
|
|
throw e;
|
|
|
|
} finally {
|
|
|
|
logger.stopLaunchRecording();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async _innerConnectPromise(transport: ConnectionTransport, options: ConnectOptions, logger: RootLogger): Promise<Browser> {
|
|
|
|
if ((options as any).__testHookBeforeCreateBrowser)
|
|
|
|
await (options as any).__testHookBeforeCreateBrowser();
|
2020-05-21 15:13:16 -07:00
|
|
|
return this._connectToTransport(transport, { slowMo: options.slowMo, logger });
|
2020-05-20 16:30:04 -07:00
|
|
|
}
|
|
|
|
|
2020-05-22 07:03:42 -07:00
|
|
|
private async _launchServer(options: LaunchServerOptions, launchType: LaunchType, logger: RootLogger, deadline: number, userDataDir?: string): Promise<BrowserServer> {
|
|
|
|
const {
|
|
|
|
ignoreDefaultArgs = false,
|
|
|
|
args = [],
|
|
|
|
executablePath = null,
|
|
|
|
env = process.env,
|
|
|
|
handleSIGINT = true,
|
|
|
|
handleSIGTERM = true,
|
|
|
|
handleSIGHUP = true,
|
|
|
|
port = 0,
|
|
|
|
} = options;
|
|
|
|
assert(!port || launchType === 'server', 'Cannot specify a port without launching as a server.');
|
|
|
|
|
|
|
|
let temporaryUserDataDir: string | null = null;
|
|
|
|
if (!userDataDir) {
|
|
|
|
userDataDir = await mkdtempAsync(path.join(os.tmpdir(), `playwright_${this._name}dev_profile-`));
|
|
|
|
temporaryUserDataDir = userDataDir;
|
|
|
|
}
|
|
|
|
|
|
|
|
const browserArguments = [];
|
|
|
|
if (!ignoreDefaultArgs)
|
|
|
|
browserArguments.push(...this._defaultArgs(options, launchType, userDataDir));
|
|
|
|
else if (Array.isArray(ignoreDefaultArgs))
|
|
|
|
browserArguments.push(...this._defaultArgs(options, launchType, userDataDir).filter(arg => ignoreDefaultArgs.indexOf(arg) === -1));
|
|
|
|
else
|
|
|
|
browserArguments.push(...args);
|
|
|
|
|
|
|
|
const executable = executablePath || this.executablePath();
|
|
|
|
if (!executable)
|
|
|
|
throw new Error(`No executable path is specified. Pass "executablePath" option directly.`);
|
|
|
|
|
|
|
|
// Note: it is important to define these variables before launchProcess, so that we don't get
|
|
|
|
// "Cannot access 'browserServer' before initialization" if something went wrong.
|
|
|
|
let transport: ConnectionTransport | undefined = undefined;
|
|
|
|
let browserServer: BrowserServer | undefined = undefined;
|
|
|
|
const { launchedProcess, gracefullyClose, downloadsPath } = await launchProcess({
|
|
|
|
executablePath: executable,
|
|
|
|
args: browserArguments,
|
|
|
|
env: this._amendEnvironment(env, userDataDir, executable, browserArguments),
|
|
|
|
handleSIGINT,
|
|
|
|
handleSIGTERM,
|
|
|
|
handleSIGHUP,
|
|
|
|
logger,
|
|
|
|
pipe: !this._webSocketRegexNotPipe,
|
|
|
|
tempDir: temporaryUserDataDir || undefined,
|
|
|
|
attemptToGracefullyClose: async () => {
|
|
|
|
if ((options as any).__testHookGracefullyClose)
|
|
|
|
await (options as any).__testHookGracefullyClose();
|
|
|
|
// We try to gracefully close to prevent crash reporting and core dumps.
|
|
|
|
// Note that it's fine to reuse the pipe transport, since
|
|
|
|
// our connection ignores kBrowserCloseMessageId.
|
|
|
|
this._attemptToGracefullyCloseBrowser(transport!);
|
|
|
|
},
|
|
|
|
onkill: (exitCode, signal) => {
|
|
|
|
if (browserServer)
|
|
|
|
browserServer.emit(Events.BrowserServer.Close, exitCode, signal);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (this._webSocketRegexNotPipe) {
|
|
|
|
const timeoutError = new TimeoutError(`Timed out while trying to connect to the browser!`);
|
|
|
|
const match = await waitForLine(launchedProcess, launchedProcess.stdout, this._webSocketRegexNotPipe, helper.timeUntilDeadline(deadline), timeoutError);
|
|
|
|
const innerEndpoint = match[1];
|
|
|
|
transport = await WebSocketTransport.connect(innerEndpoint, logger, deadline);
|
|
|
|
} else {
|
|
|
|
const stdio = launchedProcess.stdio as unknown as [NodeJS.ReadableStream, NodeJS.WritableStream, NodeJS.WritableStream, NodeJS.WritableStream, NodeJS.ReadableStream];
|
|
|
|
transport = new PipeTransport(stdio[3], stdio[4], logger);
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
// If we can't establish a connection, kill the process and exit.
|
|
|
|
helper.killProcess(launchedProcess);
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
browserServer = new BrowserServer(options, launchedProcess, gracefullyClose, transport, downloadsPath,
|
|
|
|
launchType === 'server' ? this._wrapTransportWithWebSocket(transport, logger, port) : null);
|
|
|
|
return browserServer;
|
|
|
|
}
|
|
|
|
|
|
|
|
abstract _defaultArgs(options: BrowserArgOptions, launchType: LaunchType, userDataDir: string): string[];
|
2020-05-20 16:30:04 -07:00
|
|
|
abstract _connectToTransport(transport: ConnectionTransport, options: BrowserOptions): Promise<BrowserBase>;
|
2020-05-22 07:03:42 -07:00
|
|
|
abstract _wrapTransportWithWebSocket(transport: ConnectionTransport, logger: InnerLogger, port: number): WebSocketWrapper;
|
|
|
|
abstract _amendEnvironment(env: Env, userDataDir: string, executable: string, browserArguments: string[]): Env;
|
|
|
|
abstract _attemptToGracefullyCloseBrowser(transport: ConnectionTransport): void;
|
|
|
|
}
|