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.
|
|
|
|
*/
|
|
|
|
|
2023-05-17 13:21:40 +00:00
|
|
|
import {
|
|
|
|
Browser as SupportedBrowser,
|
|
|
|
resolveBuildId,
|
|
|
|
detectBrowserPlatform,
|
|
|
|
getInstalledBrowsers,
|
|
|
|
uninstall,
|
|
|
|
} from '@puppeteer/browsers';
|
|
|
|
|
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-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';
|
2023-05-02 06:53:40 +00:00
|
|
|
this.defaultBrowserRevision = PUPPETEER_REVISIONS.chrome;
|
2022-10-21 13:09:21 +00:00
|
|
|
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);
|
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
|
2023-05-02 06:53:40 +00:00
|
|
|
* with the version of Chrome for Testing downloaded by default.
|
|
|
|
* There is no guarantee it will work with any other version. If Google Chrome
|
|
|
|
* (rather than Chrome for Testing) 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}
|
2023-05-02 06:53:40 +00:00
|
|
|
* describes some differences for Linux users. See
|
|
|
|
* {@link https://goo.gle/chrome-for-testing | this doc} for the description
|
|
|
|
* of Chrome for Testing.
|
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':
|
2023-05-02 06:53:40 +00:00
|
|
|
this.defaultBrowserRevision = PUPPETEER_REVISIONS.chrome;
|
2022-10-21 13:09:21 +00:00
|
|
|
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 {
|
2023-05-02 06:53:40 +00:00
|
|
|
return this.configuration.downloadPath ?? this.configuration.cacheDirectory;
|
2022-10-21 13:09:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
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
|
|
|
}
|
2023-05-17 13:21:40 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes all non-current Firefox and Chrome binaries in the cache directory
|
|
|
|
* identified by the provided Puppeteer configuration. The current browser
|
|
|
|
* version is determined by resolving PUPPETEER_REVISIONS from Puppeteer
|
|
|
|
* unless `configuration.browserRevision` is provided.
|
|
|
|
*
|
|
|
|
* @remarks
|
|
|
|
*
|
|
|
|
* Note that the method does not check if any other Puppeteer versions
|
|
|
|
* installed on the host that use the same cache directory require the
|
|
|
|
* non-current binaries.
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
async trimCache(): Promise<void> {
|
|
|
|
const platform = detectBrowserPlatform();
|
|
|
|
if (!platform) {
|
|
|
|
throw new Error('The current platform is not supported.');
|
|
|
|
}
|
|
|
|
|
|
|
|
const cacheDir =
|
|
|
|
this.configuration.downloadPath ?? this.configuration.cacheDirectory!;
|
|
|
|
const installedBrowsers = await getInstalledBrowsers({
|
|
|
|
cacheDir,
|
|
|
|
});
|
|
|
|
|
|
|
|
const product = this.configuration.defaultProduct!;
|
|
|
|
|
2023-05-17 14:10:51 +00:00
|
|
|
const puppeteerBrowsers: Array<{
|
|
|
|
product: Product;
|
|
|
|
browser: SupportedBrowser;
|
|
|
|
currentBuildId: string;
|
|
|
|
}> = [
|
|
|
|
{
|
|
|
|
product: 'chrome',
|
|
|
|
browser: SupportedBrowser.CHROME,
|
|
|
|
currentBuildId: '',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
product: 'firefox',
|
|
|
|
browser: SupportedBrowser.FIREFOX,
|
|
|
|
currentBuildId: '',
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
// Resolve current buildIds.
|
|
|
|
for (const item of puppeteerBrowsers) {
|
|
|
|
item.currentBuildId = await resolveBuildId(
|
|
|
|
item.browser,
|
|
|
|
platform,
|
|
|
|
(product === item.product
|
|
|
|
? this.configuration.browserRevision
|
|
|
|
: null) || PUPPETEER_REVISIONS[item.product]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const currentBrowserBuilds = new Set(
|
|
|
|
puppeteerBrowsers.map(browser => {
|
|
|
|
return `${browser.browser}_${browser.currentBuildId}`;
|
|
|
|
})
|
2023-05-17 13:21:40 +00:00
|
|
|
);
|
|
|
|
|
2023-05-17 14:10:51 +00:00
|
|
|
const currentBrowsers = new Set(
|
|
|
|
puppeteerBrowsers.map(browser => {
|
|
|
|
return browser.browser;
|
|
|
|
})
|
2023-05-17 13:21:40 +00:00
|
|
|
);
|
|
|
|
|
2023-05-17 14:10:51 +00:00
|
|
|
for (const installedBrowser of installedBrowsers) {
|
|
|
|
// Don't uninstall browsers that are not managed by Puppeteer yet.
|
|
|
|
if (!currentBrowsers.has(installedBrowser.browser)) {
|
|
|
|
continue;
|
2023-05-17 13:21:40 +00:00
|
|
|
}
|
2023-05-17 14:10:51 +00:00
|
|
|
// Keep the browser build used by the current Puppeteer installation.
|
2023-05-17 13:21:40 +00:00
|
|
|
if (
|
2023-05-17 14:10:51 +00:00
|
|
|
currentBrowserBuilds.has(
|
|
|
|
`${installedBrowser.browser}_${installedBrowser.buildId}`
|
|
|
|
)
|
2023-05-17 13:21:40 +00:00
|
|
|
) {
|
2023-05-17 14:10:51 +00:00
|
|
|
continue;
|
2023-05-17 13:21:40 +00:00
|
|
|
}
|
2023-05-17 14:10:51 +00:00
|
|
|
|
|
|
|
await uninstall({
|
|
|
|
browser: SupportedBrowser.CHROME,
|
|
|
|
platform,
|
|
|
|
cacheDir,
|
|
|
|
buildId: installedBrowser.buildId,
|
|
|
|
});
|
2023-05-17 13:21:40 +00:00
|
|
|
}
|
|
|
|
}
|
2020-10-13 15:19:26 +00:00
|
|
|
}
|