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-02-05 12:41:55 -08:00
|
|
|
import { BrowserContext } from '../browserContext';
|
|
|
|
import { BrowserServer } from './browserServer';
|
2020-04-28 17:06:01 -07:00
|
|
|
import * as browserPaths from '../install/browserPaths';
|
2020-05-20 16:30:04 -07:00
|
|
|
import { Logger, RootLogger } from '../logger';
|
|
|
|
import { ConnectionTransport, WebSocketTransport } from '../transport';
|
|
|
|
import { BrowserBase, BrowserOptions, Browser } from '../browser';
|
|
|
|
import { assert } from '../helper';
|
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-04-23 14:45:57 -07:00
|
|
|
env?: {[key: string]: string|number|boolean}
|
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-03-31 16:34:59 -07:00
|
|
|
export type ConnectOptions = {
|
|
|
|
wsEndpoint: string,
|
2020-04-20 07:52:26 -07:00
|
|
|
slowMo?: number,
|
2020-04-20 23:24:53 -07:00
|
|
|
logger?: Logger,
|
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 };
|
|
|
|
export 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-03-12 01:10:48 +00:00
|
|
|
launchPersistentContext(userDataDir: string, options?: LaunchOptions): 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-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-19 14:55:11 -07:00
|
|
|
readonly _browserPath: string;
|
2020-04-28 17:06:01 -07:00
|
|
|
|
|
|
|
constructor(packagePath: string, browser: browserPaths.BrowserDescriptor) {
|
|
|
|
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-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');
|
|
|
|
const browserServer = new BrowserServer(options);
|
|
|
|
const { transport, downloadsPath } = await this._launchServer(options, 'local', browserServer);
|
|
|
|
return await browserServer._initializeOrClose(async () => {
|
|
|
|
return this._connectToServer(browserServer, false, transport!, downloadsPath);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async launchPersistentContext(userDataDir: string, options: LaunchOptions = {}): Promise<BrowserContext> {
|
|
|
|
const browserServer = new BrowserServer(options);
|
|
|
|
const { transport, downloadsPath } = await this._launchServer(options, 'persistent', browserServer, userDataDir);
|
|
|
|
|
|
|
|
return await browserServer._initializeOrClose(async () => {
|
|
|
|
const browser = await this._connectToServer(browserServer, true, transport!, downloadsPath);
|
|
|
|
const context = browser._defaultContext!;
|
|
|
|
if (!options.ignoreDefaultArgs || Array.isArray(options.ignoreDefaultArgs))
|
|
|
|
await context._loadDefaultContext();
|
|
|
|
return context;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async launchServer(options: LaunchServerOptions = {}): Promise<BrowserServer> {
|
|
|
|
const browserServer = new BrowserServer(options);
|
|
|
|
await this._launchServer(options, 'server', browserServer);
|
|
|
|
return browserServer;
|
|
|
|
}
|
|
|
|
|
|
|
|
async connect(options: ConnectOptions): Promise<Browser> {
|
|
|
|
const logger = new RootLogger(options.logger);
|
|
|
|
return await WebSocketTransport.connect(options.wsEndpoint, async transport => {
|
|
|
|
if ((options as any).__testHookBeforeCreateBrowser)
|
|
|
|
await (options as any).__testHookBeforeCreateBrowser();
|
|
|
|
return this._connectToTransport(transport, { slowMo: options.slowMo, logger, downloadsPath: '' });
|
|
|
|
}, logger);
|
|
|
|
}
|
|
|
|
|
|
|
|
abstract _launchServer(options: LaunchServerOptions, launchType: LaunchType, browserServer: BrowserServer, userDataDir?: string): Promise<{ transport?: ConnectionTransport, downloadsPath: string }>;
|
|
|
|
abstract _connectToServer(browserServer: BrowserServer, persistent: boolean, transport: ConnectionTransport, downloadsPath: string): Promise<BrowserBase>;
|
|
|
|
abstract _connectToTransport(transport: ConnectionTransport, options: BrowserOptions): Promise<BrowserBase>;
|
2020-04-28 17:06:01 -07:00
|
|
|
}
|