2020-10-13 15:19:26 +00:00
|
|
|
/**
|
|
|
|
* Copyright 2020 Google Inc. 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.
|
|
|
|
*/
|
|
|
|
|
2022-10-21 13:09:21 +00:00
|
|
|
import {join} from 'path';
|
2023-02-15 23:09:31 +00:00
|
|
|
|
2022-10-05 12:17:03 +00:00
|
|
|
import {Browser} from '../api/Browser.js';
|
|
|
|
import {BrowserConnectOptions} from '../common/BrowserConnector.js';
|
2023-02-15 23:09:31 +00:00
|
|
|
import {Configuration} from '../common/Configuration.js';
|
2022-10-05 12:17:03 +00:00
|
|
|
import {Product} from '../common/Product.js';
|
2020-10-13 15:19:26 +00:00
|
|
|
import {
|
|
|
|
CommonPuppeteerSettings,
|
|
|
|
ConnectOptions,
|
2022-10-05 12:17:03 +00:00
|
|
|
Puppeteer,
|
2020-10-13 15:19:26 +00:00
|
|
|
} from '../common/Puppeteer.js';
|
2022-10-05 12:17:03 +00:00
|
|
|
import {PUPPETEER_REVISIONS} from '../revisions.js';
|
2023-02-15 23:09:31 +00:00
|
|
|
|
2022-06-22 13:25:44 +00:00
|
|
|
import {BrowserFetcher, BrowserFetcherOptions} from './BrowserFetcher.js';
|
2022-10-21 13:09:21 +00:00
|
|
|
import {ChromeLauncher} from './ChromeLauncher.js';
|
|
|
|
import {FirefoxLauncher} from './FirefoxLauncher.js';
|
|
|
|
import {
|
|
|
|
BrowserLaunchArgumentOptions,
|
|
|
|
ChromeReleaseChannel,
|
|
|
|
LaunchOptions,
|
|
|
|
} from './LaunchOptions.js';
|
|
|
|
import {ProductLauncher} from './ProductLauncher.js';
|
2020-10-13 15:19:26 +00:00
|
|
|
|
2022-05-31 17:54:25 +00:00
|
|
|
/**
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
export interface PuppeteerLaunchOptions
|
|
|
|
extends LaunchOptions,
|
|
|
|
BrowserLaunchArgumentOptions,
|
|
|
|
BrowserConnectOptions {
|
|
|
|
product?: Product;
|
|
|
|
extraPrefsFirefox?: Record<string, unknown>;
|
|
|
|
}
|
|
|
|
|
2020-10-13 15:19:26 +00:00
|
|
|
/**
|
2022-06-27 07:24:23 +00:00
|
|
|
* Extends the main {@link Puppeteer} class with Node specific behaviour for
|
|
|
|
* fetching and downloading browsers.
|
2020-10-13 15:19:26 +00:00
|
|
|
*
|
|
|
|
* If you're using Puppeteer in a Node environment, this is the class you'll get
|
|
|
|
* when you run `require('puppeteer')` (or the equivalent ES `import`).
|
|
|
|
*
|
|
|
|
* @remarks
|
|
|
|
* The most common method to use is {@link PuppeteerNode.launch | launch}, which
|
|
|
|
* is used to launch and connect to a new browser instance.
|
|
|
|
*
|
|
|
|
* See {@link Puppeteer | the main Puppeteer class} for methods common to all
|
|
|
|
* environments, such as {@link Puppeteer.connect}.
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
* The following is a typical example of using Puppeteer to drive automation:
|
2022-08-12 12:15:26 +00:00
|
|
|
*
|
2022-07-01 11:52:39 +00:00
|
|
|
* ```ts
|
2022-12-09 12:57:39 +00:00
|
|
|
* import puppeteer from 'puppeteer';
|
2020-10-13 15:19:26 +00:00
|
|
|
*
|
|
|
|
* (async () => {
|
|
|
|
* const browser = await puppeteer.launch();
|
|
|
|
* const page = await browser.newPage();
|
|
|
|
* await page.goto('https://www.google.com');
|
|
|
|
* // other actions...
|
|
|
|
* await browser.close();
|
|
|
|
* })();
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* Once you have created a `page` you have access to a large API to interact
|
|
|
|
* with the page, navigate, or find certain elements in that page.
|
|
|
|
* The {@link Page | `page` documentation} lists all the available methods.
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
export class PuppeteerNode extends Puppeteer {
|
2022-10-21 13:09:21 +00:00
|
|
|
#_launcher?: ProductLauncher;
|
|
|
|
#lastLaunchedProduct?: Product;
|
2022-06-13 09:16:25 +00:00
|
|
|
|
2022-09-12 11:00:37 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-10-21 13:09:21 +00:00
|
|
|
defaultBrowserRevision: string;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
configuration: Configuration = {};
|
2020-10-13 15:19:26 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
constructor(
|
|
|
|
settings: {
|
2022-10-21 13:09:21 +00:00
|
|
|
configuration?: Configuration;
|
2020-10-13 15:19:26 +00:00
|
|
|
} & CommonPuppeteerSettings
|
|
|
|
) {
|
2022-10-21 13:09:21 +00:00
|
|
|
const {configuration, ...commonSettings} = settings;
|
2020-10-13 15:19:26 +00:00
|
|
|
super(commonSettings);
|
2022-10-21 13:09:21 +00:00
|
|
|
if (configuration) {
|
|
|
|
this.configuration = configuration;
|
|
|
|
}
|
|
|
|
switch (this.configuration.defaultProduct) {
|
|
|
|
case 'firefox':
|
|
|
|
this.defaultBrowserRevision = PUPPETEER_REVISIONS.firefox;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
this.configuration.defaultProduct = 'chrome';
|
|
|
|
this.defaultBrowserRevision = PUPPETEER_REVISIONS.chromium;
|
|
|
|
break;
|
2022-10-05 12:17:03 +00:00
|
|
|
}
|
2022-06-09 17:00:50 +00:00
|
|
|
|
|
|
|
this.connect = this.connect.bind(this);
|
|
|
|
this.launch = this.launch.bind(this);
|
|
|
|
this.executablePath = this.executablePath.bind(this);
|
|
|
|
this.defaultArgs = this.defaultArgs.bind(this);
|
|
|
|
this.createBrowserFetcher = this.createBrowserFetcher.bind(this);
|
2020-10-13 15:19:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This method attaches Puppeteer to an existing browser instance.
|
|
|
|
*
|
|
|
|
* @param options - Set of configurable options to set on the browser.
|
|
|
|
* @returns Promise which resolves to browser instance.
|
|
|
|
*/
|
2022-05-31 14:34:16 +00:00
|
|
|
override connect(options: ConnectOptions): Promise<Browser> {
|
2020-10-13 15:19:26 +00:00
|
|
|
return super.connect(options);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-10-21 13:09:21 +00:00
|
|
|
* Launches a browser instance with given arguments and options when
|
|
|
|
* specified.
|
2020-10-13 15:19:26 +00:00
|
|
|
*
|
2022-10-24 07:07:05 +00:00
|
|
|
* When using with `puppeteer-core`,
|
2023-03-28 18:02:00 +00:00
|
|
|
* {@link LaunchOptions | options.executablePath} or
|
|
|
|
* {@link LaunchOptions | options.channel} must be provided.
|
2022-10-24 07:07:05 +00:00
|
|
|
*
|
2020-10-13 15:19:26 +00:00
|
|
|
* @example
|
2023-03-28 18:02:00 +00:00
|
|
|
* You can use {@link LaunchOptions | options.ignoreDefaultArgs}
|
2022-10-24 07:07:05 +00:00
|
|
|
* to filter out `--mute-audio` from default arguments:
|
2022-08-12 12:15:26 +00:00
|
|
|
*
|
2022-07-01 11:52:39 +00:00
|
|
|
* ```ts
|
2020-10-13 15:19:26 +00:00
|
|
|
* const browser = await puppeteer.launch({
|
2022-08-12 12:15:26 +00:00
|
|
|
* ignoreDefaultArgs: ['--mute-audio'],
|
2020-10-13 15:19:26 +00:00
|
|
|
* });
|
|
|
|
* ```
|
|
|
|
*
|
2022-06-27 07:24:23 +00:00
|
|
|
* @remarks
|
2022-10-21 13:09:21 +00:00
|
|
|
* Puppeteer can also be used to control the Chrome browser, but it works best
|
2022-10-24 07:07:05 +00:00
|
|
|
* with the version of Chromium downloaded by default by Puppeteer. There is
|
|
|
|
* no guarantee it will work with any other version. If Google Chrome (rather
|
|
|
|
* than Chromium) is preferred, a
|
2022-10-21 13:09:21 +00:00
|
|
|
* {@link https://www.google.com/chrome/browser/canary.html | Chrome Canary}
|
2022-06-27 07:24:23 +00:00
|
|
|
* or
|
|
|
|
* {@link https://www.chromium.org/getting-involved/dev-channel | Dev Channel}
|
2022-10-24 07:07:05 +00:00
|
|
|
* build is suggested. See
|
2022-06-27 07:24:23 +00:00
|
|
|
* {@link https://www.howtogeek.com/202825/what%E2%80%99s-the-difference-between-chromium-and-chrome/ | this article}
|
|
|
|
* for a description of the differences between Chromium and Chrome.
|
|
|
|
* {@link https://chromium.googlesource.com/chromium/src/+/lkgr/docs/chromium_browser_vs_google_chrome.md | This article}
|
|
|
|
* describes some differences for Linux users.
|
2020-10-13 15:19:26 +00:00
|
|
|
*
|
2022-10-21 13:09:21 +00:00
|
|
|
* @param options - Options to configure launching behavior.
|
2020-10-13 15:19:26 +00:00
|
|
|
*/
|
2022-05-31 17:54:25 +00:00
|
|
|
launch(options: PuppeteerLaunchOptions = {}): Promise<Browser> {
|
2022-10-21 13:09:21 +00:00
|
|
|
const {product = this.defaultProduct} = options;
|
|
|
|
this.#lastLaunchedProduct = product;
|
|
|
|
return this.#launcher.launch(options);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
get #launcher(): ProductLauncher {
|
|
|
|
if (
|
|
|
|
this.#_launcher &&
|
|
|
|
this.#_launcher.product === this.lastLaunchedProduct
|
|
|
|
) {
|
|
|
|
return this.#_launcher;
|
2022-06-14 11:55:35 +00:00
|
|
|
}
|
2022-10-21 13:09:21 +00:00
|
|
|
switch (this.lastLaunchedProduct) {
|
|
|
|
case 'chrome':
|
|
|
|
this.defaultBrowserRevision = PUPPETEER_REVISIONS.chromium;
|
|
|
|
this.#_launcher = new ChromeLauncher(this);
|
|
|
|
break;
|
|
|
|
case 'firefox':
|
|
|
|
this.defaultBrowserRevision = PUPPETEER_REVISIONS.firefox;
|
|
|
|
this.#_launcher = new FirefoxLauncher(this);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error(`Unknown product: ${this.#lastLaunchedProduct}`);
|
|
|
|
}
|
|
|
|
return this.#_launcher;
|
2020-10-13 15:19:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-03-30 11:54:00 +00:00
|
|
|
* The default executable path.
|
2020-10-13 15:19:26 +00:00
|
|
|
*/
|
2022-10-21 13:09:21 +00:00
|
|
|
executablePath(channel?: ChromeReleaseChannel): string {
|
|
|
|
return this.#launcher.executablePath(channel);
|
2020-10-13 15:19:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-10-21 13:09:21 +00:00
|
|
|
get browserRevision(): string {
|
2023-01-13 10:57:48 +00:00
|
|
|
return (
|
|
|
|
this.#_launcher?.getActualBrowserRevision() ??
|
|
|
|
this.configuration.browserRevision ??
|
|
|
|
this.defaultBrowserRevision!
|
|
|
|
);
|
2020-10-13 15:19:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-03-30 11:54:00 +00:00
|
|
|
* The default download path for puppeteer. For puppeteer-core, this
|
2022-10-21 13:09:21 +00:00
|
|
|
* code should never be called as it is never defined.
|
2020-10-13 15:19:26 +00:00
|
|
|
*
|
2022-10-21 13:09:21 +00:00
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
get defaultDownloadPath(): string | undefined {
|
|
|
|
return (
|
|
|
|
this.configuration.downloadPath ??
|
|
|
|
join(this.configuration.cacheDirectory!, this.product)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-03-30 11:54:00 +00:00
|
|
|
* The name of the browser that was last launched.
|
2022-10-21 13:09:21 +00:00
|
|
|
*/
|
|
|
|
get lastLaunchedProduct(): Product {
|
|
|
|
return this.#lastLaunchedProduct ?? this.defaultProduct;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-03-30 11:54:00 +00:00
|
|
|
* The name of the browser that will be launched by default. For
|
2022-10-21 13:09:21 +00:00
|
|
|
* `puppeteer`, this is influenced by your configuration. Otherwise, it's
|
|
|
|
* `chrome`.
|
|
|
|
*/
|
|
|
|
get defaultProduct(): Product {
|
|
|
|
return this.configuration.defaultProduct ?? 'chrome';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @deprecated Do not use as this field as it does not take into account
|
2022-10-24 07:07:05 +00:00
|
|
|
* multiple browsers of different types. Use
|
|
|
|
* {@link PuppeteerNode.defaultProduct | defaultProduct} or
|
|
|
|
* {@link PuppeteerNode.lastLaunchedProduct | lastLaunchedProduct}.
|
2022-10-21 13:09:21 +00:00
|
|
|
*
|
|
|
|
* @returns The name of the browser that is under automation.
|
2020-10-13 15:19:26 +00:00
|
|
|
*/
|
|
|
|
get product(): string {
|
2022-10-21 13:09:21 +00:00
|
|
|
return this.#launcher.product;
|
2020-10-13 15:19:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param options - Set of configurable options to set on the browser.
|
2022-10-21 13:09:21 +00:00
|
|
|
*
|
2020-10-13 15:19:26 +00:00
|
|
|
* @returns The default flags that Chromium will be launched with.
|
|
|
|
*/
|
2021-02-16 09:39:31 +00:00
|
|
|
defaultArgs(options: BrowserLaunchArgumentOptions = {}): string[] {
|
2022-10-21 13:09:21 +00:00
|
|
|
return this.#launcher.defaultArgs(options);
|
2020-10-13 15:19:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-06-27 07:24:23 +00:00
|
|
|
* @param options - Set of configurable options to specify the settings of the
|
|
|
|
* BrowserFetcher.
|
2022-10-20 11:25:24 +00:00
|
|
|
*
|
2022-11-10 16:11:18 +00:00
|
|
|
* @remarks
|
|
|
|
* If you are using `puppeteer-core`, do not use this method. Just
|
|
|
|
* construct {@link BrowserFetcher} manually.
|
|
|
|
*
|
2020-10-13 15:19:26 +00:00
|
|
|
* @returns A new BrowserFetcher instance.
|
|
|
|
*/
|
2022-10-21 13:09:21 +00:00
|
|
|
createBrowserFetcher(
|
2023-01-11 15:15:53 +00:00
|
|
|
options: Partial<BrowserFetcherOptions> = {}
|
2022-10-21 13:09:21 +00:00
|
|
|
): BrowserFetcher {
|
|
|
|
const downloadPath = this.defaultDownloadPath;
|
2023-01-13 10:57:48 +00:00
|
|
|
if (!options.path && downloadPath) {
|
2022-10-21 13:09:21 +00:00
|
|
|
options.path = downloadPath;
|
|
|
|
}
|
|
|
|
if (!options.path) {
|
|
|
|
throw new Error('A `path` must be specified for `puppeteer-core`.');
|
|
|
|
}
|
2023-01-13 10:57:48 +00:00
|
|
|
if (
|
|
|
|
!('useMacOSARMBinary' in options) &&
|
|
|
|
this.configuration.experiments?.macArmChromiumEnabled
|
|
|
|
) {
|
2022-10-21 13:09:21 +00:00
|
|
|
options.useMacOSARMBinary = true;
|
|
|
|
}
|
2023-01-13 10:57:48 +00:00
|
|
|
if (!('host' in options) && this.configuration.downloadHost) {
|
2022-10-21 13:09:21 +00:00
|
|
|
options.host = this.configuration.downloadHost;
|
|
|
|
}
|
2023-01-13 10:57:48 +00:00
|
|
|
if (!('product' in options) && this.configuration.defaultProduct) {
|
2023-01-11 15:15:53 +00:00
|
|
|
options.product = this.configuration.defaultProduct;
|
|
|
|
}
|
2022-10-21 13:09:21 +00:00
|
|
|
return new BrowserFetcher(options as BrowserFetcherOptions);
|
2020-10-13 15:19:26 +00:00
|
|
|
}
|
|
|
|
}
|