refactor: unify working with the cache folder structure (#9681)

This commit is contained in:
Alex Rudenko 2023-02-15 20:41:22 +01:00 committed by GitHub
parent cfb60d01a8
commit 0b27cbe86d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 168 additions and 106 deletions

View File

@ -1,9 +1,24 @@
/**
* 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.
*/
import yargs from 'yargs'; import yargs from 'yargs';
import ProgressBar from 'progress'; import ProgressBar from 'progress';
import {hideBin} from 'yargs/helpers'; import {hideBin} from 'yargs/helpers';
import {Browser, BrowserPlatform} from './browsers/types.js'; import {Browser, BrowserPlatform} from './browsers/types.js';
import {fetch} from './fetch.js'; import {fetch} from './fetch.js';
import path from 'path';
type Arguments = { type Arguments = {
browser: { browser: {
@ -44,11 +59,8 @@ export class CLI {
browser: args.browser.name, browser: args.browser.name,
revision: args.browser.revision, revision: args.browser.revision,
platform: args.platform, platform: args.platform,
outputDir: path.join( cacheDir: args.path ?? this.#cachePath,
args.path ?? this.#cachePath, downloadProgressCallback: this.#makeProgressCallback(
args.browser.name
),
progressCallback: this.#makeProgressBar(
args.browser.name, args.browser.name,
args.browser.revision args.browser.revision
), ),
@ -82,8 +94,8 @@ export class CLI {
return `${Math.round(mb * 10) / 10} Mb`; return `${Math.round(mb * 10) / 10} Mb`;
} }
#makeProgressBar(browser: Browser, revision: string) { #makeProgressCallback(browser: Browser, revision: string) {
let progressBar: ProgressBar | null = null; let progressBar: ProgressBar;
let lastDownloadedBytes = 0; let lastDownloadedBytes = 0;
return (downloadedBytes: number, totalBytes: number) => { return (downloadedBytes: number, totalBytes: number) => {
if (!progressBar) { if (!progressBar) {

View File

@ -0,0 +1,52 @@
/**
* 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.
*/
import {Browser, BrowserPlatform} from './browsers/types.js';
import path from 'path';
/**
* The cache used by Puppeteer relies on the following structure:
*
* - rootDir
* -- <browser1> | browserRoot(browser1)
* ---- <platform>-<revision> | installationDir()
* ------ the browser-platform-revision
* ------ specific structure.
* -- <browser2> | browserRoot(browser2)
* ---- <platform>-<revision> | installationDir()
* ------ the browser-platform-revision
* ------ specific structure.
* @internal
*/
export class CacheStructure {
#rootDir: string;
constructor(rootDir: string) {
this.#rootDir = rootDir;
}
browserRoot(browser: Browser): string {
return path.join(this.#rootDir, browser);
}
installationDir(
browser: Browser,
platform: BrowserPlatform,
revision: string
): string {
return path.join(this.browserRoot(browser), `${platform}-${revision}`);
}
}

View File

@ -1,5 +1,21 @@
#!/usr/bin/env node #!/usr/bin/env node
/**
* 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.
*/
import {CLI} from './CLI.js'; import {CLI} from './CLI.js';
new CLI().run(process.argv); new CLI().run(process.argv);

View File

@ -24,8 +24,8 @@ export const downloadUrls = {
}; };
export const executablePathByBrowser = { export const executablePathByBrowser = {
[Browser.CHROME]: chrome.executablePath, [Browser.CHROME]: chrome.relativeExecutablePath,
[Browser.FIREFOX]: firefox.executablePath, [Browser.FIREFOX]: firefox.relativeExecutablePath,
}; };
export {Browser, BrowserPlatform}; export {Browser, BrowserPlatform};

View File

@ -57,26 +57,18 @@ export function resolveDownloadUrl(
)}.zip`; )}.zip`;
} }
export function executablePath( export function relativeExecutablePath(
platform: BrowserPlatform, platform: BrowserPlatform,
revision: string, _revision: string
basePath = ''
): string { ): string {
const browserPath = path.join(basePath, `${platform}-${revision}`);
switch (platform) { switch (platform) {
case BrowserPlatform.MAC: case BrowserPlatform.MAC:
case BrowserPlatform.MAC_ARM: case BrowserPlatform.MAC_ARM:
return path.join( return path.join('Chromium.app', 'Contents', 'MacOS', 'Chromium');
browserPath,
'Chromium.app',
'Contents',
'MacOS',
'Chromium'
);
case BrowserPlatform.LINUX: case BrowserPlatform.LINUX:
return path.join(browserPath, 'chrome'); return 'chrome';
case BrowserPlatform.WIN32: case BrowserPlatform.WIN32:
case BrowserPlatform.WIN64: case BrowserPlatform.WIN64:
return path.join(browserPath, 'chrome.exe'); return 'chrome.exe';
} }
} }

View File

@ -38,26 +38,18 @@ export function resolveDownloadUrl(
return `${baseUrl}/${archive(platform, revision)}`; return `${baseUrl}/${archive(platform, revision)}`;
} }
export function executablePath( export function relativeExecutablePath(
platform: BrowserPlatform, platform: BrowserPlatform,
revision: string, _revision: string
basePath = ''
): string { ): string {
const browserPath = path.join(basePath, `${platform}-${revision}`);
switch (platform) { switch (platform) {
case BrowserPlatform.MAC_ARM: case BrowserPlatform.MAC_ARM:
case BrowserPlatform.MAC: case BrowserPlatform.MAC:
return path.join( return path.join('Firefox Nightly.app', 'Contents', 'MacOS', 'firefox');
browserPath,
'Firefox Nightly.app',
'Contents',
'MacOS',
'firefox'
);
case BrowserPlatform.LINUX: case BrowserPlatform.LINUX:
return path.join(browserPath, 'firefox', 'firefox'); return path.join('firefox', 'firefox');
case BrowserPlatform.WIN32: case BrowserPlatform.WIN32:
case BrowserPlatform.WIN64: case BrowserPlatform.WIN64:
return path.join(browserPath, 'firefox', 'firefox.exe'); return path.join('firefox', 'firefox.exe');
} }
} }

View File

@ -17,7 +17,7 @@
import os from 'os'; import os from 'os';
import {BrowserPlatform} from './browsers/browsers.js'; import {BrowserPlatform} from './browsers/browsers.js';
export function detectPlatform(): BrowserPlatform | undefined { export function detectBrowserPlatform(): BrowserPlatform | undefined {
const platform = os.platform(); const platform = os.platform();
switch (platform) { switch (platform) {
case 'darwin': case 'darwin':

View File

@ -24,7 +24,8 @@ import {Browser, BrowserPlatform, downloadUrls} from './browsers/browsers.js';
import {downloadFile, headHttpRequest} from './httpUtil.js'; import {downloadFile, headHttpRequest} from './httpUtil.js';
import assert from 'assert'; import assert from 'assert';
import {unpackArchive} from './fileUtil.js'; import {unpackArchive} from './fileUtil.js';
import {detectPlatform} from './detectPlatform.js'; import {detectBrowserPlatform} from './detectPlatform.js';
import {CacheStructure} from './CacheStructure.js';
const debugFetch = debug('puppeteer:browsers:fetcher'); const debugFetch = debug('puppeteer:browsers:fetcher');
@ -35,7 +36,7 @@ export interface Options {
/** /**
* Determines the path to download browsers to. * Determines the path to download browsers to.
*/ */
outputDir: string; cacheDir: string;
/** /**
* Determines which platform the browser will be suited for. * Determines which platform the browser will be suited for.
* *
@ -54,7 +55,10 @@ export interface Options {
/** /**
* Provides information about the progress of the download. * Provides information about the progress of the download.
*/ */
progressCallback?: (downloadedBytes: number, totalBytes: number) => void; downloadProgressCallback?: (
downloadedBytes: number,
totalBytes: number
) => void;
} }
export type InstalledBrowser = { export type InstalledBrowser = {
@ -65,7 +69,7 @@ export type InstalledBrowser = {
}; };
export async function fetch(options: Options): Promise<InstalledBrowser> { export async function fetch(options: Options): Promise<InstalledBrowser> {
options.platform ??= detectPlatform(); options.platform ??= detectBrowserPlatform();
if (!options.platform) { if (!options.platform) {
throw new Error( throw new Error(
`Cannot download a binary for the provided platform: ${os.platform()} (${os.arch()})` `Cannot download a binary for the provided platform: ${os.platform()} (${os.arch()})`
@ -78,10 +82,16 @@ export async function fetch(options: Options): Promise<InstalledBrowser> {
); );
const fileName = url.toString().split('/').pop(); const fileName = url.toString().split('/').pop();
assert(fileName, `A malformed download URL was found: ${url}.`); assert(fileName, `A malformed download URL was found: ${url}.`);
const archivePath = path.join(options.outputDir, fileName); const structure = new CacheStructure(options.cacheDir);
const outputPath = path.resolve( const browserRoot = structure.browserRoot(options.browser);
options.outputDir, const archivePath = path.join(browserRoot, fileName);
`${options.platform}-${options.revision}` if (!existsSync(browserRoot)) {
await mkdir(browserRoot, {recursive: true});
}
const outputPath = structure.installationDir(
options.browser,
options.platform,
options.revision
); );
if (existsSync(outputPath)) { if (existsSync(outputPath)) {
return { return {
@ -91,12 +101,9 @@ export async function fetch(options: Options): Promise<InstalledBrowser> {
revision: options.revision, revision: options.revision,
}; };
} }
if (!existsSync(options.outputDir)) {
await mkdir(options.outputDir, {recursive: true});
}
try { try {
debugFetch(`Downloading binary from ${url}`); debugFetch(`Downloading binary from ${url}`);
await downloadFile(url, archivePath, options.progressCallback); await downloadFile(url, archivePath, options.downloadProgressCallback);
debugFetch(`Installing ${archivePath} to ${outputPath}`); debugFetch(`Installing ${archivePath} to ${outputPath}`);
await unpackArchive(archivePath, outputPath); await unpackArchive(archivePath, outputPath);
} finally { } finally {
@ -113,7 +120,7 @@ export async function fetch(options: Options): Promise<InstalledBrowser> {
} }
export async function canFetch(options: Options): Promise<boolean> { export async function canFetch(options: Options): Promise<boolean> {
options.platform ??= detectPlatform(); options.platform ??= detectBrowserPlatform();
if (!options.platform) { if (!options.platform) {
throw new Error( throw new Error(
`Cannot download a binary for the provided platform: ${os.platform()} (${os.arch()})` `Cannot download a binary for the provided platform: ${os.platform()} (${os.arch()})`

View File

@ -19,8 +19,10 @@ import {
BrowserPlatform, BrowserPlatform,
executablePathByBrowser, executablePathByBrowser,
} from './browsers/browsers.js'; } from './browsers/browsers.js';
import {detectPlatform} from './detectPlatform.js'; import {detectBrowserPlatform} from './detectPlatform.js';
import os from 'os'; import os from 'os';
import path from 'path';
import {CacheStructure} from './CacheStructure.js';
/** /**
* @public * @public
@ -29,7 +31,7 @@ export interface Options {
/** /**
* Root path to the storage directory. * Root path to the storage directory.
*/ */
path: string; cacheDir: string;
/** /**
* Determines which platform the browser will be suited for. * Determines which platform the browser will be suited for.
* *
@ -48,15 +50,19 @@ export interface Options {
} }
export function computeExecutablePath(options: Options): string { export function computeExecutablePath(options: Options): string {
options.platform ??= detectPlatform(); options.platform ??= detectBrowserPlatform();
if (!options.platform) { if (!options.platform) {
throw new Error( throw new Error(
`Cannot download a binary for the provided platform: ${os.platform()} (${os.arch()})` `Cannot download a binary for the provided platform: ${os.platform()} (${os.arch()})`
); );
} }
return executablePathByBrowser[options.browser]( const installationDir = new CacheStructure(options.cacheDir).installationDir(
options.browser,
options.platform, options.platform,
options.revision, options.revision
options.path );
return path.join(
installationDir,
executablePathByBrowser[options.browser](options.platform, options.revision)
); );
} }

View File

@ -16,7 +16,7 @@
import { import {
resolveDownloadUrl, resolveDownloadUrl,
executablePath, relativeExecutablePath,
} from '../../lib/cjs/browsers/chrome.js'; } from '../../lib/cjs/browsers/chrome.js';
import {BrowserPlatform} from '../../lib/cjs/browsers/browsers.js'; import {BrowserPlatform} from '../../lib/cjs/browsers/browsers.js';
import assert from 'assert'; import assert from 'assert';
@ -48,30 +48,24 @@ describe('Chrome', () => {
it('should resolve executable paths', () => { it('should resolve executable paths', () => {
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.LINUX, '12372323'), relativeExecutablePath(BrowserPlatform.LINUX, '12372323'),
path.join('linux-12372323', 'chrome') path.join('chrome')
); );
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.MAC, '12372323'), relativeExecutablePath(BrowserPlatform.MAC, '12372323'),
path.join('mac-12372323', 'Chromium.app', 'Contents', 'MacOS', 'Chromium') path.join('Chromium.app', 'Contents', 'MacOS', 'Chromium')
); );
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.MAC_ARM, '12372323'), relativeExecutablePath(BrowserPlatform.MAC_ARM, '12372323'),
path.join( path.join('Chromium.app', 'Contents', 'MacOS', 'Chromium')
'mac_arm-12372323',
'Chromium.app',
'Contents',
'MacOS',
'Chromium'
)
); );
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.WIN32, '12372323'), relativeExecutablePath(BrowserPlatform.WIN32, '12372323'),
path.join('win32-12372323', 'chrome.exe') path.join('chrome.exe')
); );
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.WIN64, '12372323'), relativeExecutablePath(BrowserPlatform.WIN64, '12372323'),
path.join('win64-12372323', 'chrome.exe') path.join('chrome.exe')
); );
}); });
}); });

View File

@ -42,7 +42,7 @@ describe('fetch', () => {
it('should check if a revision can be downloaded', async () => { it('should check if a revision can be downloaded', async () => {
assert.ok( assert.ok(
await canFetch({ await canFetch({
outputDir: tmpDir, cacheDir: tmpDir,
browser: Browser.CHROME, browser: Browser.CHROME,
platform: BrowserPlatform.LINUX, platform: BrowserPlatform.LINUX,
revision: testChromeRevision, revision: testChromeRevision,
@ -53,7 +53,7 @@ describe('fetch', () => {
it('should report if a revision is not downloadable', async () => { it('should report if a revision is not downloadable', async () => {
assert.strictEqual( assert.strictEqual(
await canFetch({ await canFetch({
outputDir: tmpDir, cacheDir: tmpDir,
browser: Browser.CHROME, browser: Browser.CHROME,
platform: BrowserPlatform.LINUX, platform: BrowserPlatform.LINUX,
revision: 'unknown', revision: 'unknown',
@ -66,11 +66,12 @@ describe('fetch', () => {
this.timeout(60000); this.timeout(60000);
const expectedOutputPath = path.join( const expectedOutputPath = path.join(
tmpDir, tmpDir,
'chrome',
`${BrowserPlatform.LINUX}-${testChromeRevision}` `${BrowserPlatform.LINUX}-${testChromeRevision}`
); );
assert.strictEqual(fs.existsSync(expectedOutputPath), false); assert.strictEqual(fs.existsSync(expectedOutputPath), false);
let browser = await fetch({ let browser = await fetch({
outputDir: tmpDir, cacheDir: tmpDir,
browser: Browser.CHROME, browser: Browser.CHROME,
platform: BrowserPlatform.LINUX, platform: BrowserPlatform.LINUX,
revision: testChromeRevision, revision: testChromeRevision,
@ -79,7 +80,7 @@ describe('fetch', () => {
assert.ok(fs.existsSync(expectedOutputPath)); assert.ok(fs.existsSync(expectedOutputPath));
// Second iteration should be no-op. // Second iteration should be no-op.
browser = await fetch({ browser = await fetch({
outputDir: tmpDir, cacheDir: tmpDir,
browser: Browser.CHROME, browser: Browser.CHROME,
platform: BrowserPlatform.LINUX, platform: BrowserPlatform.LINUX,
revision: testChromeRevision, revision: testChromeRevision,
@ -92,11 +93,12 @@ describe('fetch', () => {
this.timeout(60000); this.timeout(60000);
const expectedOutputPath = path.join( const expectedOutputPath = path.join(
tmpDir, tmpDir,
'firefox',
`${BrowserPlatform.LINUX}-${testFirefoxRevision}` `${BrowserPlatform.LINUX}-${testFirefoxRevision}`
); );
assert.strictEqual(fs.existsSync(expectedOutputPath), false); assert.strictEqual(fs.existsSync(expectedOutputPath), false);
const browser = await fetch({ const browser = await fetch({
outputDir: tmpDir, cacheDir: tmpDir,
browser: Browser.FIREFOX, browser: Browser.FIREFOX,
platform: BrowserPlatform.LINUX, platform: BrowserPlatform.LINUX,
revision: testFirefoxRevision, revision: testFirefoxRevision,
@ -113,11 +115,12 @@ describe('fetch', () => {
this.timeout(120000); this.timeout(120000);
const expectedOutputPath = path.join( const expectedOutputPath = path.join(
tmpDir, tmpDir,
'firefox',
`${BrowserPlatform.MAC}-${testFirefoxRevision}` `${BrowserPlatform.MAC}-${testFirefoxRevision}`
); );
assert.strictEqual(fs.existsSync(expectedOutputPath), false); assert.strictEqual(fs.existsSync(expectedOutputPath), false);
const browser = await fetch({ const browser = await fetch({
outputDir: tmpDir, cacheDir: tmpDir,
browser: Browser.FIREFOX, browser: Browser.FIREFOX,
platform: BrowserPlatform.MAC, platform: BrowserPlatform.MAC,
revision: testFirefoxRevision, revision: testFirefoxRevision,

View File

@ -15,7 +15,7 @@
*/ */
import { import {
executablePath, relativeExecutablePath,
resolveDownloadUrl, resolveDownloadUrl,
} from '../../lib/cjs/browsers/firefox.js'; } from '../../lib/cjs/browsers/firefox.js';
import {BrowserPlatform} from '../../lib/cjs/browsers/browsers.js'; import {BrowserPlatform} from '../../lib/cjs/browsers/browsers.js';
@ -48,36 +48,24 @@ describe('Firefox', () => {
it('should resolve executable paths', () => { it('should resolve executable paths', () => {
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.LINUX, '111.0a1'), relativeExecutablePath(BrowserPlatform.LINUX, '111.0a1'),
path.join('linux-111.0a1', 'firefox', 'firefox') path.join('firefox', 'firefox')
); );
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.MAC, '111.0a1'), relativeExecutablePath(BrowserPlatform.MAC, '111.0a1'),
path.join( path.join('Firefox Nightly.app', 'Contents', 'MacOS', 'firefox')
'mac-111.0a1',
'Firefox Nightly.app',
'Contents',
'MacOS',
'firefox'
)
); );
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.MAC_ARM, '111.0a1'), relativeExecutablePath(BrowserPlatform.MAC_ARM, '111.0a1'),
path.join( path.join('Firefox Nightly.app', 'Contents', 'MacOS', 'firefox')
'mac_arm-111.0a1',
'Firefox Nightly.app',
'Contents',
'MacOS',
'firefox'
)
); );
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.WIN32, '111.0a1'), relativeExecutablePath(BrowserPlatform.WIN32, '111.0a1'),
path.join('win32-111.0a1', 'firefox', 'firefox.exe') path.join('firefox', 'firefox.exe')
); );
assert.strictEqual( assert.strictEqual(
executablePath(BrowserPlatform.WIN64, '111.0a1'), relativeExecutablePath(BrowserPlatform.WIN64, '111.0a1'),
path.join('win64-111.0a1', 'firefox', 'firefox.exe') path.join('firefox', 'firefox.exe')
); );
}); });
}); });

View File

@ -27,9 +27,9 @@ describe('launcher', () => {
browser: Browser.CHROME, browser: Browser.CHROME,
platform: BrowserPlatform.LINUX, platform: BrowserPlatform.LINUX,
revision: '123', revision: '123',
path: 'cache', cacheDir: 'cache',
}), }),
path.join('cache', 'linux-123', 'chrome') path.join('cache', 'chrome', 'linux-123', 'chrome')
); );
}); });
it('should compute executable path for Firefox', () => { it('should compute executable path for Firefox', () => {
@ -38,9 +38,9 @@ describe('launcher', () => {
browser: Browser.FIREFOX, browser: Browser.FIREFOX,
platform: BrowserPlatform.LINUX, platform: BrowserPlatform.LINUX,
revision: '123', revision: '123',
path: 'cache', cacheDir: 'cache',
}), }),
path.join('cache', 'linux-123', 'firefox', 'firefox') path.join('cache', 'firefox', 'linux-123', 'firefox', 'firefox')
); );
}); });
}); });