appium-chromedriver 5.2.4 → 5.2.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +1 -0
- package/build/index.js +1 -6
- package/build/lib/chromedriver.js +5 -138
- package/build/lib/chromedriver.js.map +1 -1
- package/build/lib/install.js +1 -13
- package/build/lib/install.js.map +1 -1
- package/build/lib/protocol-helpers.js +1 -10
- package/build/lib/protocol-helpers.js.map +1 -1
- package/build/lib/storage-client.js +1 -85
- package/build/lib/storage-client.js.map +1 -1
- package/build/lib/utils.js +1 -37
- package/build/lib/utils.js.map +1 -1
- package/config/mapping.json +1 -0
- package/package.json +4 -2
|
@@ -1,78 +1,55 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
|
|
3
3
|
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
-
|
|
5
4
|
Object.defineProperty(exports, "__esModule", {
|
|
6
5
|
value: true
|
|
7
6
|
});
|
|
8
7
|
exports.default = void 0;
|
|
9
|
-
|
|
10
8
|
require("source-map-support/register");
|
|
11
|
-
|
|
12
9
|
var _utils = require("./utils");
|
|
13
|
-
|
|
14
10
|
var _lodash = _interopRequireDefault(require("lodash"));
|
|
15
|
-
|
|
16
11
|
var _xpath = _interopRequireDefault(require("xpath"));
|
|
17
|
-
|
|
18
12
|
var _xmldom = require("@xmldom/xmldom");
|
|
19
|
-
|
|
20
13
|
var _bluebird = _interopRequireDefault(require("bluebird"));
|
|
21
|
-
|
|
22
14
|
var _path = _interopRequireDefault(require("path"));
|
|
23
|
-
|
|
24
15
|
var _support = require("appium/support");
|
|
25
|
-
|
|
26
16
|
const TIMEOUT_MS = 15000;
|
|
27
17
|
const MAX_PARALLEL_DOWNLOADS = 5;
|
|
28
|
-
|
|
29
18
|
const log = _support.logger.getLogger('ChromedriverStorageClient');
|
|
30
|
-
|
|
31
19
|
async function isCrcOk(src, checksum) {
|
|
32
20
|
const md5 = await _support.fs.hash(src, 'md5');
|
|
33
21
|
return _lodash.default.toLower(md5) === _lodash.default.toLower(checksum);
|
|
34
22
|
}
|
|
35
|
-
|
|
36
23
|
function findChildNode(parent, childName = null, text = null) {
|
|
37
24
|
if (!childName && !text) {
|
|
38
25
|
return null;
|
|
39
26
|
}
|
|
40
|
-
|
|
41
27
|
if (!parent.hasChildNodes()) {
|
|
42
28
|
return null;
|
|
43
29
|
}
|
|
44
|
-
|
|
45
30
|
for (let childNodeIdx = 0; childNodeIdx < parent.childNodes.length; childNodeIdx++) {
|
|
46
31
|
const childNode = parent.childNodes[childNodeIdx];
|
|
47
|
-
|
|
48
32
|
if (childName && !text && childName === childNode.localName) {
|
|
49
33
|
return childNode;
|
|
50
34
|
}
|
|
51
|
-
|
|
52
35
|
if (text) {
|
|
53
36
|
const childText = extractNodeText(childNode);
|
|
54
|
-
|
|
55
37
|
if (!childText) {
|
|
56
38
|
continue;
|
|
57
39
|
}
|
|
58
|
-
|
|
59
40
|
if (childName && childName === childNode.localName && text === childText) {
|
|
60
41
|
return childNode;
|
|
61
42
|
}
|
|
62
|
-
|
|
63
43
|
if (!childName && text === childText) {
|
|
64
44
|
return childNode;
|
|
65
45
|
}
|
|
66
46
|
}
|
|
67
47
|
}
|
|
68
|
-
|
|
69
48
|
return null;
|
|
70
49
|
}
|
|
71
|
-
|
|
72
50
|
function extractNodeText(node) {
|
|
73
51
|
return !node || !node.firstChild || !_support.util.hasValue(node.firstChild.nodeValue) ? null : node.firstChild.nodeValue;
|
|
74
52
|
}
|
|
75
|
-
|
|
76
53
|
class ChromedriverStorageClient {
|
|
77
54
|
constructor(args = {}) {
|
|
78
55
|
const {
|
|
@@ -87,17 +64,13 @@ class ChromedriverStorageClient {
|
|
|
87
64
|
parseNotes(content) {
|
|
88
65
|
const result = {};
|
|
89
66
|
const versionMatch = /^\s*[-]+ChromeDriver[\D]+([\d.]+)/im.exec(content);
|
|
90
|
-
|
|
91
67
|
if (versionMatch) {
|
|
92
68
|
result.version = versionMatch[1];
|
|
93
69
|
}
|
|
94
|
-
|
|
95
70
|
const minBrowserVersionMatch = /^\s*Supports Chrome[\D]+(\d+)/im.exec(content);
|
|
96
|
-
|
|
97
71
|
if (minBrowserVersionMatch) {
|
|
98
72
|
result.minBrowserVersion = minBrowserVersionMatch[1];
|
|
99
73
|
}
|
|
100
|
-
|
|
101
74
|
return result;
|
|
102
75
|
}
|
|
103
76
|
|
|
@@ -111,40 +84,30 @@ class ChromedriverStorageClient {
|
|
|
111
84
|
const {
|
|
112
85
|
minBrowserVersion
|
|
113
86
|
} = this.parseNotes(notes);
|
|
114
|
-
|
|
115
87
|
if (!minBrowserVersion) {
|
|
116
88
|
log.debug(`The driver '${driverKey}' does not contain valid release notes at ${notesUrl}. ` + `Skipping it`);
|
|
117
89
|
return;
|
|
118
90
|
}
|
|
119
|
-
|
|
120
91
|
infoDict.minBrowserVersion = minBrowserVersion;
|
|
121
92
|
}
|
|
122
93
|
|
|
123
94
|
async parseStorageXml(doc, shouldParseNotes = true) {
|
|
124
95
|
const driverNodes = _xpath.default.select(`//*[local-name(.)='Contents']`, doc);
|
|
125
|
-
|
|
126
96
|
log.debug(`Parsed ${driverNodes.length} entries from storage XML`);
|
|
127
|
-
|
|
128
97
|
if (_lodash.default.isEmpty(driverNodes)) {
|
|
129
98
|
return;
|
|
130
99
|
}
|
|
131
|
-
|
|
132
100
|
const promises = [];
|
|
133
|
-
|
|
134
101
|
for (const driverNode of driverNodes) {
|
|
135
102
|
const key = extractNodeText(findChildNode(driverNode, 'Key'));
|
|
136
|
-
|
|
137
103
|
if (!_lodash.default.includes(key, '/chromedriver_')) {
|
|
138
104
|
continue;
|
|
139
105
|
}
|
|
140
|
-
|
|
141
106
|
const etag = extractNodeText(findChildNode(driverNode, 'ETag'));
|
|
142
|
-
|
|
143
107
|
if (!etag) {
|
|
144
108
|
log.debug(`The entry '${key}' does not contain the checksum. Skipping it`);
|
|
145
109
|
continue;
|
|
146
110
|
}
|
|
147
|
-
|
|
148
111
|
const cdInfo = {
|
|
149
112
|
url: `${_utils.CD_CDN}/${key}`,
|
|
150
113
|
etag: _lodash.default.trim(etag, '"'),
|
|
@@ -153,26 +116,20 @@ class ChromedriverStorageClient {
|
|
|
153
116
|
this.mapping[key] = cdInfo;
|
|
154
117
|
const notesPath = `${cdInfo.version}/notes.txt`;
|
|
155
118
|
const isNotesPresent = !!driverNodes.reduce((acc, node) => acc || findChildNode(node, 'Key', notesPath), false);
|
|
156
|
-
|
|
157
119
|
if (!isNotesPresent) {
|
|
158
120
|
cdInfo.minBrowserVersion = null;
|
|
159
|
-
|
|
160
121
|
if (shouldParseNotes) {
|
|
161
122
|
log.info(`The entry '${key}' does not contain any notes. Skipping it`);
|
|
162
123
|
}
|
|
163
|
-
|
|
164
124
|
continue;
|
|
165
125
|
} else if (!shouldParseNotes) {
|
|
166
126
|
continue;
|
|
167
127
|
}
|
|
168
|
-
|
|
169
128
|
promises.push(this.retrieveAdditionalDriverInfo(key, `${_utils.CD_CDN}/${notesPath}`, cdInfo));
|
|
170
|
-
|
|
171
129
|
if (promises.length % MAX_PARALLEL_DOWNLOADS === 0) {
|
|
172
130
|
await _bluebird.default.all(promises);
|
|
173
131
|
}
|
|
174
132
|
}
|
|
175
|
-
|
|
176
133
|
await _bluebird.default.all(promises);
|
|
177
134
|
log.info(`The total count of entries in the mapping: ${_lodash.default.size(this.mapping)}`);
|
|
178
135
|
}
|
|
@@ -191,15 +148,12 @@ class ChromedriverStorageClient {
|
|
|
191
148
|
|
|
192
149
|
async unzipDriver(src, dst) {
|
|
193
150
|
const tmpRoot = await _support.tempDir.openDir();
|
|
194
|
-
|
|
195
151
|
try {
|
|
196
152
|
await _support.zip.extractAllTo(src, tmpRoot);
|
|
197
153
|
const chromedriverPath = await _support.fs.walkDir(tmpRoot, true, (itemPath, isDirectory) => !isDirectory && _lodash.default.toLower(_path.default.parse(itemPath).name) === 'chromedriver');
|
|
198
|
-
|
|
199
154
|
if (!chromedriverPath) {
|
|
200
155
|
throw new Error('The archive was unzipped properly, but we could not find any chromedriver executable');
|
|
201
156
|
}
|
|
202
|
-
|
|
203
157
|
log.debug(`Moving the extracted '${_path.default.basename(chromedriverPath)}' to '${dst}'`);
|
|
204
158
|
await _support.fs.mv(chromedriverPath, dst, {
|
|
205
159
|
mkdirp: true
|
|
@@ -214,63 +168,49 @@ class ChromedriverStorageClient {
|
|
|
214
168
|
minBrowserVersion,
|
|
215
169
|
versions = []
|
|
216
170
|
} = opts;
|
|
217
|
-
|
|
218
171
|
let driversToSync = _lodash.default.keys(this.mapping);
|
|
219
|
-
|
|
220
172
|
if (!_lodash.default.isEmpty(versions)) {
|
|
221
173
|
log.debug(`Selecting chromedrivers whose versions match to ${versions}`);
|
|
222
174
|
driversToSync = driversToSync.filter(cdName => versions.includes(`${this.mapping[cdName].version}`));
|
|
223
175
|
log.debug(`Got ${_support.util.pluralize('item', driversToSync.length, true)}`);
|
|
224
|
-
|
|
225
176
|
if (_lodash.default.isEmpty(driversToSync)) {
|
|
226
177
|
return [];
|
|
227
178
|
}
|
|
228
179
|
}
|
|
229
|
-
|
|
230
180
|
if (!isNaN(minBrowserVersion)) {
|
|
231
181
|
const minBrowserVersionInt = parseInt(minBrowserVersion, 10);
|
|
232
182
|
log.debug(`Selecting chromedrivers whose minimum supported browser version matches to ${minBrowserVersionInt}`);
|
|
233
183
|
let closestMatchedVersionNumber = 0;
|
|
234
|
-
|
|
235
184
|
for (const cdName of driversToSync) {
|
|
236
185
|
const currentMinBrowserVersion = parseInt(this.mapping[cdName].minBrowserVersion, 10);
|
|
237
|
-
|
|
238
186
|
if (!isNaN(currentMinBrowserVersion) && currentMinBrowserVersion <= minBrowserVersionInt && closestMatchedVersionNumber < currentMinBrowserVersion) {
|
|
239
187
|
closestMatchedVersionNumber = currentMinBrowserVersion;
|
|
240
188
|
}
|
|
241
189
|
}
|
|
242
|
-
|
|
243
190
|
driversToSync = driversToSync.filter(cdName => `${this.mapping[cdName].minBrowserVersion}` === `${closestMatchedVersionNumber > 0 ? closestMatchedVersionNumber : minBrowserVersionInt}`);
|
|
244
191
|
log.debug(`Got ${_support.util.pluralize('item', driversToSync.length, true)}`);
|
|
245
|
-
|
|
246
192
|
if (_lodash.default.isEmpty(driversToSync)) {
|
|
247
193
|
return [];
|
|
248
194
|
}
|
|
249
|
-
|
|
250
195
|
log.debug(`Will select candidate ${_support.util.pluralize('driver', driversToSync.length)} ` + `versioned as '${_lodash.default.uniq(driversToSync.map(cdName => this.mapping[cdName].version))}'`);
|
|
251
196
|
}
|
|
252
|
-
|
|
253
197
|
if (!_lodash.default.isEmpty(osInfo)) {
|
|
254
198
|
let {
|
|
255
199
|
name,
|
|
256
200
|
arch,
|
|
257
201
|
hardwareName
|
|
258
202
|
} = osInfo;
|
|
259
|
-
|
|
260
203
|
if (arch === _utils.X64 && !driversToSync.some(cdName => cdName.includes(`_${name}${_utils.X64}`))) {
|
|
261
204
|
arch = _utils.X86;
|
|
262
205
|
}
|
|
263
|
-
|
|
264
206
|
if (name === _utils.OS.mac && _lodash.default.includes(hardwareName, 'arm') && driversToSync.some(cdName => cdName.includes(_utils.M1_ARCH_SUFFIX))) {
|
|
265
207
|
arch += _utils.M1_ARCH_SUFFIX;
|
|
266
208
|
}
|
|
267
|
-
|
|
268
209
|
log.debug(`Selecting chromedrivers whose platform matches to ${name}${arch}`);
|
|
269
210
|
const platformRe = new RegExp(`(\\b|_)${name}${arch}\\b`);
|
|
270
211
|
driversToSync = driversToSync.filter(cdName => platformRe.test(cdName));
|
|
271
212
|
log.debug(`Got ${_support.util.pluralize('item', driversToSync.length, true)}`);
|
|
272
213
|
}
|
|
273
|
-
|
|
274
214
|
return driversToSync;
|
|
275
215
|
}
|
|
276
216
|
|
|
@@ -280,11 +220,8 @@ class ChromedriverStorageClient {
|
|
|
280
220
|
etag,
|
|
281
221
|
version
|
|
282
222
|
} = this.mapping[driverKey];
|
|
283
|
-
|
|
284
223
|
const archivePath = _path.default.resolve(archivesRoot, `${index}.zip`);
|
|
285
|
-
|
|
286
224
|
log.debug(`Retrieving '${url}' to '${archivePath}'`);
|
|
287
|
-
|
|
288
225
|
try {
|
|
289
226
|
await _support.net.downloadFile(url, archivePath, {
|
|
290
227
|
isMetered: false,
|
|
@@ -292,30 +229,22 @@ class ChromedriverStorageClient {
|
|
|
292
229
|
});
|
|
293
230
|
} catch (e) {
|
|
294
231
|
const msg = `Cannot download chromedriver archive. Original error: ${e.message}`;
|
|
295
|
-
|
|
296
232
|
if (isStrict) {
|
|
297
233
|
throw new Error(msg);
|
|
298
234
|
}
|
|
299
|
-
|
|
300
235
|
log.error(msg);
|
|
301
236
|
return false;
|
|
302
237
|
}
|
|
303
|
-
|
|
304
238
|
if (!(await isCrcOk(archivePath, etag))) {
|
|
305
239
|
const msg = `The checksum for the downloaded chromedriver '${driverKey}' did not match`;
|
|
306
|
-
|
|
307
240
|
if (isStrict) {
|
|
308
241
|
throw new Error(msg);
|
|
309
242
|
}
|
|
310
|
-
|
|
311
243
|
log.error(msg);
|
|
312
244
|
return false;
|
|
313
245
|
}
|
|
314
|
-
|
|
315
246
|
const fileName = `${_path.default.parse(url).name}_v${version}` + (_support.system.isWindows() ? '.exe' : '');
|
|
316
|
-
|
|
317
247
|
const targetPath = _path.default.resolve(this.chromedriverDir, fileName);
|
|
318
|
-
|
|
319
248
|
try {
|
|
320
249
|
await this.unzipDriver(archivePath, targetPath);
|
|
321
250
|
await _support.fs.chmod(targetPath, 0o755);
|
|
@@ -324,11 +253,9 @@ class ChromedriverStorageClient {
|
|
|
324
253
|
if (isStrict) {
|
|
325
254
|
throw e;
|
|
326
255
|
}
|
|
327
|
-
|
|
328
256
|
log.error(e.message);
|
|
329
257
|
return false;
|
|
330
258
|
}
|
|
331
|
-
|
|
332
259
|
return true;
|
|
333
260
|
}
|
|
334
261
|
|
|
@@ -336,23 +263,18 @@ class ChromedriverStorageClient {
|
|
|
336
263
|
if (_lodash.default.isEmpty(this.mapping)) {
|
|
337
264
|
await this.retrieveMapping(!!opts.minBrowserVersion);
|
|
338
265
|
}
|
|
339
|
-
|
|
340
266
|
if (_lodash.default.isEmpty(this.mapping)) {
|
|
341
267
|
throw new Error('Cannot retrieve chromedrivers mapping from Google storage');
|
|
342
268
|
}
|
|
343
|
-
|
|
344
269
|
const driversToSync = this.selectMatchingDrivers(opts.osInfo ?? (await (0, _utils.getOsInfo)()), opts);
|
|
345
|
-
|
|
346
270
|
if (_lodash.default.isEmpty(driversToSync)) {
|
|
347
271
|
log.debug(`There are no drivers to sync. Exiting`);
|
|
348
272
|
return [];
|
|
349
273
|
}
|
|
350
|
-
|
|
351
274
|
log.debug(`Got ${_support.util.pluralize('driver', driversToSync.length, true)} to sync: ` + JSON.stringify(driversToSync, null, 2));
|
|
352
275
|
const synchronizedDrivers = [];
|
|
353
276
|
const promises = [];
|
|
354
277
|
const archivesRoot = await _support.tempDir.openDir();
|
|
355
|
-
|
|
356
278
|
try {
|
|
357
279
|
for (const [idx, driverKey] of driversToSync.entries()) {
|
|
358
280
|
promises.push((async () => {
|
|
@@ -360,28 +282,22 @@ class ChromedriverStorageClient {
|
|
|
360
282
|
synchronizedDrivers.push(driverKey);
|
|
361
283
|
}
|
|
362
284
|
})());
|
|
363
|
-
|
|
364
285
|
if (promises.length % MAX_PARALLEL_DOWNLOADS === 0) {
|
|
365
286
|
await _bluebird.default.all(promises);
|
|
366
287
|
}
|
|
367
288
|
}
|
|
368
|
-
|
|
369
289
|
await _bluebird.default.all(promises);
|
|
370
290
|
} finally {
|
|
371
291
|
await _support.fs.rimraf(archivesRoot);
|
|
372
292
|
}
|
|
373
|
-
|
|
374
293
|
if (!_lodash.default.isEmpty(synchronizedDrivers)) {
|
|
375
294
|
log.info(`Successfully synchronized ` + `${_support.util.pluralize('chromedriver', synchronizedDrivers.length, true)}`);
|
|
376
295
|
} else {
|
|
377
296
|
log.info(`No chromedrivers were synchronized`);
|
|
378
297
|
}
|
|
379
|
-
|
|
380
298
|
return synchronizedDrivers;
|
|
381
299
|
}
|
|
382
|
-
|
|
383
300
|
}
|
|
384
|
-
|
|
385
301
|
var _default = ChromedriverStorageClient;
|
|
386
302
|
exports.default = _default;
|
|
387
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["TIMEOUT_MS","MAX_PARALLEL_DOWNLOADS","log","logger","getLogger","isCrcOk","src","checksum","md5","fs","hash","_","toLower","findChildNode","parent","childName","text","hasChildNodes","childNodeIdx","childNodes","length","childNode","localName","childText","extractNodeText","node","firstChild","util","hasValue","nodeValue","ChromedriverStorageClient","constructor","args","chromedriverDir","getChromedriverDir","timeout","mapping","parseNotes","content","result","versionMatch","exec","version","minBrowserVersionMatch","minBrowserVersion","retrieveAdditionalDriverInfo","driverKey","notesUrl","infoDict","notes","retrieveData","accept","debug","parseStorageXml","doc","shouldParseNotes","driverNodes","xpath","select","isEmpty","promises","driverNode","key","includes","etag","cdInfo","url","CD_CDN","trim","first","split","notesPath","isNotesPresent","reduce","acc","info","push","B","all","size","retrieveMapping","xml","DOMParser","parseFromString","cloneDeep","unzipDriver","dst","tmpRoot","tempDir","openDir","zip","extractAllTo","chromedriverPath","walkDir","itemPath","isDirectory","path","parse","name","Error","basename","mv","mkdirp","rimraf","selectMatchingDrivers","osInfo","opts","versions","driversToSync","keys","filter","cdName","pluralize","isNaN","minBrowserVersionInt","parseInt","closestMatchedVersionNumber","currentMinBrowserVersion","uniq","map","arch","hardwareName","X64","some","X86","OS","mac","M1_ARCH_SUFFIX","platformRe","RegExp","test","retrieveDriver","index","archivesRoot","isStrict","archivePath","resolve","net","downloadFile","isMetered","e","msg","message","error","fileName","system","isWindows","targetPath","chmod","syncDrivers","getOsInfo","JSON","stringify","synchronizedDrivers","idx","entries"],"sources":["../../lib/storage-client.js"],"sourcesContent":["import {\n  getChromedriverDir, CD_CDN, retrieveData, getOsInfo,\n  OS, X64, X86, M1_ARCH_SUFFIX,\n} from './utils';\nimport _ from 'lodash';\nimport xpath from 'xpath';\nimport { DOMParser } from '@xmldom/xmldom';\nimport B from 'bluebird';\nimport path from 'path';\nimport { system, fs, logger, tempDir, zip, util, net } from 'appium/support';\n\n\nconst TIMEOUT_MS = 15000;\nconst MAX_PARALLEL_DOWNLOADS = 5;\n\nconst log = logger.getLogger('ChromedriverStorageClient');\n\n\nasync function isCrcOk (src, checksum) {\n  const md5 = await fs.hash(src, 'md5');\n  return _.toLower(md5) === _.toLower(checksum);\n}\n\nfunction findChildNode (parent, childName = null, text = null) {\n  if (!childName && !text) {\n    return null;\n  }\n  if (!parent.hasChildNodes()) {\n    return null;\n  }\n\n  for (let childNodeIdx = 0; childNodeIdx < parent.childNodes.length; childNodeIdx++) {\n    const childNode = parent.childNodes[childNodeIdx];\n    if (childName && !text && childName === childNode.localName) {\n      return childNode;\n    }\n    if (text) {\n      const childText = extractNodeText(childNode);\n      if (!childText) {\n        continue;\n      }\n      if (childName && childName === childNode.localName && text === childText) {\n        return childNode;\n      }\n      if (!childName && text === childText) {\n        return childNode;\n      }\n    }\n  }\n  return null;\n}\n\nfunction extractNodeText (node) {\n  return (!node || !node.firstChild || !util.hasValue(node.firstChild.nodeValue))\n    ? null\n    : node.firstChild.nodeValue;\n}\n\n\nclass ChromedriverStorageClient {\n  constructor (args = {}) {\n    const {\n      chromedriverDir = getChromedriverDir(),\n      timeout = TIMEOUT_MS,\n    } = args;\n    this.chromedriverDir = chromedriverDir;\n    this.timeout = timeout;\n    this.mapping = {};\n  }\n\n  /**\n   * @typedef {Object} AdditionalDriverDetails\n   * @property {?string} version - Chromedriver version\n   * or `null` if it cannot be found\n   * @property {?string} minBrowserVersion - The minimum browser version\n   * supported by chromedriver or `null` if it cannot be found\n   */\n\n  /**\n   * Gets additional chromedriver details from chromedriver\n   * release notes\n   *\n   * @param {string} content - Release notes of the corresponding chromedriver\n   * @returns {AdditionalDriverDetails}\n   */\n  parseNotes (content) {\n    const result = {};\n    const versionMatch = /^\\s*[-]+ChromeDriver[\\D]+([\\d.]+)/im.exec(content);\n    if (versionMatch) {\n      result.version = versionMatch[1];\n    }\n    const minBrowserVersionMatch = /^\\s*Supports Chrome[\\D]+(\\d+)/im.exec(content);\n    if (minBrowserVersionMatch) {\n      result.minBrowserVersion = minBrowserVersionMatch[1];\n    }\n    return result;\n  }\n\n  /**\n   * Downloads chromedriver release notes and puts them\n   * into the dictionary argument\n   *\n   * @param {string} driverKey - Driver version plus archive name\n   * @param {string} notesUrl - The URL of chromedriver notes\n   * @param {Object} infoDict - The dictionary containing driver info.\n   * The method call mutates by merging `AdditionalDriverDetails`\n   * @throws {Error} if the release notes cannot be downloaded\n   */\n  async retrieveAdditionalDriverInfo (driverKey, notesUrl, infoDict) {\n    const notes = await retrieveData(notesUrl, {\n      'user-agent': 'appium',\n      accept: '*/*',\n    }, { timeout: this.timeout });\n    const { minBrowserVersion } = this.parseNotes(notes);\n    if (!minBrowserVersion) {\n      log.debug(`The driver '${driverKey}' does not contain valid release notes at ${notesUrl}. ` +\n        `Skipping it`);\n      return;\n    }\n    infoDict.minBrowserVersion = minBrowserVersion;\n  }\n\n  /**\n   * Parses chromedriver storage XML and stores\n   * the parsed results into `this.mapping`\n   *\n   * @param {DOMDocument} doc - The DOM representation\n   * of the chromedriver storage XML\n   * @param {boolean} shouldParseNotes [true] - If set to `true`\n   * then additional drivers information is going to be parsed\n   * and assigned to `this.mapping`\n   */\n  async parseStorageXml (doc, shouldParseNotes = true) {\n    const driverNodes = xpath.select(`//*[local-name(.)='Contents']`, doc);\n    log.debug(`Parsed ${driverNodes.length} entries from storage XML`);\n    if (_.isEmpty(driverNodes)) {\n      return;\n    }\n\n    const promises = [];\n    for (const driverNode of driverNodes) {\n      const key = extractNodeText(findChildNode(driverNode, 'Key'));\n      if (!_.includes(key, '/chromedriver_')) {\n        continue;\n      }\n\n      const etag = extractNodeText(findChildNode(driverNode, 'ETag'));\n      if (!etag) {\n        log.debug(`The entry '${key}' does not contain the checksum. Skipping it`);\n        continue;\n      }\n\n      const cdInfo = {\n        url: `${CD_CDN}/${key}`,\n        etag: _.trim(etag, '\"'),\n        version: _.first(key.split('/')),\n      };\n      this.mapping[key] = cdInfo;\n\n      const notesPath = `${cdInfo.version}/notes.txt`;\n      const isNotesPresent = !!driverNodes\n        .reduce((acc, node) => acc || findChildNode(node, 'Key', notesPath), false);\n      if (!isNotesPresent) {\n        cdInfo.minBrowserVersion = null;\n        if (shouldParseNotes) {\n          log.info(`The entry '${key}' does not contain any notes. Skipping it`);\n        }\n        continue;\n      } else if (!shouldParseNotes) {\n        continue;\n      }\n\n      promises.push(this.retrieveAdditionalDriverInfo(key, `${CD_CDN}/${notesPath}`, cdInfo));\n      if (promises.length % MAX_PARALLEL_DOWNLOADS === 0) {\n        await B.all(promises);\n      }\n    }\n    await B.all(promises);\n    log.info(`The total count of entries in the mapping: ${_.size(this.mapping)}`);\n  }\n\n  /**\n   * @typedef {Object} DriverDetails\n   * @property {string} url - The full url to the corresponding driver in\n   * the remote storage\n   * @property {string} etag - The CRC of the driver archive\n   * @property {string} version - Chromedriver version\n   */\n\n  /**\n   * @typedef {Object} ChromedriversMapping\n   * @property {DriverDetails} - The keys are unique driver identifiers\n   * (version/archive name). The corresponding values have `DriverDetails`\n   * containing chromedriver details\n   */\n\n  /**\n   * Retrieves chromedriver mapping from the storage\n   *\n   * @param {boolean} shouldParseNotes [true] - if set to `true`\n   * then additional chromedrivers info is going to be retrieved and\n   * parsed from release notes\n   * @returns {ChromedriversMapping}\n   */\n  async retrieveMapping (shouldParseNotes = true) {\n    const xml = await retrieveData(CD_CDN, {\n      'user-agent': 'appium',\n      accept: 'application/xml, */*',\n    }, { timeout: this.timeout });\n    const doc = new DOMParser().parseFromString(xml);\n    await this.parseStorageXml(doc, shouldParseNotes);\n    return _.cloneDeep(this.mapping);\n  }\n\n  /**\n   * Extracts downloaded chromedriver archive\n   * into the given destination\n   *\n   * @param {string} src - The source archive path\n   * @param {string} dst - The destination chromedriver path\n   */\n  async unzipDriver (src, dst) {\n    const tmpRoot = await tempDir.openDir();\n    try {\n      await zip.extractAllTo(src, tmpRoot);\n      const chromedriverPath = await fs.walkDir(tmpRoot, true, (itemPath, isDirectory) =>\n        !isDirectory && _.toLower(path.parse(itemPath).name) === 'chromedriver');\n      if (!chromedriverPath) {\n        throw new Error('The archive was unzipped properly, but we could not find any chromedriver executable');\n      }\n      log.debug(`Moving the extracted '${path.basename(chromedriverPath)}' to '${dst}'`);\n      await fs.mv(chromedriverPath, dst, {\n        mkdirp: true\n      });\n    } finally {\n      await fs.rimraf(tmpRoot);\n    }\n  }\n\n  /**\n   * @typedef {Object} OSInfo\n   * @property {string} name - The name of the host OS\n   * Can be either `mac`, `windows` or `linux`\n   * @property {string} arch - The architecture of the host OD.\n   * Can be either `32` or `64`\n   * @property {?string} hardwareName - The output of `uname -m` command\n   * on linux and mac systems. `null` on Windows\n   */\n\n  /**\n   * Filters `this.mapping` to only select matching\n   * chromedriver entries by operating system information\n   * and/or additional synchronization options (if provided)\n   *\n   * @param {?OSInfo} osInfo\n   * @param {?SyncOptions} opts\n   * @returns {Array<String>} The list of filtered chromedriver\n   * entry names (version/archive name)\n   */\n  selectMatchingDrivers (osInfo, opts = {}) {\n    const {\n      minBrowserVersion,\n      versions = [],\n    } = opts;\n    let driversToSync = _.keys(this.mapping);\n\n    if (!_.isEmpty(versions)) {\n      // Handle only selected versions if requested\n      log.debug(`Selecting chromedrivers whose versions match to ${versions}`);\n      driversToSync = driversToSync\n        .filter((cdName) => versions.includes(`${this.mapping[cdName].version}`));\n\n      log.debug(`Got ${util.pluralize('item', driversToSync.length, true)}`);\n      if (_.isEmpty(driversToSync)) {\n        return [];\n      }\n    }\n\n    if (!isNaN(minBrowserVersion)) {\n      // Only select drivers that support the current browser whose major version number equals to `minBrowserVersion`\n      const minBrowserVersionInt = parseInt(minBrowserVersion, 10);\n      log.debug(`Selecting chromedrivers whose minimum supported browser version matches to ${minBrowserVersionInt}`);\n      let closestMatchedVersionNumber = 0;\n      // Select the newest available and compatible chromedriver\n      for (const cdName of driversToSync) {\n        const currentMinBrowserVersion = parseInt(this.mapping[cdName].minBrowserVersion, 10);\n        if (!isNaN(currentMinBrowserVersion)\n            && currentMinBrowserVersion <= minBrowserVersionInt\n            && closestMatchedVersionNumber < currentMinBrowserVersion) {\n          closestMatchedVersionNumber = currentMinBrowserVersion;\n        }\n      }\n      driversToSync = driversToSync.filter((cdName) => `${this.mapping[cdName].minBrowserVersion}` ===\n        `${closestMatchedVersionNumber > 0 ? closestMatchedVersionNumber : minBrowserVersionInt}`);\n\n      log.debug(`Got ${util.pluralize('item', driversToSync.length, true)}`);\n      if (_.isEmpty(driversToSync)) {\n        return [];\n      }\n      log.debug(`Will select candidate ${util.pluralize('driver', driversToSync.length)} ` +\n        `versioned as '${_.uniq(driversToSync.map((cdName) => this.mapping[cdName].version))}'`);\n    }\n\n    if (!_.isEmpty(osInfo)) {\n      // Filter out drivers for unsupported system architectures\n      let {name, arch, hardwareName} = osInfo;\n      if (arch === X64 && !driversToSync.some((cdName) => cdName.includes(`_${name}${X64}`))) {\n        // Fall back to x86 build if x64 one is not available for the given OS\n        arch = X86;\n      }\n      if (name === OS.mac && _.includes(hardwareName, 'arm')\n          && driversToSync.some((cdName) => cdName.includes(M1_ARCH_SUFFIX))) {\n        // prefer executable for M1 arch if present\n        arch += M1_ARCH_SUFFIX;\n      }\n      log.debug(`Selecting chromedrivers whose platform matches to ${name}${arch}`);\n      const platformRe = new RegExp(`(\\\\b|_)${name}${arch}\\\\b`);\n      driversToSync = driversToSync.filter((cdName) => platformRe.test(cdName));\n      log.debug(`Got ${util.pluralize('item', driversToSync.length, true)}`);\n    }\n\n    return driversToSync;\n  }\n\n  /**\n   * Retrieves the given chromedriver from the storage\n   * and unpacks it into `this.chromedriverDir` folder\n   *\n   * @param {number} index - The unique driver index\n   * @param {string} driverKey - The driver key in `this.mapping`\n   * @param {string} archivesRoot - The temporary folder path to extract\n   * downloaded archives to\n   * @param {boolean} isStrict [true] - Whether to throw an error (`true`)\n   * or return a boolean result if the driver retrieval process fails\n   * @throws {Error} if there was a failure while retrieving the driver\n   * and `isStrict` is set to `true`\n   * @returns {boolean} if `true` then the chromedriver is successfully\n   * downloaded and extracted.\n   */\n  async retrieveDriver (index, driverKey, archivesRoot, isStrict = false) {\n    const { url, etag, version } = this.mapping[driverKey];\n    const archivePath = path.resolve(archivesRoot, `${index}.zip`);\n    log.debug(`Retrieving '${url}' to '${archivePath}'`);\n    try {\n      await net.downloadFile(url, archivePath, {\n        isMetered: false,\n        timeout: TIMEOUT_MS\n      });\n    } catch (e) {\n      const msg = `Cannot download chromedriver archive. Original error: ${e.message}`;\n      if (isStrict) {\n        throw new Error(msg);\n      }\n      log.error(msg);\n      return false;\n    }\n    if (!await isCrcOk(archivePath, etag)) {\n      const msg = `The checksum for the downloaded chromedriver '${driverKey}' did not match`;\n      if (isStrict) {\n        throw new Error(msg);\n      }\n      log.error(msg);\n      return false;\n    }\n    const fileName = `${path.parse(url).name}_v${version}` +\n      (system.isWindows() ? '.exe' : '');\n    const targetPath = path.resolve(this.chromedriverDir, fileName);\n    try {\n      await this.unzipDriver(archivePath, targetPath);\n      await fs.chmod(targetPath, 0o755);\n      log.debug(`Permissions of the file '${targetPath}' have been changed to 755`);\n    } catch (e) {\n      if (isStrict) {\n        throw e;\n      }\n      log.error(e.message);\n      return false;\n    }\n    return true;\n  }\n\n  /**\n   * @typedef {Object} SyncOptions\n   * @property {Array<String>} versions - The list of chromedriver\n   * versions to sync. If empty (the default value) then all available\n   * chromedrivers are going to be downloaded and extracted\n   * @property {string|number} minBrowserVersion - The minumum supported\n   * Chrome version that downloaded chromedrivers should support. Can match\n   * multiple drivers.\n   * @property {?OSInfo} osInfo - System information used to filter out\n   * the list of the retrieved drivers. If not provided then the script\n   * will try to retrieve it.\n   */\n\n  /**\n   * Retrieves chromedrivers from the remote storage\n   * to the local file system\n   *\n   * @param {?SyncOptions} opts\n   * @throws {Error} if there was a problem while retrieving\n   * the drivers\n   * @returns {Array<String} The list of successfully synchronized driver keys\n   */\n  async syncDrivers (opts = {}) {\n    if (_.isEmpty(this.mapping)) {\n      await this.retrieveMapping(!!opts.minBrowserVersion);\n    }\n    if (_.isEmpty(this.mapping)) {\n      throw new Error('Cannot retrieve chromedrivers mapping from Google storage');\n    }\n\n    const driversToSync = this.selectMatchingDrivers(opts.osInfo ?? await getOsInfo(), opts);\n    if (_.isEmpty(driversToSync)) {\n      log.debug(`There are no drivers to sync. Exiting`);\n      return [];\n    }\n    log.debug(`Got ${util.pluralize('driver', driversToSync.length, true)} to sync: ` +\n      JSON.stringify(driversToSync, null, 2));\n\n    const synchronizedDrivers = [];\n    const promises = [];\n    const archivesRoot = await tempDir.openDir();\n    try {\n      for (const [idx, driverKey] of driversToSync.entries()) {\n        promises.push((async () => {\n          if (await this.retrieveDriver(idx, driverKey, archivesRoot, !_.isEmpty(opts))) {\n            synchronizedDrivers.push(driverKey);\n          }\n        })());\n\n        if (promises.length % MAX_PARALLEL_DOWNLOADS === 0) {\n          await B.all(promises);\n        }\n      }\n      await B.all(promises);\n    } finally {\n      await fs.rimraf(archivesRoot);\n    }\n    if (!_.isEmpty(synchronizedDrivers)) {\n      log.info(`Successfully synchronized ` +\n        `${util.pluralize('chromedriver', synchronizedDrivers.length, true)}`);\n    } else {\n      log.info(`No chromedrivers were synchronized`);\n    }\n    return synchronizedDrivers;\n  }\n}\n\n\nexport default ChromedriverStorageClient;\n"],"mappings":";;;;;;;;;;;AAAA;;AAIA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAGA,MAAMA,UAAU,GAAG,KAAnB;AACA,MAAMC,sBAAsB,GAAG,CAA/B;;AAEA,MAAMC,GAAG,GAAGC,eAAA,CAAOC,SAAP,CAAiB,2BAAjB,CAAZ;;AAGA,eAAeC,OAAf,CAAwBC,GAAxB,EAA6BC,QAA7B,EAAuC;EACrC,MAAMC,GAAG,GAAG,MAAMC,WAAA,CAAGC,IAAH,CAAQJ,GAAR,EAAa,KAAb,CAAlB;EACA,OAAOK,eAAA,CAAEC,OAAF,CAAUJ,GAAV,MAAmBG,eAAA,CAAEC,OAAF,CAAUL,QAAV,CAA1B;AACD;;AAED,SAASM,aAAT,CAAwBC,MAAxB,EAAgCC,SAAS,GAAG,IAA5C,EAAkDC,IAAI,GAAG,IAAzD,EAA+D;EAC7D,IAAI,CAACD,SAAD,IAAc,CAACC,IAAnB,EAAyB;IACvB,OAAO,IAAP;EACD;;EACD,IAAI,CAACF,MAAM,CAACG,aAAP,EAAL,EAA6B;IAC3B,OAAO,IAAP;EACD;;EAED,KAAK,IAAIC,YAAY,GAAG,CAAxB,EAA2BA,YAAY,GAAGJ,MAAM,CAACK,UAAP,CAAkBC,MAA5D,EAAoEF,YAAY,EAAhF,EAAoF;IAClF,MAAMG,SAAS,GAAGP,MAAM,CAACK,UAAP,CAAkBD,YAAlB,CAAlB;;IACA,IAAIH,SAAS,IAAI,CAACC,IAAd,IAAsBD,SAAS,KAAKM,SAAS,CAACC,SAAlD,EAA6D;MAC3D,OAAOD,SAAP;IACD;;IACD,IAAIL,IAAJ,EAAU;MACR,MAAMO,SAAS,GAAGC,eAAe,CAACH,SAAD,CAAjC;;MACA,IAAI,CAACE,SAAL,EAAgB;QACd;MACD;;MACD,IAAIR,SAAS,IAAIA,SAAS,KAAKM,SAAS,CAACC,SAArC,IAAkDN,IAAI,KAAKO,SAA/D,EAA0E;QACxE,OAAOF,SAAP;MACD;;MACD,IAAI,CAACN,SAAD,IAAcC,IAAI,KAAKO,SAA3B,EAAsC;QACpC,OAAOF,SAAP;MACD;IACF;EACF;;EACD,OAAO,IAAP;AACD;;AAED,SAASG,eAAT,CAA0BC,IAA1B,EAAgC;EAC9B,OAAQ,CAACA,IAAD,IAAS,CAACA,IAAI,CAACC,UAAf,IAA6B,CAACC,aAAA,CAAKC,QAAL,CAAcH,IAAI,CAACC,UAAL,CAAgBG,SAA9B,CAA/B,GACH,IADG,GAEHJ,IAAI,CAACC,UAAL,CAAgBG,SAFpB;AAGD;;AAGD,MAAMC,yBAAN,CAAgC;EAC9BC,WAAW,CAAEC,IAAI,GAAG,EAAT,EAAa;IACtB,MAAM;MACJC,eAAe,GAAG,IAAAC,yBAAA,GADd;MAEJC,OAAO,GAAGnC;IAFN,IAGFgC,IAHJ;IAIA,KAAKC,eAAL,GAAuBA,eAAvB;IACA,KAAKE,OAAL,GAAeA,OAAf;IACA,KAAKC,OAAL,GAAe,EAAf;EACD;;EAiBDC,UAAU,CAAEC,OAAF,EAAW;IACnB,MAAMC,MAAM,GAAG,EAAf;IACA,MAAMC,YAAY,GAAG,sCAAsCC,IAAtC,CAA2CH,OAA3C,CAArB;;IACA,IAAIE,YAAJ,EAAkB;MAChBD,MAAM,CAACG,OAAP,GAAiBF,YAAY,CAAC,CAAD,CAA7B;IACD;;IACD,MAAMG,sBAAsB,GAAG,kCAAkCF,IAAlC,CAAuCH,OAAvC,CAA/B;;IACA,IAAIK,sBAAJ,EAA4B;MAC1BJ,MAAM,CAACK,iBAAP,GAA2BD,sBAAsB,CAAC,CAAD,CAAjD;IACD;;IACD,OAAOJ,MAAP;EACD;;EAYiC,MAA5BM,4BAA4B,CAAEC,SAAF,EAAaC,QAAb,EAAuBC,QAAvB,EAAiC;IACjE,MAAMC,KAAK,GAAG,MAAM,IAAAC,mBAAA,EAAaH,QAAb,EAAuB;MACzC,cAAc,QAD2B;MAEzCI,MAAM,EAAE;IAFiC,CAAvB,EAGjB;MAAEhB,OAAO,EAAE,KAAKA;IAAhB,CAHiB,CAApB;IAIA,MAAM;MAAES;IAAF,IAAwB,KAAKP,UAAL,CAAgBY,KAAhB,CAA9B;;IACA,IAAI,CAACL,iBAAL,EAAwB;MACtB1C,GAAG,CAACkD,KAAJ,CAAW,eAAcN,SAAU,6CAA4CC,QAAS,IAA9E,GACP,aADH;MAEA;IACD;;IACDC,QAAQ,CAACJ,iBAAT,GAA6BA,iBAA7B;EACD;;EAYoB,MAAfS,eAAe,CAAEC,GAAF,EAAOC,gBAAgB,GAAG,IAA1B,EAAgC;IACnD,MAAMC,WAAW,GAAGC,cAAA,CAAMC,MAAN,CAAc,+BAAd,EAA8CJ,GAA9C,CAApB;;IACApD,GAAG,CAACkD,KAAJ,CAAW,UAASI,WAAW,CAACpC,MAAO,2BAAvC;;IACA,IAAIT,eAAA,CAAEgD,OAAF,CAAUH,WAAV,CAAJ,EAA4B;MAC1B;IACD;;IAED,MAAMI,QAAQ,GAAG,EAAjB;;IACA,KAAK,MAAMC,UAAX,IAAyBL,WAAzB,EAAsC;MACpC,MAAMM,GAAG,GAAGtC,eAAe,CAACX,aAAa,CAACgD,UAAD,EAAa,KAAb,CAAd,CAA3B;;MACA,IAAI,CAAClD,eAAA,CAAEoD,QAAF,CAAWD,GAAX,EAAgB,gBAAhB,CAAL,EAAwC;QACtC;MACD;;MAED,MAAME,IAAI,GAAGxC,eAAe,CAACX,aAAa,CAACgD,UAAD,EAAa,MAAb,CAAd,CAA5B;;MACA,IAAI,CAACG,IAAL,EAAW;QACT9D,GAAG,CAACkD,KAAJ,CAAW,cAAaU,GAAI,8CAA5B;QACA;MACD;;MAED,MAAMG,MAAM,GAAG;QACbC,GAAG,EAAG,GAAEC,aAAO,IAAGL,GAAI,EADT;QAEbE,IAAI,EAAErD,eAAA,CAAEyD,IAAF,CAAOJ,IAAP,EAAa,GAAb,CAFO;QAGbtB,OAAO,EAAE/B,eAAA,CAAE0D,KAAF,CAAQP,GAAG,CAACQ,KAAJ,CAAU,GAAV,CAAR;MAHI,CAAf;MAKA,KAAKlC,OAAL,CAAa0B,GAAb,IAAoBG,MAApB;MAEA,MAAMM,SAAS,GAAI,GAAEN,MAAM,CAACvB,OAAQ,YAApC;MACA,MAAM8B,cAAc,GAAG,CAAC,CAAChB,WAAW,CACjCiB,MADsB,CACf,CAACC,GAAD,EAAMjD,IAAN,KAAeiD,GAAG,IAAI7D,aAAa,CAACY,IAAD,EAAO,KAAP,EAAc8C,SAAd,CADpB,EAC8C,KAD9C,CAAzB;;MAEA,IAAI,CAACC,cAAL,EAAqB;QACnBP,MAAM,CAACrB,iBAAP,GAA2B,IAA3B;;QACA,IAAIW,gBAAJ,EAAsB;UACpBrD,GAAG,CAACyE,IAAJ,CAAU,cAAab,GAAI,2CAA3B;QACD;;QACD;MACD,CAND,MAMO,IAAI,CAACP,gBAAL,EAAuB;QAC5B;MACD;;MAEDK,QAAQ,CAACgB,IAAT,CAAc,KAAK/B,4BAAL,CAAkCiB,GAAlC,EAAwC,GAAEK,aAAO,IAAGI,SAAU,EAA9D,EAAiEN,MAAjE,CAAd;;MACA,IAAIL,QAAQ,CAACxC,MAAT,GAAkBnB,sBAAlB,KAA6C,CAAjD,EAAoD;QAClD,MAAM4E,iBAAA,CAAEC,GAAF,CAAMlB,QAAN,CAAN;MACD;IACF;;IACD,MAAMiB,iBAAA,CAAEC,GAAF,CAAMlB,QAAN,CAAN;IACA1D,GAAG,CAACyE,IAAJ,CAAU,8CAA6ChE,eAAA,CAAEoE,IAAF,CAAO,KAAK3C,OAAZ,CAAqB,EAA5E;EACD;;EAyBoB,MAAf4C,eAAe,CAAEzB,gBAAgB,GAAG,IAArB,EAA2B;IAC9C,MAAM0B,GAAG,GAAG,MAAM,IAAA/B,mBAAA,EAAaiB,aAAb,EAAqB;MACrC,cAAc,QADuB;MAErChB,MAAM,EAAE;IAF6B,CAArB,EAGf;MAAEhB,OAAO,EAAE,KAAKA;IAAhB,CAHe,CAAlB;IAIA,MAAMmB,GAAG,GAAG,IAAI4B,iBAAJ,GAAgBC,eAAhB,CAAgCF,GAAhC,CAAZ;IACA,MAAM,KAAK5B,eAAL,CAAqBC,GAArB,EAA0BC,gBAA1B,CAAN;IACA,OAAO5C,eAAA,CAAEyE,SAAF,CAAY,KAAKhD,OAAjB,CAAP;EACD;;EASgB,MAAXiD,WAAW,CAAE/E,GAAF,EAAOgF,GAAP,EAAY;IAC3B,MAAMC,OAAO,GAAG,MAAMC,gBAAA,CAAQC,OAAR,EAAtB;;IACA,IAAI;MACF,MAAMC,YAAA,CAAIC,YAAJ,CAAiBrF,GAAjB,EAAsBiF,OAAtB,CAAN;MACA,MAAMK,gBAAgB,GAAG,MAAMnF,WAAA,CAAGoF,OAAH,CAAWN,OAAX,EAAoB,IAApB,EAA0B,CAACO,QAAD,EAAWC,WAAX,KACvD,CAACA,WAAD,IAAgBpF,eAAA,CAAEC,OAAF,CAAUoF,aAAA,CAAKC,KAAL,CAAWH,QAAX,EAAqBI,IAA/B,MAAyC,cAD5B,CAA/B;;MAEA,IAAI,CAACN,gBAAL,EAAuB;QACrB,MAAM,IAAIO,KAAJ,CAAU,sFAAV,CAAN;MACD;;MACDjG,GAAG,CAACkD,KAAJ,CAAW,yBAAwB4C,aAAA,CAAKI,QAAL,CAAcR,gBAAd,CAAgC,SAAQN,GAAI,GAA/E;MACA,MAAM7E,WAAA,CAAG4F,EAAH,CAAMT,gBAAN,EAAwBN,GAAxB,EAA6B;QACjCgB,MAAM,EAAE;MADyB,CAA7B,CAAN;IAGD,CAXD,SAWU;MACR,MAAM7F,WAAA,CAAG8F,MAAH,CAAUhB,OAAV,CAAN;IACD;EACF;;EAsBDiB,qBAAqB,CAAEC,MAAF,EAAUC,IAAI,GAAG,EAAjB,EAAqB;IACxC,MAAM;MACJ9D,iBADI;MAEJ+D,QAAQ,GAAG;IAFP,IAGFD,IAHJ;;IAIA,IAAIE,aAAa,GAAGjG,eAAA,CAAEkG,IAAF,CAAO,KAAKzE,OAAZ,CAApB;;IAEA,IAAI,CAACzB,eAAA,CAAEgD,OAAF,CAAUgD,QAAV,CAAL,EAA0B;MAExBzG,GAAG,CAACkD,KAAJ,CAAW,mDAAkDuD,QAAS,EAAtE;MACAC,aAAa,GAAGA,aAAa,CAC1BE,MADa,CACLC,MAAD,IAAYJ,QAAQ,CAAC5C,QAAT,CAAmB,GAAE,KAAK3B,OAAL,CAAa2E,MAAb,EAAqBrE,OAAQ,EAAlD,CADN,CAAhB;MAGAxC,GAAG,CAACkD,KAAJ,CAAW,OAAMzB,aAAA,CAAKqF,SAAL,CAAe,MAAf,EAAuBJ,aAAa,CAACxF,MAArC,EAA6C,IAA7C,CAAmD,EAApE;;MACA,IAAIT,eAAA,CAAEgD,OAAF,CAAUiD,aAAV,CAAJ,EAA8B;QAC5B,OAAO,EAAP;MACD;IACF;;IAED,IAAI,CAACK,KAAK,CAACrE,iBAAD,CAAV,EAA+B;MAE7B,MAAMsE,oBAAoB,GAAGC,QAAQ,CAACvE,iBAAD,EAAoB,EAApB,CAArC;MACA1C,GAAG,CAACkD,KAAJ,CAAW,8EAA6E8D,oBAAqB,EAA7G;MACA,IAAIE,2BAA2B,GAAG,CAAlC;;MAEA,KAAK,MAAML,MAAX,IAAqBH,aAArB,EAAoC;QAClC,MAAMS,wBAAwB,GAAGF,QAAQ,CAAC,KAAK/E,OAAL,CAAa2E,MAAb,EAAqBnE,iBAAtB,EAAyC,EAAzC,CAAzC;;QACA,IAAI,CAACqE,KAAK,CAACI,wBAAD,CAAN,IACGA,wBAAwB,IAAIH,oBAD/B,IAEGE,2BAA2B,GAAGC,wBAFrC,EAE+D;UAC7DD,2BAA2B,GAAGC,wBAA9B;QACD;MACF;;MACDT,aAAa,GAAGA,aAAa,CAACE,MAAd,CAAsBC,MAAD,IAAa,GAAE,KAAK3E,OAAL,CAAa2E,MAAb,EAAqBnE,iBAAkB,EAA1C,KAC9C,GAAEwE,2BAA2B,GAAG,CAA9B,GAAkCA,2BAAlC,GAAgEF,oBAAqB,EAD1E,CAAhB;MAGAhH,GAAG,CAACkD,KAAJ,CAAW,OAAMzB,aAAA,CAAKqF,SAAL,CAAe,MAAf,EAAuBJ,aAAa,CAACxF,MAArC,EAA6C,IAA7C,CAAmD,EAApE;;MACA,IAAIT,eAAA,CAAEgD,OAAF,CAAUiD,aAAV,CAAJ,EAA8B;QAC5B,OAAO,EAAP;MACD;;MACD1G,GAAG,CAACkD,KAAJ,CAAW,yBAAwBzB,aAAA,CAAKqF,SAAL,CAAe,QAAf,EAAyBJ,aAAa,CAACxF,MAAvC,CAA+C,GAAxE,GACP,iBAAgBT,eAAA,CAAE2G,IAAF,CAAOV,aAAa,CAACW,GAAd,CAAmBR,MAAD,IAAY,KAAK3E,OAAL,CAAa2E,MAAb,EAAqBrE,OAAnD,CAAP,CAAoE,GADvF;IAED;;IAED,IAAI,CAAC/B,eAAA,CAAEgD,OAAF,CAAU8C,MAAV,CAAL,EAAwB;MAEtB,IAAI;QAACP,IAAD;QAAOsB,IAAP;QAAaC;MAAb,IAA6BhB,MAAjC;;MACA,IAAIe,IAAI,KAAKE,UAAT,IAAgB,CAACd,aAAa,CAACe,IAAd,CAAoBZ,MAAD,IAAYA,MAAM,CAAChD,QAAP,CAAiB,IAAGmC,IAAK,GAAEwB,UAAI,EAA/B,CAA/B,CAArB,EAAwF;QAEtFF,IAAI,GAAGI,UAAP;MACD;;MACD,IAAI1B,IAAI,KAAK2B,SAAA,CAAGC,GAAZ,IAAmBnH,eAAA,CAAEoD,QAAF,CAAW0D,YAAX,EAAyB,KAAzB,CAAnB,IACGb,aAAa,CAACe,IAAd,CAAoBZ,MAAD,IAAYA,MAAM,CAAChD,QAAP,CAAgBgE,qBAAhB,CAA/B,CADP,EACwE;QAEtEP,IAAI,IAAIO,qBAAR;MACD;;MACD7H,GAAG,CAACkD,KAAJ,CAAW,qDAAoD8C,IAAK,GAAEsB,IAAK,EAA3E;MACA,MAAMQ,UAAU,GAAG,IAAIC,MAAJ,CAAY,UAAS/B,IAAK,GAAEsB,IAAK,KAAjC,CAAnB;MACAZ,aAAa,GAAGA,aAAa,CAACE,MAAd,CAAsBC,MAAD,IAAYiB,UAAU,CAACE,IAAX,CAAgBnB,MAAhB,CAAjC,CAAhB;MACA7G,GAAG,CAACkD,KAAJ,CAAW,OAAMzB,aAAA,CAAKqF,SAAL,CAAe,MAAf,EAAuBJ,aAAa,CAACxF,MAArC,EAA6C,IAA7C,CAAmD,EAApE;IACD;;IAED,OAAOwF,aAAP;EACD;;EAiBmB,MAAduB,cAAc,CAAEC,KAAF,EAAStF,SAAT,EAAoBuF,YAApB,EAAkCC,QAAQ,GAAG,KAA7C,EAAoD;IACtE,MAAM;MAAEpE,GAAF;MAAOF,IAAP;MAAatB;IAAb,IAAyB,KAAKN,OAAL,CAAaU,SAAb,CAA/B;;IACA,MAAMyF,WAAW,GAAGvC,aAAA,CAAKwC,OAAL,CAAaH,YAAb,EAA4B,GAAED,KAAM,MAApC,CAApB;;IACAlI,GAAG,CAACkD,KAAJ,CAAW,eAAcc,GAAI,SAAQqE,WAAY,GAAjD;;IACA,IAAI;MACF,MAAME,YAAA,CAAIC,YAAJ,CAAiBxE,GAAjB,EAAsBqE,WAAtB,EAAmC;QACvCI,SAAS,EAAE,KAD4B;QAEvCxG,OAAO,EAAEnC;MAF8B,CAAnC,CAAN;IAID,CALD,CAKE,OAAO4I,CAAP,EAAU;MACV,MAAMC,GAAG,GAAI,yDAAwDD,CAAC,CAACE,OAAQ,EAA/E;;MACA,IAAIR,QAAJ,EAAc;QACZ,MAAM,IAAInC,KAAJ,CAAU0C,GAAV,CAAN;MACD;;MACD3I,GAAG,CAAC6I,KAAJ,CAAUF,GAAV;MACA,OAAO,KAAP;IACD;;IACD,IAAI,EAAC,MAAMxI,OAAO,CAACkI,WAAD,EAAcvE,IAAd,CAAd,CAAJ,EAAuC;MACrC,MAAM6E,GAAG,GAAI,iDAAgD/F,SAAU,iBAAvE;;MACA,IAAIwF,QAAJ,EAAc;QACZ,MAAM,IAAInC,KAAJ,CAAU0C,GAAV,CAAN;MACD;;MACD3I,GAAG,CAAC6I,KAAJ,CAAUF,GAAV;MACA,OAAO,KAAP;IACD;;IACD,MAAMG,QAAQ,GAAI,GAAEhD,aAAA,CAAKC,KAAL,CAAW/B,GAAX,EAAgBgC,IAAK,KAAIxD,OAAQ,EAApC,IACduG,eAAA,CAAOC,SAAP,KAAqB,MAArB,GAA8B,EADhB,CAAjB;;IAEA,MAAMC,UAAU,GAAGnD,aAAA,CAAKwC,OAAL,CAAa,KAAKvG,eAAlB,EAAmC+G,QAAnC,CAAnB;;IACA,IAAI;MACF,MAAM,KAAK3D,WAAL,CAAiBkD,WAAjB,EAA8BY,UAA9B,CAAN;MACA,MAAM1I,WAAA,CAAG2I,KAAH,CAASD,UAAT,EAAqB,KAArB,CAAN;MACAjJ,GAAG,CAACkD,KAAJ,CAAW,4BAA2B+F,UAAW,4BAAjD;IACD,CAJD,CAIE,OAAOP,CAAP,EAAU;MACV,IAAIN,QAAJ,EAAc;QACZ,MAAMM,CAAN;MACD;;MACD1I,GAAG,CAAC6I,KAAJ,CAAUH,CAAC,CAACE,OAAZ;MACA,OAAO,KAAP;IACD;;IACD,OAAO,IAAP;EACD;;EAwBgB,MAAXO,WAAW,CAAE3C,IAAI,GAAG,EAAT,EAAa;IAC5B,IAAI/F,eAAA,CAAEgD,OAAF,CAAU,KAAKvB,OAAf,CAAJ,EAA6B;MAC3B,MAAM,KAAK4C,eAAL,CAAqB,CAAC,CAAC0B,IAAI,CAAC9D,iBAA5B,CAAN;IACD;;IACD,IAAIjC,eAAA,CAAEgD,OAAF,CAAU,KAAKvB,OAAf,CAAJ,EAA6B;MAC3B,MAAM,IAAI+D,KAAJ,CAAU,2DAAV,CAAN;IACD;;IAED,MAAMS,aAAa,GAAG,KAAKJ,qBAAL,CAA2BE,IAAI,CAACD,MAAL,KAAe,MAAM,IAAA6C,gBAAA,GAArB,CAA3B,EAA6D5C,IAA7D,CAAtB;;IACA,IAAI/F,eAAA,CAAEgD,OAAF,CAAUiD,aAAV,CAAJ,EAA8B;MAC5B1G,GAAG,CAACkD,KAAJ,CAAW,uCAAX;MACA,OAAO,EAAP;IACD;;IACDlD,GAAG,CAACkD,KAAJ,CAAW,OAAMzB,aAAA,CAAKqF,SAAL,CAAe,QAAf,EAAyBJ,aAAa,CAACxF,MAAvC,EAA+C,IAA/C,CAAqD,YAA5D,GACRmI,IAAI,CAACC,SAAL,CAAe5C,aAAf,EAA8B,IAA9B,EAAoC,CAApC,CADF;IAGA,MAAM6C,mBAAmB,GAAG,EAA5B;IACA,MAAM7F,QAAQ,GAAG,EAAjB;IACA,MAAMyE,YAAY,GAAG,MAAM7C,gBAAA,CAAQC,OAAR,EAA3B;;IACA,IAAI;MACF,KAAK,MAAM,CAACiE,GAAD,EAAM5G,SAAN,CAAX,IAA+B8D,aAAa,CAAC+C,OAAd,EAA/B,EAAwD;QACtD/F,QAAQ,CAACgB,IAAT,CAAc,CAAC,YAAY;UACzB,IAAI,MAAM,KAAKuD,cAAL,CAAoBuB,GAApB,EAAyB5G,SAAzB,EAAoCuF,YAApC,EAAkD,CAAC1H,eAAA,CAAEgD,OAAF,CAAU+C,IAAV,CAAnD,CAAV,EAA+E;YAC7E+C,mBAAmB,CAAC7E,IAApB,CAAyB9B,SAAzB;UACD;QACF,CAJa,GAAd;;QAMA,IAAIc,QAAQ,CAACxC,MAAT,GAAkBnB,sBAAlB,KAA6C,CAAjD,EAAoD;UAClD,MAAM4E,iBAAA,CAAEC,GAAF,CAAMlB,QAAN,CAAN;QACD;MACF;;MACD,MAAMiB,iBAAA,CAAEC,GAAF,CAAMlB,QAAN,CAAN;IACD,CAbD,SAaU;MACR,MAAMnD,WAAA,CAAG8F,MAAH,CAAU8B,YAAV,CAAN;IACD;;IACD,IAAI,CAAC1H,eAAA,CAAEgD,OAAF,CAAU8F,mBAAV,CAAL,EAAqC;MACnCvJ,GAAG,CAACyE,IAAJ,CAAU,4BAAD,GACN,GAAEhD,aAAA,CAAKqF,SAAL,CAAe,cAAf,EAA+ByC,mBAAmB,CAACrI,MAAnD,EAA2D,IAA3D,CAAiE,EADtE;IAED,CAHD,MAGO;MACLlB,GAAG,CAACyE,IAAJ,CAAU,oCAAV;IACD;;IACD,OAAO8E,mBAAP;EACD;;AAlY6B;;eAsYjB3H,yB"}
|
|
303
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["TIMEOUT_MS","MAX_PARALLEL_DOWNLOADS","log","logger","getLogger","isCrcOk","src","checksum","md5","fs","hash","_","toLower","findChildNode","parent","childName","text","hasChildNodes","childNodeIdx","childNodes","length","childNode","localName","childText","extractNodeText","node","firstChild","util","hasValue","nodeValue","ChromedriverStorageClient","constructor","args","chromedriverDir","getChromedriverDir","timeout","mapping","parseNotes","content","result","versionMatch","exec","version","minBrowserVersionMatch","minBrowserVersion","retrieveAdditionalDriverInfo","driverKey","notesUrl","infoDict","notes","retrieveData","accept","debug","parseStorageXml","doc","shouldParseNotes","driverNodes","xpath","select","isEmpty","promises","driverNode","key","includes","etag","cdInfo","url","CD_CDN","trim","first","split","notesPath","isNotesPresent","reduce","acc","info","push","B","all","size","retrieveMapping","xml","DOMParser","parseFromString","cloneDeep","unzipDriver","dst","tmpRoot","tempDir","openDir","zip","extractAllTo","chromedriverPath","walkDir","itemPath","isDirectory","path","parse","name","Error","basename","mv","mkdirp","rimraf","selectMatchingDrivers","osInfo","opts","versions","driversToSync","keys","filter","cdName","pluralize","isNaN","minBrowserVersionInt","parseInt","closestMatchedVersionNumber","currentMinBrowserVersion","uniq","map","arch","hardwareName","X64","some","X86","OS","mac","M1_ARCH_SUFFIX","platformRe","RegExp","test","retrieveDriver","index","archivesRoot","isStrict","archivePath","resolve","net","downloadFile","isMetered","e","msg","message","error","fileName","system","isWindows","targetPath","chmod","syncDrivers","getOsInfo","JSON","stringify","synchronizedDrivers","idx","entries"],"sources":["../../lib/storage-client.js"],"sourcesContent":["import {\n  getChromedriverDir, CD_CDN, retrieveData, getOsInfo,\n  OS, X64, X86, M1_ARCH_SUFFIX,\n} from './utils';\nimport _ from 'lodash';\nimport xpath from 'xpath';\nimport { DOMParser } from '@xmldom/xmldom';\nimport B from 'bluebird';\nimport path from 'path';\nimport { system, fs, logger, tempDir, zip, util, net } from 'appium/support';\n\n\nconst TIMEOUT_MS = 15000;\nconst MAX_PARALLEL_DOWNLOADS = 5;\n\nconst log = logger.getLogger('ChromedriverStorageClient');\n\n\nasync function isCrcOk (src, checksum) {\n  const md5 = await fs.hash(src, 'md5');\n  return _.toLower(md5) === _.toLower(checksum);\n}\n\nfunction findChildNode (parent, childName = null, text = null) {\n  if (!childName && !text) {\n    return null;\n  }\n  if (!parent.hasChildNodes()) {\n    return null;\n  }\n\n  for (let childNodeIdx = 0; childNodeIdx < parent.childNodes.length; childNodeIdx++) {\n    const childNode = parent.childNodes[childNodeIdx];\n    if (childName && !text && childName === childNode.localName) {\n      return childNode;\n    }\n    if (text) {\n      const childText = extractNodeText(childNode);\n      if (!childText) {\n        continue;\n      }\n      if (childName && childName === childNode.localName && text === childText) {\n        return childNode;\n      }\n      if (!childName && text === childText) {\n        return childNode;\n      }\n    }\n  }\n  return null;\n}\n\nfunction extractNodeText (node) {\n  return (!node || !node.firstChild || !util.hasValue(node.firstChild.nodeValue))\n    ? null\n    : node.firstChild.nodeValue;\n}\n\n\nclass ChromedriverStorageClient {\n  constructor (args = {}) {\n    const {\n      chromedriverDir = getChromedriverDir(),\n      timeout = TIMEOUT_MS,\n    } = args;\n    this.chromedriverDir = chromedriverDir;\n    this.timeout = timeout;\n    this.mapping = {};\n  }\n\n  /**\n   * @typedef {Object} AdditionalDriverDetails\n   * @property {?string} version - Chromedriver version\n   * or `null` if it cannot be found\n   * @property {?string} minBrowserVersion - The minimum browser version\n   * supported by chromedriver or `null` if it cannot be found\n   */\n\n  /**\n   * Gets additional chromedriver details from chromedriver\n   * release notes\n   *\n   * @param {string} content - Release notes of the corresponding chromedriver\n   * @returns {AdditionalDriverDetails}\n   */\n  parseNotes (content) {\n    const result = {};\n    const versionMatch = /^\\s*[-]+ChromeDriver[\\D]+([\\d.]+)/im.exec(content);\n    if (versionMatch) {\n      result.version = versionMatch[1];\n    }\n    const minBrowserVersionMatch = /^\\s*Supports Chrome[\\D]+(\\d+)/im.exec(content);\n    if (minBrowserVersionMatch) {\n      result.minBrowserVersion = minBrowserVersionMatch[1];\n    }\n    return result;\n  }\n\n  /**\n   * Downloads chromedriver release notes and puts them\n   * into the dictionary argument\n   *\n   * @param {string} driverKey - Driver version plus archive name\n   * @param {string} notesUrl - The URL of chromedriver notes\n   * @param {Object} infoDict - The dictionary containing driver info.\n   * The method call mutates by merging `AdditionalDriverDetails`\n   * @throws {Error} if the release notes cannot be downloaded\n   */\n  async retrieveAdditionalDriverInfo (driverKey, notesUrl, infoDict) {\n    const notes = await retrieveData(notesUrl, {\n      'user-agent': 'appium',\n      accept: '*/*',\n    }, { timeout: this.timeout });\n    const { minBrowserVersion } = this.parseNotes(notes);\n    if (!minBrowserVersion) {\n      log.debug(`The driver '${driverKey}' does not contain valid release notes at ${notesUrl}. ` +\n        `Skipping it`);\n      return;\n    }\n    infoDict.minBrowserVersion = minBrowserVersion;\n  }\n\n  /**\n   * Parses chromedriver storage XML and stores\n   * the parsed results into `this.mapping`\n   *\n   * @param {DOMDocument} doc - The DOM representation\n   * of the chromedriver storage XML\n   * @param {boolean} shouldParseNotes [true] - If set to `true`\n   * then additional drivers information is going to be parsed\n   * and assigned to `this.mapping`\n   */\n  async parseStorageXml (doc, shouldParseNotes = true) {\n    const driverNodes = xpath.select(`//*[local-name(.)='Contents']`, doc);\n    log.debug(`Parsed ${driverNodes.length} entries from storage XML`);\n    if (_.isEmpty(driverNodes)) {\n      return;\n    }\n\n    const promises = [];\n    for (const driverNode of driverNodes) {\n      const key = extractNodeText(findChildNode(driverNode, 'Key'));\n      if (!_.includes(key, '/chromedriver_')) {\n        continue;\n      }\n\n      const etag = extractNodeText(findChildNode(driverNode, 'ETag'));\n      if (!etag) {\n        log.debug(`The entry '${key}' does not contain the checksum. Skipping it`);\n        continue;\n      }\n\n      const cdInfo = {\n        url: `${CD_CDN}/${key}`,\n        etag: _.trim(etag, '\"'),\n        version: _.first(key.split('/')),\n      };\n      this.mapping[key] = cdInfo;\n\n      const notesPath = `${cdInfo.version}/notes.txt`;\n      const isNotesPresent = !!driverNodes\n        .reduce((acc, node) => acc || findChildNode(node, 'Key', notesPath), false);\n      if (!isNotesPresent) {\n        cdInfo.minBrowserVersion = null;\n        if (shouldParseNotes) {\n          log.info(`The entry '${key}' does not contain any notes. Skipping it`);\n        }\n        continue;\n      } else if (!shouldParseNotes) {\n        continue;\n      }\n\n      promises.push(this.retrieveAdditionalDriverInfo(key, `${CD_CDN}/${notesPath}`, cdInfo));\n      if (promises.length % MAX_PARALLEL_DOWNLOADS === 0) {\n        await B.all(promises);\n      }\n    }\n    await B.all(promises);\n    log.info(`The total count of entries in the mapping: ${_.size(this.mapping)}`);\n  }\n\n  /**\n   * @typedef {Object} DriverDetails\n   * @property {string} url - The full url to the corresponding driver in\n   * the remote storage\n   * @property {string} etag - The CRC of the driver archive\n   * @property {string} version - Chromedriver version\n   */\n\n  /**\n   * @typedef {Object} ChromedriversMapping\n   * @property {DriverDetails} - The keys are unique driver identifiers\n   * (version/archive name). The corresponding values have `DriverDetails`\n   * containing chromedriver details\n   */\n\n  /**\n   * Retrieves chromedriver mapping from the storage\n   *\n   * @param {boolean} shouldParseNotes [true] - if set to `true`\n   * then additional chromedrivers info is going to be retrieved and\n   * parsed from release notes\n   * @returns {ChromedriversMapping}\n   */\n  async retrieveMapping (shouldParseNotes = true) {\n    const xml = await retrieveData(CD_CDN, {\n      'user-agent': 'appium',\n      accept: 'application/xml, */*',\n    }, { timeout: this.timeout });\n    const doc = new DOMParser().parseFromString(xml);\n    await this.parseStorageXml(doc, shouldParseNotes);\n    return _.cloneDeep(this.mapping);\n  }\n\n  /**\n   * Extracts downloaded chromedriver archive\n   * into the given destination\n   *\n   * @param {string} src - The source archive path\n   * @param {string} dst - The destination chromedriver path\n   */\n  async unzipDriver (src, dst) {\n    const tmpRoot = await tempDir.openDir();\n    try {\n      await zip.extractAllTo(src, tmpRoot);\n      const chromedriverPath = await fs.walkDir(tmpRoot, true, (itemPath, isDirectory) =>\n        !isDirectory && _.toLower(path.parse(itemPath).name) === 'chromedriver');\n      if (!chromedriverPath) {\n        throw new Error('The archive was unzipped properly, but we could not find any chromedriver executable');\n      }\n      log.debug(`Moving the extracted '${path.basename(chromedriverPath)}' to '${dst}'`);\n      await fs.mv(chromedriverPath, dst, {\n        mkdirp: true\n      });\n    } finally {\n      await fs.rimraf(tmpRoot);\n    }\n  }\n\n  /**\n   * @typedef {Object} OSInfo\n   * @property {string} name - The name of the host OS\n   * Can be either `mac`, `windows` or `linux`\n   * @property {string} arch - The architecture of the host OD.\n   * Can be either `32` or `64`\n   * @property {?string} hardwareName - The output of `uname -m` command\n   * on linux and mac systems. `null` on Windows\n   */\n\n  /**\n   * Filters `this.mapping` to only select matching\n   * chromedriver entries by operating system information\n   * and/or additional synchronization options (if provided)\n   *\n   * @param {?OSInfo} osInfo\n   * @param {?SyncOptions} opts\n   * @returns {Array<String>} The list of filtered chromedriver\n   * entry names (version/archive name)\n   */\n  selectMatchingDrivers (osInfo, opts = {}) {\n    const {\n      minBrowserVersion,\n      versions = [],\n    } = opts;\n    let driversToSync = _.keys(this.mapping);\n\n    if (!_.isEmpty(versions)) {\n      // Handle only selected versions if requested\n      log.debug(`Selecting chromedrivers whose versions match to ${versions}`);\n      driversToSync = driversToSync\n        .filter((cdName) => versions.includes(`${this.mapping[cdName].version}`));\n\n      log.debug(`Got ${util.pluralize('item', driversToSync.length, true)}`);\n      if (_.isEmpty(driversToSync)) {\n        return [];\n      }\n    }\n\n    if (!isNaN(minBrowserVersion)) {\n      // Only select drivers that support the current browser whose major version number equals to `minBrowserVersion`\n      const minBrowserVersionInt = parseInt(minBrowserVersion, 10);\n      log.debug(`Selecting chromedrivers whose minimum supported browser version matches to ${minBrowserVersionInt}`);\n      let closestMatchedVersionNumber = 0;\n      // Select the newest available and compatible chromedriver\n      for (const cdName of driversToSync) {\n        const currentMinBrowserVersion = parseInt(this.mapping[cdName].minBrowserVersion, 10);\n        if (!isNaN(currentMinBrowserVersion)\n            && currentMinBrowserVersion <= minBrowserVersionInt\n            && closestMatchedVersionNumber < currentMinBrowserVersion) {\n          closestMatchedVersionNumber = currentMinBrowserVersion;\n        }\n      }\n      driversToSync = driversToSync.filter((cdName) => `${this.mapping[cdName].minBrowserVersion}` ===\n        `${closestMatchedVersionNumber > 0 ? closestMatchedVersionNumber : minBrowserVersionInt}`);\n\n      log.debug(`Got ${util.pluralize('item', driversToSync.length, true)}`);\n      if (_.isEmpty(driversToSync)) {\n        return [];\n      }\n      log.debug(`Will select candidate ${util.pluralize('driver', driversToSync.length)} ` +\n        `versioned as '${_.uniq(driversToSync.map((cdName) => this.mapping[cdName].version))}'`);\n    }\n\n    if (!_.isEmpty(osInfo)) {\n      // Filter out drivers for unsupported system architectures\n      let {name, arch, hardwareName} = osInfo;\n      if (arch === X64 && !driversToSync.some((cdName) => cdName.includes(`_${name}${X64}`))) {\n        // Fall back to x86 build if x64 one is not available for the given OS\n        arch = X86;\n      }\n      if (name === OS.mac && _.includes(hardwareName, 'arm')\n          && driversToSync.some((cdName) => cdName.includes(M1_ARCH_SUFFIX))) {\n        // prefer executable for M1 arch if present\n        arch += M1_ARCH_SUFFIX;\n      }\n      log.debug(`Selecting chromedrivers whose platform matches to ${name}${arch}`);\n      const platformRe = new RegExp(`(\\\\b|_)${name}${arch}\\\\b`);\n      driversToSync = driversToSync.filter((cdName) => platformRe.test(cdName));\n      log.debug(`Got ${util.pluralize('item', driversToSync.length, true)}`);\n    }\n\n    return driversToSync;\n  }\n\n  /**\n   * Retrieves the given chromedriver from the storage\n   * and unpacks it into `this.chromedriverDir` folder\n   *\n   * @param {number} index - The unique driver index\n   * @param {string} driverKey - The driver key in `this.mapping`\n   * @param {string} archivesRoot - The temporary folder path to extract\n   * downloaded archives to\n   * @param {boolean} isStrict [true] - Whether to throw an error (`true`)\n   * or return a boolean result if the driver retrieval process fails\n   * @throws {Error} if there was a failure while retrieving the driver\n   * and `isStrict` is set to `true`\n   * @returns {boolean} if `true` then the chromedriver is successfully\n   * downloaded and extracted.\n   */\n  async retrieveDriver (index, driverKey, archivesRoot, isStrict = false) {\n    const { url, etag, version } = this.mapping[driverKey];\n    const archivePath = path.resolve(archivesRoot, `${index}.zip`);\n    log.debug(`Retrieving '${url}' to '${archivePath}'`);\n    try {\n      await net.downloadFile(url, archivePath, {\n        isMetered: false,\n        timeout: TIMEOUT_MS\n      });\n    } catch (e) {\n      const msg = `Cannot download chromedriver archive. Original error: ${e.message}`;\n      if (isStrict) {\n        throw new Error(msg);\n      }\n      log.error(msg);\n      return false;\n    }\n    if (!await isCrcOk(archivePath, etag)) {\n      const msg = `The checksum for the downloaded chromedriver '${driverKey}' did not match`;\n      if (isStrict) {\n        throw new Error(msg);\n      }\n      log.error(msg);\n      return false;\n    }\n    const fileName = `${path.parse(url).name}_v${version}` +\n      (system.isWindows() ? '.exe' : '');\n    const targetPath = path.resolve(this.chromedriverDir, fileName);\n    try {\n      await this.unzipDriver(archivePath, targetPath);\n      await fs.chmod(targetPath, 0o755);\n      log.debug(`Permissions of the file '${targetPath}' have been changed to 755`);\n    } catch (e) {\n      if (isStrict) {\n        throw e;\n      }\n      log.error(e.message);\n      return false;\n    }\n    return true;\n  }\n\n  /**\n   * @typedef {Object} SyncOptions\n   * @property {Array<String>} versions - The list of chromedriver\n   * versions to sync. If empty (the default value) then all available\n   * chromedrivers are going to be downloaded and extracted\n   * @property {string|number} minBrowserVersion - The minumum supported\n   * Chrome version that downloaded chromedrivers should support. Can match\n   * multiple drivers.\n   * @property {?OSInfo} osInfo - System information used to filter out\n   * the list of the retrieved drivers. If not provided then the script\n   * will try to retrieve it.\n   */\n\n  /**\n   * Retrieves chromedrivers from the remote storage\n   * to the local file system\n   *\n   * @param {?SyncOptions} opts\n   * @throws {Error} if there was a problem while retrieving\n   * the drivers\n   * @returns {Array<String} The list of successfully synchronized driver keys\n   */\n  async syncDrivers (opts = {}) {\n    if (_.isEmpty(this.mapping)) {\n      await this.retrieveMapping(!!opts.minBrowserVersion);\n    }\n    if (_.isEmpty(this.mapping)) {\n      throw new Error('Cannot retrieve chromedrivers mapping from Google storage');\n    }\n\n    const driversToSync = this.selectMatchingDrivers(opts.osInfo ?? await getOsInfo(), opts);\n    if (_.isEmpty(driversToSync)) {\n      log.debug(`There are no drivers to sync. Exiting`);\n      return [];\n    }\n    log.debug(`Got ${util.pluralize('driver', driversToSync.length, true)} to sync: ` +\n      JSON.stringify(driversToSync, null, 2));\n\n    const synchronizedDrivers = [];\n    const promises = [];\n    const archivesRoot = await tempDir.openDir();\n    try {\n      for (const [idx, driverKey] of driversToSync.entries()) {\n        promises.push((async () => {\n          if (await this.retrieveDriver(idx, driverKey, archivesRoot, !_.isEmpty(opts))) {\n            synchronizedDrivers.push(driverKey);\n          }\n        })());\n\n        if (promises.length % MAX_PARALLEL_DOWNLOADS === 0) {\n          await B.all(promises);\n        }\n      }\n      await B.all(promises);\n    } finally {\n      await fs.rimraf(archivesRoot);\n    }\n    if (!_.isEmpty(synchronizedDrivers)) {\n      log.info(`Successfully synchronized ` +\n        `${util.pluralize('chromedriver', synchronizedDrivers.length, true)}`);\n    } else {\n      log.info(`No chromedrivers were synchronized`);\n    }\n    return synchronizedDrivers;\n  }\n}\n\n\nexport default ChromedriverStorageClient;\n"],"mappings":"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"}
|