appium-chromedriver
Version:
Node.js wrapper around chromedriver.
173 lines (166 loc) • 5.46 kB
text/typescript
import path from 'node:path';
import {logger} from '@appium/support';
import * as semver from 'semver';
import {ARCH, CPU} from '../constants';
import type {ChromedriverDetailsMapping} from '../types';
interface VersionEntry {
version: string;
revision?: string;
downloads?: {
chromedriver?: Array<{
platform: string;
url: string;
}>;
};
}
interface KnownGoodVersionsJson {
timestamp?: string;
versions?: VersionEntry[];
}
interface LatestKnownGoodVersionsJson {
timestamp?: string;
channels?: {
Stable?: {
channel?: string;
version?: string;
revision?: string;
};
};
}
const log = logger.getLogger('ChromedriverChromelabsStorageClient');
/**
* Parses the output of the JSON API that retrieves Chromedriver versions
*
* See https://github.com/GoogleChromeLabs/chrome-for-testing#json-api-endpoints for more details.
*
* @param jsonStr - The JSON string from the known-good-versions-with-downloads API
* @returns A mapping of chromedriver entry keys to their details
* @throws {Error} if the JSON cannot be parsed or has an unsupported format
*/
export function parseKnownGoodVersionsWithDownloadsJson(
jsonStr: string,
): ChromedriverDetailsMapping {
let json: KnownGoodVersionsJson;
try {
json = JSON.parse(jsonStr);
} catch (e) {
throw new Error(
`Storage JSON cannot be parsed. Original error: ${e instanceof Error ? e.message : String(e)}`,
{cause: e},
);
}
/**
* Example output:
* {
* "timestamp":"2023-07-28T13:09:17.042Z",
* "versions":[
* {
* "version":"113.0.5672.0",
* "revision":"1121455",
* "downloads":{
* "chromedriver":[
* {
* "platform":"linux64",
* "url":"https://edgedl.me.gvt1.com/edgedl/chrome/chrome-for-testing/113.0.5672.0/linux64/chrome-linux64.zip"
* },
* {
* "platform":"mac-arm64",
* "url":"https://edgedl.me.gvt1.com/edgedl/chrome/chrome-for-testing/113.0.5672.0/mac-arm64/chrome-mac-arm64.zip"
* },
* {
* "platform":"mac-x64",
* "url":"https://edgedl.me.gvt1.com/edgedl/chrome/chrome-for-testing/113.0.5672.0/mac-x64/chrome-mac-x64.zip"
* },
* {
* "platform":"win32",
* "url":"https://edgedl.me.gvt1.com/edgedl/chrome/chrome-for-testing/113.0.5672.0/win32/chrome-win32.zip"
* },
* {
* "platform":"win64",
* "url":"https://edgedl.me.gvt1.com/edgedl/chrome/chrome-for-testing/113.0.5672.0/win64/chrome-win64.zip"
* }
* ]
* }
* },
* {
* "version":"113.0.5672.35",
* ...
*/
const mapping: ChromedriverDetailsMapping = {};
if (!Array.isArray(json?.versions)) {
log.debug(jsonStr);
throw new Error('The format of the storage JSON is not supported');
}
for (const {version, downloads} of json.versions) {
if (!Array.isArray(downloads?.chromedriver)) {
continue;
}
const versionObj = semver.parse(version, {loose: true});
if (!versionObj) {
continue;
}
for (const downloadEntry of downloads.chromedriver) {
if (!downloadEntry?.url || !downloadEntry?.platform) {
continue;
}
const osNameMatch = /^[a-z]+/i.exec(downloadEntry.platform);
if (!osNameMatch) {
log.debug(
`The entry '${downloadEntry.url}' does not contain valid platform name. Skipping it`,
);
continue;
}
const key =
`${path.basename(path.dirname(path.dirname(downloadEntry.url)))}/` +
`${path.basename(downloadEntry.url)}`;
mapping[key] = {
url: downloadEntry.url,
etag: null,
version,
minBrowserVersion: `${versionObj.major}`,
os: {
name: osNameMatch[0],
arch: downloadEntry.platform.includes(ARCH.X64) ? ARCH.X64 : ARCH.X86,
cpu: downloadEntry.platform.includes(CPU.ARM) ? CPU.ARM : CPU.INTEL,
},
};
}
}
log.info(`The total count of entries in the mapping: ${Object.keys(mapping).length}`);
return mapping;
}
/**
* Parses the output of the JSON API that retrieves the most recent stable Chromedriver version
*
* See https://github.com/GoogleChromeLabs/chrome-for-testing#json-api-endpoints for more details.
*
* @param jsonStr - The JSON string from the last-known-good-versions API
* @returns The most recent available chromedriver version string
* @throws {Error} if the JSON cannot be parsed or has an unsupported format
*/
export function parseLatestKnownGoodVersionsJson(jsonStr: string): string {
let json: LatestKnownGoodVersionsJson;
try {
json = JSON.parse(jsonStr);
} catch (e) {
throw new Error(
`Storage JSON cannot be parsed. Original error: ${e instanceof Error ? e.message : String(e)}`,
{cause: e},
);
}
/**
* Example output:
* "timestamp":"2023-07-28T13:09:17.036Z",
* "channels":{
* "Stable":{
* "channel":"Stable",
* "version":"115.0.5790.102",
* "revision":"1148114"
* ...
*/
if (!json?.channels?.Stable?.version) {
log.debug(jsonStr);
throw new Error('The format of the storage JSON is not supported');
}
return json.channels.Stable.version;
}