feat: implement system channels for chrome in browsers (#9844)
This commit is contained in:
parent
73ca94fe66
commit
dec48a9592
@ -19,10 +19,18 @@ import yargs from 'yargs';
|
|||||||
import {hideBin} from 'yargs/helpers';
|
import {hideBin} from 'yargs/helpers';
|
||||||
|
|
||||||
import {resolveBuildId} from './browsers/browsers.js';
|
import {resolveBuildId} from './browsers/browsers.js';
|
||||||
import {Browser, BrowserPlatform} from './browsers/types.js';
|
import {
|
||||||
|
Browser,
|
||||||
|
BrowserPlatform,
|
||||||
|
ChromeReleaseChannel,
|
||||||
|
} from './browsers/types.js';
|
||||||
import {detectBrowserPlatform} from './detectPlatform.js';
|
import {detectBrowserPlatform} from './detectPlatform.js';
|
||||||
import {fetch} from './fetch.js';
|
import {fetch} from './fetch.js';
|
||||||
import {computeExecutablePath, launch} from './launcher.js';
|
import {
|
||||||
|
computeExecutablePath,
|
||||||
|
computeSystemExecutablePath,
|
||||||
|
launch,
|
||||||
|
} from './launcher.js';
|
||||||
|
|
||||||
type InstallArgs = {
|
type InstallArgs = {
|
||||||
browser: {
|
browser: {
|
||||||
@ -41,6 +49,7 @@ type LaunchArgs = {
|
|||||||
path?: string;
|
path?: string;
|
||||||
platform?: BrowserPlatform;
|
platform?: BrowserPlatform;
|
||||||
detached: boolean;
|
detached: boolean;
|
||||||
|
system: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export class CLI {
|
export class CLI {
|
||||||
@ -132,18 +141,30 @@ export class CLI {
|
|||||||
this.#definePathParameter(yargs);
|
this.#definePathParameter(yargs);
|
||||||
yargs.option('detached', {
|
yargs.option('detached', {
|
||||||
type: 'boolean',
|
type: 'boolean',
|
||||||
desc: 'Whether to detach the child process.',
|
desc: 'Detach the child process.',
|
||||||
|
default: false,
|
||||||
|
});
|
||||||
|
yargs.option('system', {
|
||||||
|
type: 'boolean',
|
||||||
|
desc: 'Search for a browser installed on the system instead of the cache folder.',
|
||||||
default: false,
|
default: false,
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
async argv => {
|
async argv => {
|
||||||
const args = argv as unknown as LaunchArgs;
|
const args = argv as unknown as LaunchArgs;
|
||||||
const executablePath = computeExecutablePath({
|
const executablePath = args.system
|
||||||
browser: args.browser.name,
|
? computeSystemExecutablePath({
|
||||||
buildId: args.browser.buildId,
|
browser: args.browser.name,
|
||||||
cacheDir: args.path ?? this.#cachePath,
|
// TODO: throw an error if not a ChromeReleaseChannel is provided.
|
||||||
platform: args.platform,
|
channel: args.browser.buildId as ChromeReleaseChannel,
|
||||||
});
|
platform: args.platform,
|
||||||
|
})
|
||||||
|
: computeExecutablePath({
|
||||||
|
browser: args.browser.name,
|
||||||
|
buildId: args.browser.buildId,
|
||||||
|
cacheDir: args.path ?? this.#cachePath,
|
||||||
|
platform: args.platform,
|
||||||
|
});
|
||||||
launch({
|
launch({
|
||||||
executablePath,
|
executablePath,
|
||||||
detached: args.detached,
|
detached: args.detached,
|
||||||
|
@ -16,7 +16,13 @@
|
|||||||
|
|
||||||
import * as chrome from './chrome.js';
|
import * as chrome from './chrome.js';
|
||||||
import * as firefox from './firefox.js';
|
import * as firefox from './firefox.js';
|
||||||
import {Browser, BrowserPlatform, BrowserTag, ProfileOptions} from './types.js';
|
import {
|
||||||
|
Browser,
|
||||||
|
BrowserPlatform,
|
||||||
|
BrowserTag,
|
||||||
|
ChromeReleaseChannel,
|
||||||
|
ProfileOptions,
|
||||||
|
} from './types.js';
|
||||||
|
|
||||||
export const downloadUrls = {
|
export const downloadUrls = {
|
||||||
[Browser.CHROME]: chrome.resolveDownloadUrl,
|
[Browser.CHROME]: chrome.resolveDownloadUrl,
|
||||||
@ -66,3 +72,19 @@ export async function createProfile(
|
|||||||
throw new Error(`Profile creation is not support for ${browser} yet`);
|
throw new Error(`Profile creation is not support for ${browser} yet`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function resolveSystemExecutablePath(
|
||||||
|
browser: Browser,
|
||||||
|
platform: BrowserPlatform,
|
||||||
|
channel: ChromeReleaseChannel
|
||||||
|
): string {
|
||||||
|
switch (browser) {
|
||||||
|
case Browser.FIREFOX:
|
||||||
|
throw new Error(
|
||||||
|
'System browser detection is not supported for Firefox yet.'
|
||||||
|
);
|
||||||
|
case Browser.CHROME:
|
||||||
|
case Browser.CHROMIUM:
|
||||||
|
return chrome.resolveSystemExecutablePath(platform, channel);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -18,7 +18,7 @@ import path from 'path';
|
|||||||
|
|
||||||
import {httpRequest} from '../httpUtil.js';
|
import {httpRequest} from '../httpUtil.js';
|
||||||
|
|
||||||
import {BrowserPlatform} from './types.js';
|
import {BrowserPlatform, ChromeReleaseChannel} from './types.js';
|
||||||
|
|
||||||
function archive(platform: BrowserPlatform, buildId: string): string {
|
function archive(platform: BrowserPlatform, buildId: string): string {
|
||||||
switch (platform) {
|
switch (platform) {
|
||||||
@ -81,7 +81,6 @@ export function relativeExecutablePath(
|
|||||||
return path.join('chrome-win', 'chrome.exe');
|
return path.join('chrome-win', 'chrome.exe');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function resolveBuildId(
|
export async function resolveBuildId(
|
||||||
platform: BrowserPlatform,
|
platform: BrowserPlatform,
|
||||||
// We will need it for other channels/keywords.
|
// We will need it for other channels/keywords.
|
||||||
@ -118,3 +117,48 @@ export async function resolveBuildId(
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function resolveSystemExecutablePath(
|
||||||
|
platform: BrowserPlatform,
|
||||||
|
channel: ChromeReleaseChannel
|
||||||
|
): string {
|
||||||
|
switch (platform) {
|
||||||
|
case BrowserPlatform.WIN64:
|
||||||
|
case BrowserPlatform.WIN32:
|
||||||
|
switch (channel) {
|
||||||
|
case ChromeReleaseChannel.STABLE:
|
||||||
|
return `${process.env['PROGRAMFILES']}\\Google\\Chrome\\Application\\chrome.exe`;
|
||||||
|
case ChromeReleaseChannel.BETA:
|
||||||
|
return `${process.env['PROGRAMFILES']}\\Google\\Chrome Beta\\Application\\chrome.exe`;
|
||||||
|
case ChromeReleaseChannel.CANARY:
|
||||||
|
return `${process.env['PROGRAMFILES']}\\Google\\Chrome SxS\\Application\\chrome.exe`;
|
||||||
|
case ChromeReleaseChannel.DEV:
|
||||||
|
return `${process.env['PROGRAMFILES']}\\Google\\Chrome Dev\\Application\\chrome.exe`;
|
||||||
|
}
|
||||||
|
case BrowserPlatform.MAC_ARM:
|
||||||
|
case BrowserPlatform.MAC:
|
||||||
|
switch (channel) {
|
||||||
|
case ChromeReleaseChannel.STABLE:
|
||||||
|
return '/Applications/Google Chrome.app/Contents/MacOS/Google Chrome';
|
||||||
|
case ChromeReleaseChannel.BETA:
|
||||||
|
return '/Applications/Google Chrome Beta.app/Contents/MacOS/Google Chrome Beta';
|
||||||
|
case ChromeReleaseChannel.CANARY:
|
||||||
|
return '/Applications/Google Chrome Canary.app/Contents/MacOS/Google Chrome Canary';
|
||||||
|
case ChromeReleaseChannel.DEV:
|
||||||
|
return '/Applications/Google Chrome Dev.app/Contents/MacOS/Google Chrome Dev';
|
||||||
|
}
|
||||||
|
case BrowserPlatform.LINUX:
|
||||||
|
switch (channel) {
|
||||||
|
case ChromeReleaseChannel.STABLE:
|
||||||
|
return '/opt/google/chrome/chrome';
|
||||||
|
case ChromeReleaseChannel.BETA:
|
||||||
|
return '/opt/google/chrome-beta/chrome';
|
||||||
|
case ChromeReleaseChannel.DEV:
|
||||||
|
return '/opt/google/chrome-unstable/chrome';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error(
|
||||||
|
`Unable to detect browser executable path for '${channel}' on ${platform}.`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
@ -52,3 +52,10 @@ export interface ProfileOptions {
|
|||||||
preferences: Record<string, unknown>;
|
preferences: Record<string, unknown>;
|
||||||
path: string;
|
path: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export enum ChromeReleaseChannel {
|
||||||
|
STABLE = 'stable',
|
||||||
|
DEV = 'dev',
|
||||||
|
CANARY = 'canary',
|
||||||
|
BETA = 'beta',
|
||||||
|
}
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import childProcess from 'child_process';
|
import childProcess from 'child_process';
|
||||||
|
import {accessSync} from 'fs';
|
||||||
import os from 'os';
|
import os from 'os';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import readline from 'readline';
|
import readline from 'readline';
|
||||||
@ -23,7 +24,9 @@ import {
|
|||||||
Browser,
|
Browser,
|
||||||
BrowserPlatform,
|
BrowserPlatform,
|
||||||
executablePathByBrowser,
|
executablePathByBrowser,
|
||||||
|
resolveSystemExecutablePath,
|
||||||
} from './browsers/browsers.js';
|
} from './browsers/browsers.js';
|
||||||
|
import {ChromeReleaseChannel} from './browsers/types.js';
|
||||||
import {CacheStructure} from './CacheStructure.js';
|
import {CacheStructure} from './CacheStructure.js';
|
||||||
import {debug} from './debug.js';
|
import {debug} from './debug.js';
|
||||||
import {detectBrowserPlatform} from './detectPlatform.js';
|
import {detectBrowserPlatform} from './detectPlatform.js';
|
||||||
@ -49,7 +52,7 @@ export interface Options {
|
|||||||
*/
|
*/
|
||||||
browser: Browser;
|
browser: Browser;
|
||||||
/**
|
/**
|
||||||
* Determines which buildId to dowloand. BuildId should uniquely identify
|
* Determines which buildId to download. BuildId should uniquely identify
|
||||||
* binaries and they are used for caching.
|
* binaries and they are used for caching.
|
||||||
*/
|
*/
|
||||||
buildId: string;
|
buildId: string;
|
||||||
@ -73,6 +76,47 @@ export function computeExecutablePath(options: Options): string {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
export interface SystemOptions {
|
||||||
|
/**
|
||||||
|
* Determines which platform the browser will be suited for.
|
||||||
|
*
|
||||||
|
* @defaultValue Auto-detected.
|
||||||
|
*/
|
||||||
|
platform?: BrowserPlatform;
|
||||||
|
/**
|
||||||
|
* Determines which browser to fetch.
|
||||||
|
*/
|
||||||
|
browser: Browser;
|
||||||
|
/**
|
||||||
|
* Release channel to look for on the system.
|
||||||
|
*/
|
||||||
|
channel: ChromeReleaseChannel;
|
||||||
|
}
|
||||||
|
export function computeSystemExecutablePath(options: SystemOptions): string {
|
||||||
|
options.platform ??= detectBrowserPlatform();
|
||||||
|
if (!options.platform) {
|
||||||
|
throw new Error(
|
||||||
|
`Cannot download a binary for the provided platform: ${os.platform()} (${os.arch()})`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
const path = resolveSystemExecutablePath(
|
||||||
|
options.browser,
|
||||||
|
options.platform,
|
||||||
|
options.channel
|
||||||
|
);
|
||||||
|
try {
|
||||||
|
accessSync(path);
|
||||||
|
} catch (error) {
|
||||||
|
throw new Error(
|
||||||
|
`Could not find Google Chrome executable for channel '${options.channel}' at '${path}'.`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
type LaunchOptions = {
|
type LaunchOptions = {
|
||||||
executablePath: string;
|
executablePath: string;
|
||||||
pipe?: boolean;
|
pipe?: boolean;
|
||||||
|
26
packages/browsers/src/main.ts
Normal file
26
packages/browsers/src/main.ts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
export {
|
||||||
|
launch,
|
||||||
|
computeExecutablePath,
|
||||||
|
computeSystemExecutablePath,
|
||||||
|
CDP_WEBSOCKET_ENDPOINT_REGEX,
|
||||||
|
WEBDRIVER_BIDI_WEBSOCKET_ENDPOINT_REGEX,
|
||||||
|
} from './launcher.js';
|
||||||
|
export {fetch, canFetch} from './fetch.js';
|
||||||
|
export {detectBrowserPlatform} from './detectPlatform.js';
|
||||||
|
export {Browser, BrowserPlatform} from './browsers/browsers.js';
|
@ -21,7 +21,9 @@ import {BrowserPlatform} from '../../lib/cjs/browsers/browsers.js';
|
|||||||
import {
|
import {
|
||||||
resolveDownloadUrl,
|
resolveDownloadUrl,
|
||||||
relativeExecutablePath,
|
relativeExecutablePath,
|
||||||
|
resolveSystemExecutablePath,
|
||||||
} from '../../lib/cjs/browsers/chrome.js';
|
} from '../../lib/cjs/browsers/chrome.js';
|
||||||
|
import {ChromeReleaseChannel} from '../../lib/cjs/browsers/types.js';
|
||||||
|
|
||||||
describe('Chrome', () => {
|
describe('Chrome', () => {
|
||||||
it('should resolve download URLs', () => {
|
it('should resolve download URLs', () => {
|
||||||
@ -69,4 +71,36 @@ describe('Chrome', () => {
|
|||||||
path.join('chrome-win', 'chrome.exe')
|
path.join('chrome-win', 'chrome.exe')
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should resolve system executable path', () => {
|
||||||
|
process.env['PROGRAMFILES'] = 'C:\\ProgramFiles';
|
||||||
|
try {
|
||||||
|
assert.strictEqual(
|
||||||
|
resolveSystemExecutablePath(
|
||||||
|
BrowserPlatform.WIN32,
|
||||||
|
ChromeReleaseChannel.DEV
|
||||||
|
),
|
||||||
|
'C:\\ProgramFiles\\Google\\Chrome Dev\\Application\\chrome.exe'
|
||||||
|
);
|
||||||
|
} finally {
|
||||||
|
delete process.env['PROGRAMFILES'];
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.strictEqual(
|
||||||
|
resolveSystemExecutablePath(
|
||||||
|
BrowserPlatform.MAC,
|
||||||
|
ChromeReleaseChannel.BETA
|
||||||
|
),
|
||||||
|
'/Applications/Google Chrome Beta.app/Contents/MacOS/Google Chrome Beta'
|
||||||
|
);
|
||||||
|
assert.throws(() => {
|
||||||
|
assert.strictEqual(
|
||||||
|
resolveSystemExecutablePath(
|
||||||
|
BrowserPlatform.LINUX,
|
||||||
|
ChromeReleaseChannel.CANARY
|
||||||
|
),
|
||||||
|
path.join('chrome-linux', 'chrome')
|
||||||
|
);
|
||||||
|
}, new Error(`Unable to detect browser executable path for 'canary' on linux.`));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -92,7 +92,7 @@ describe('fetch', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should download a buildId that is a bzip2 archive', async function () {
|
it('should download a buildId that is a bzip2 archive', async function () {
|
||||||
this.timeout(60000);
|
this.timeout(90000);
|
||||||
const expectedOutputPath = path.join(
|
const expectedOutputPath = path.join(
|
||||||
tmpDir,
|
tmpDir,
|
||||||
'firefox',
|
'firefox',
|
||||||
|
Loading…
Reference in New Issue
Block a user