appium-chromedriver 5.0.0 → 5.0.3
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/build/index.js +26 -0
- package/build/lib/chromedriver.js +72 -65
- package/build/lib/utils.js +11 -2
- package/config/mapping.json +4 -0
- package/lib/chromedriver.js +70 -67
- package/lib/utils.js +15 -1
- package/package.json +6 -8
package/build/index.js
ADDED
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
+
|
|
5
|
+
Object.defineProperty(exports, "__esModule", {
|
|
6
|
+
value: true
|
|
7
|
+
});
|
|
8
|
+
Object.defineProperty(exports, "ChromedriverStorageClient", {
|
|
9
|
+
enumerable: true,
|
|
10
|
+
get: function () {
|
|
11
|
+
return _storageClient.default;
|
|
12
|
+
}
|
|
13
|
+
});
|
|
14
|
+
exports.default = void 0;
|
|
15
|
+
|
|
16
|
+
require("source-map-support/register");
|
|
17
|
+
|
|
18
|
+
var _chromedriver = _interopRequireDefault(require("./lib/chromedriver"));
|
|
19
|
+
|
|
20
|
+
var _storageClient = _interopRequireDefault(require("./lib/storage-client"));
|
|
21
|
+
|
|
22
|
+
var _default = _chromedriver.default;
|
|
23
|
+
exports.default = _default;require('source-map-support').install();
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImluZGV4LmpzIl0sIm5hbWVzIjpbImNocm9tZWRyaXZlciJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7ZUFHZUEscUIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBkZWZhdWx0IGFzIGNocm9tZWRyaXZlciB9IGZyb20gJy4vbGliL2Nocm9tZWRyaXZlcic7XG5pbXBvcnQgQ2hyb21lZHJpdmVyU3RvcmFnZUNsaWVudCBmcm9tICcuL2xpYi9zdG9yYWdlLWNsaWVudCc7XG5cbmV4cG9ydCB7IENocm9tZWRyaXZlclN0b3JhZ2VDbGllbnQgfTtcbmV4cG9ydCBkZWZhdWx0IGNocm9tZWRyaXZlcjtcbiJdLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiLi4ifQ==
|
|
@@ -37,8 +37,6 @@ var _storageClient = _interopRequireDefault(require("./storage-client"));
|
|
|
37
37
|
|
|
38
38
|
var _protocolHelpers = require("./protocol-helpers");
|
|
39
39
|
|
|
40
|
-
const log = _support.logger.getLogger('Chromedriver');
|
|
41
|
-
|
|
42
40
|
const NEW_CD_VERSION_FORMAT_MAJOR_VERSION = 73;
|
|
43
41
|
const DEFAULT_HOST = '127.0.0.1';
|
|
44
42
|
const MIN_CD_VERSION_WITH_W3C_SUPPORT = 75;
|
|
@@ -69,6 +67,7 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
69
67
|
details,
|
|
70
68
|
isAutodownloadEnabled = false
|
|
71
69
|
} = args;
|
|
70
|
+
this._log = _support.logger.getLogger((0, _utils.generateLogPrefix)(this));
|
|
72
71
|
this.proxyHost = host;
|
|
73
72
|
this.proxyPort = port;
|
|
74
73
|
this.adb = adb;
|
|
@@ -83,7 +82,8 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
83
82
|
this.state = Chromedriver.STATE_STOPPED;
|
|
84
83
|
this.jwproxy = new _baseDriver.JWProxy({
|
|
85
84
|
server: this.proxyHost,
|
|
86
|
-
port: this.proxyPort
|
|
85
|
+
port: this.proxyPort,
|
|
86
|
+
log: this._log
|
|
87
87
|
});
|
|
88
88
|
this.verbose = verbose;
|
|
89
89
|
this.logPath = logPath;
|
|
@@ -96,25 +96,29 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
96
96
|
this.desiredProtocol = _baseDriver.PROTOCOLS.MJSONWP;
|
|
97
97
|
}
|
|
98
98
|
|
|
99
|
+
get log() {
|
|
100
|
+
return this._log;
|
|
101
|
+
}
|
|
102
|
+
|
|
99
103
|
async getDriversMapping() {
|
|
100
104
|
let mapping = _lodash.default.cloneDeep(_utils.CHROMEDRIVER_CHROME_MAPPING);
|
|
101
105
|
|
|
102
106
|
if (this.mappingPath) {
|
|
103
|
-
log.debug(`Attempting to use Chromedriver->Chrome mapping from '${this.mappingPath}'`);
|
|
107
|
+
this.log.debug(`Attempting to use Chromedriver->Chrome mapping from '${this.mappingPath}'`);
|
|
104
108
|
|
|
105
109
|
if (!(await _support.fs.exists(this.mappingPath))) {
|
|
106
|
-
log.warn(`No file found at '${this.mappingPath}'`);
|
|
107
|
-
log.info('Defaulting to the static Chromedriver->Chrome mapping');
|
|
110
|
+
this.log.warn(`No file found at '${this.mappingPath}'`);
|
|
111
|
+
this.log.info('Defaulting to the static Chromedriver->Chrome mapping');
|
|
108
112
|
} else {
|
|
109
113
|
try {
|
|
110
114
|
mapping = JSON.parse(await _support.fs.readFile(this.mappingPath, 'utf8'));
|
|
111
115
|
} catch (err) {
|
|
112
|
-
log.warn(`Error parsing mapping from '${this.mappingPath}': ${err.message}`);
|
|
113
|
-
log.info('Defaulting to the static Chromedriver->Chrome mapping');
|
|
116
|
+
this.log.warn(`Error parsing mapping from '${this.mappingPath}': ${err.message}`);
|
|
117
|
+
this.log.info('Defaulting to the static Chromedriver->Chrome mapping');
|
|
114
118
|
}
|
|
115
119
|
}
|
|
116
120
|
} else {
|
|
117
|
-
log.debug('Using the static Chromedriver->Chrome mapping');
|
|
121
|
+
this.log.debug('Using the static Chromedriver->Chrome mapping');
|
|
118
122
|
}
|
|
119
123
|
|
|
120
124
|
for (const [cdVersion, chromeVersion] of _lodash.default.toPairs(mapping)) {
|
|
@@ -123,7 +127,7 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
123
127
|
if (coercedVersion) {
|
|
124
128
|
mapping[cdVersion] = coercedVersion.version;
|
|
125
129
|
} else {
|
|
126
|
-
log.info(`'${chromeVersion}' is not a valid version number. Skipping it`);
|
|
130
|
+
this.log.info(`'${chromeVersion}' is not a valid version number. Skipping it`);
|
|
127
131
|
}
|
|
128
132
|
}
|
|
129
133
|
|
|
@@ -132,7 +136,7 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
132
136
|
|
|
133
137
|
async getChromedrivers(mapping) {
|
|
134
138
|
const executables = await _support.fs.glob(`${this.executableDir}/*`);
|
|
135
|
-
log.debug(`Found ${_support.util.pluralize('executable', executables.length, true)} ` + `in '${this.executableDir}'`);
|
|
139
|
+
this.log.debug(`Found ${_support.util.pluralize('executable', executables.length, true)} ` + `in '${this.executableDir}'`);
|
|
136
140
|
const cds = (await (0, _asyncbox.asyncmap)(executables, async function mapChromedriver(executable) {
|
|
137
141
|
const logError = ({
|
|
138
142
|
message,
|
|
@@ -149,7 +153,7 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
149
153
|
errMsg += `\nStderr: ${stderr}`;
|
|
150
154
|
}
|
|
151
155
|
|
|
152
|
-
log.warn(errMsg);
|
|
156
|
+
this.log.warn(errMsg);
|
|
153
157
|
return null;
|
|
154
158
|
};
|
|
155
159
|
|
|
@@ -205,14 +209,14 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
205
209
|
})).filter(cd => !!cd).sort((a, b) => (0, _compareVersions.default)(b.version, a.version));
|
|
206
210
|
|
|
207
211
|
if (_lodash.default.isEmpty(cds)) {
|
|
208
|
-
log.info(`No Chromedrivers were found in '${this.executableDir}'`);
|
|
212
|
+
this.log.info(`No Chromedrivers were found in '${this.executableDir}'`);
|
|
209
213
|
return cds;
|
|
210
214
|
}
|
|
211
215
|
|
|
212
|
-
log.debug(`The following Chromedriver executables were found:`);
|
|
216
|
+
this.log.debug(`The following Chromedriver executables were found:`);
|
|
213
217
|
|
|
214
218
|
for (const cd of cds) {
|
|
215
|
-
log.debug(` '${cd.executable}' (version '${cd.version}', minimum Chrome version '${cd.minChromeVersion ? cd.minChromeVersion : 'Unknown'}')`);
|
|
219
|
+
this.log.debug(` '${cd.executable}' (version '${cd.version}', minimum Chrome version '${cd.minChromeVersion ? cd.minChromeVersion : 'Unknown'}')`);
|
|
216
220
|
}
|
|
217
221
|
|
|
218
222
|
return cds;
|
|
@@ -224,7 +228,7 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
224
228
|
if ((_this$details = this.details) !== null && _this$details !== void 0 && _this$details.info) {
|
|
225
229
|
var _this$details2, _this$details2$info;
|
|
226
230
|
|
|
227
|
-
log.debug(`Browser version in the supplied details: ${(_this$details2 = this.details) === null || _this$details2 === void 0 ? void 0 : (_this$details2$info = _this$details2.info) === null || _this$details2$info === void 0 ? void 0 : _this$details2$info.Browser}`);
|
|
231
|
+
this.log.debug(`Browser version in the supplied details: ${(_this$details2 = this.details) === null || _this$details2 === void 0 ? void 0 : (_this$details2$info = _this$details2.info) === null || _this$details2$info === void 0 ? void 0 : _this$details2$info.Browser}`);
|
|
228
232
|
}
|
|
229
233
|
|
|
230
234
|
const versionMatch = VERSION_PATTERN.exec((_this$details3 = this.details) === null || _this$details3 === void 0 ? void 0 : (_this$details3$info = _this$details3.info) === null || _this$details3$info === void 0 ? void 0 : _this$details3$info.Browser);
|
|
@@ -288,7 +292,7 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
288
292
|
await _support.fs.writeFile(this.mappingPath, JSON.stringify(newMapping, null, 2), 'utf8');
|
|
289
293
|
shouldUpdateStaticMapping = false;
|
|
290
294
|
} catch (e) {
|
|
291
|
-
log.warn(`Cannot store the updated chromedrivers mapping into '${this.mappingPath}'. ` + `This may reduce the performance of further executions. Original error: ${e.message}`);
|
|
295
|
+
this.log.warn(`Cannot store the updated chromedrivers mapping into '${this.mappingPath}'. ` + `This may reduce the performance of further executions. Original error: ${e.message}`);
|
|
292
296
|
}
|
|
293
297
|
}
|
|
294
298
|
|
|
@@ -305,7 +309,7 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
305
309
|
const mapping = await this.getDriversMapping();
|
|
306
310
|
|
|
307
311
|
if (!_lodash.default.isEmpty(mapping)) {
|
|
308
|
-
log.debug(`The most recent known Chrome version: ${_lodash.default.values(mapping)[0]}`);
|
|
312
|
+
this.log.debug(`The most recent known Chrome version: ${_lodash.default.values(mapping)[0]}`);
|
|
309
313
|
}
|
|
310
314
|
|
|
311
315
|
let didStorageSync = false;
|
|
@@ -313,7 +317,7 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
313
317
|
const syncChromedrivers = async chromeVersion => {
|
|
314
318
|
didStorageSync = true;
|
|
315
319
|
const retrievedMapping = await this.storageClient.retrieveMapping();
|
|
316
|
-
log.debug('Got chromedrivers mapping from the storage: ' + JSON.stringify(retrievedMapping, null, 2));
|
|
320
|
+
this.log.debug('Got chromedrivers mapping from the storage: ' + JSON.stringify(retrievedMapping, null, 2));
|
|
317
321
|
const driverKeys = await this.storageClient.syncDrivers({
|
|
318
322
|
minBrowserVersion: chromeVersion.major
|
|
319
323
|
});
|
|
@@ -357,21 +361,21 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
357
361
|
}
|
|
358
362
|
|
|
359
363
|
if (!_lodash.default.isEmpty(missingVersions)) {
|
|
360
|
-
log.info(`Found ${_support.util.pluralize('Chromedriver', _lodash.default.size(missingVersions), true)}, ` + `which ${_lodash.default.size(missingVersions) === 1 ? 'is' : 'are'} missing in the list of known versions: ` + JSON.stringify(missingVersions));
|
|
364
|
+
this.log.info(`Found ${_support.util.pluralize('Chromedriver', _lodash.default.size(missingVersions), true)}, ` + `which ${_lodash.default.size(missingVersions) === 1 ? 'is' : 'are'} missing in the list of known versions: ` + JSON.stringify(missingVersions));
|
|
361
365
|
await this.updateDriversMapping(Object.assign(mapping, missingVersions));
|
|
362
366
|
}
|
|
363
367
|
|
|
364
368
|
if (this.disableBuildCheck) {
|
|
365
369
|
if (_lodash.default.isEmpty(cds)) {
|
|
366
|
-
log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` + `'chromedriverDisableBuildCheck' capability is set to 'true'`);
|
|
370
|
+
this.log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` + `'chromedriverDisableBuildCheck' capability is set to 'true'`);
|
|
367
371
|
}
|
|
368
372
|
|
|
369
373
|
const {
|
|
370
374
|
version,
|
|
371
375
|
executable
|
|
372
376
|
} = cds[0];
|
|
373
|
-
log.warn(`Chrome build check disabled. Using most recent Chromedriver version (${version}, at '${executable}')`);
|
|
374
|
-
log.warn(`If this is wrong, set 'chromedriverDisableBuildCheck' capability to 'false'`);
|
|
377
|
+
this.log.warn(`Chrome build check disabled. Using most recent Chromedriver version (${version}, at '${executable}')`);
|
|
378
|
+
this.log.warn(`If this is wrong, set 'chromedriverDisableBuildCheck' capability to 'false'`);
|
|
375
379
|
return executable;
|
|
376
380
|
}
|
|
377
381
|
|
|
@@ -379,18 +383,18 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
379
383
|
|
|
380
384
|
if (!chromeVersion) {
|
|
381
385
|
if (_lodash.default.isEmpty(cds)) {
|
|
382
|
-
log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` + `the current Chrome version cannot be determined`);
|
|
386
|
+
this.log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` + `the current Chrome version cannot be determined`);
|
|
383
387
|
}
|
|
384
388
|
|
|
385
389
|
const {
|
|
386
390
|
version,
|
|
387
391
|
executable
|
|
388
392
|
} = cds[0];
|
|
389
|
-
log.warn(`Unable to discover Chrome version. Using Chromedriver ${version} at '${executable}'`);
|
|
393
|
+
this.log.warn(`Unable to discover Chrome version. Using Chromedriver ${version} at '${executable}'`);
|
|
390
394
|
return executable;
|
|
391
395
|
}
|
|
392
396
|
|
|
393
|
-
log.debug(`Found Chrome bundle '${this.bundleId}' version '${chromeVersion}'`);
|
|
397
|
+
this.log.debug(`Found Chrome bundle '${this.bundleId}' version '${chromeVersion}'`);
|
|
394
398
|
const matchingDrivers = cds.filter(({
|
|
395
399
|
minChromeVersion
|
|
396
400
|
}) => {
|
|
@@ -410,8 +414,8 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
410
414
|
continue;
|
|
411
415
|
}
|
|
412
416
|
} catch (e) {
|
|
413
|
-
log.warn(`Cannot synchronize local chromedrivers with the remote storage at ${_utils.CD_CDN}: ` + e.message);
|
|
414
|
-
log.debug(e.stack);
|
|
417
|
+
this.log.warn(`Cannot synchronize local chromedrivers with the remote storage at ${_utils.CD_CDN}: ` + e.message);
|
|
418
|
+
this.log.debug(e.stack);
|
|
415
419
|
}
|
|
416
420
|
}
|
|
417
421
|
|
|
@@ -420,8 +424,8 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
420
424
|
}
|
|
421
425
|
|
|
422
426
|
const binPath = matchingDrivers[0].executable;
|
|
423
|
-
log.debug(`Found ${_support.util.pluralize('executable', matchingDrivers.length, true)} ` + `capable of automating Chrome '${chromeVersion}'.\nChoosing the most recent, '${binPath}'.`);
|
|
424
|
-
log.debug('If a specific version is required, specify it with the `chromedriverExecutable`' + 'desired capability.');
|
|
427
|
+
this.log.debug(`Found ${_support.util.pluralize('executable', matchingDrivers.length, true)} ` + `capable of automating Chrome '${chromeVersion}'.\nChoosing the most recent, '${binPath}'.`);
|
|
428
|
+
this.log.debug('If a specific version is required, specify it with the `chromedriverExecutable`' + 'desired capability.');
|
|
425
429
|
return binPath;
|
|
426
430
|
} while (true);
|
|
427
431
|
}
|
|
@@ -438,21 +442,21 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
438
442
|
}
|
|
439
443
|
|
|
440
444
|
this.executableVerified = true;
|
|
441
|
-
log.info(`Set chromedriver binary as: ${this.chromedriver}`);
|
|
445
|
+
this.log.info(`Set chromedriver binary as: ${this.chromedriver}`);
|
|
442
446
|
}
|
|
443
447
|
|
|
444
448
|
syncProtocol(cdVersion = null) {
|
|
445
449
|
const coercedVersion = _semver.default.coerce(cdVersion);
|
|
446
450
|
|
|
447
451
|
if (!coercedVersion || coercedVersion.major < MIN_CD_VERSION_WITH_W3C_SUPPORT) {
|
|
448
|
-
log.debug(`Chromedriver v. ${cdVersion} does not fully support ${_baseDriver.PROTOCOLS.W3C} protocol. ` + `Defaulting to ${_baseDriver.PROTOCOLS.MJSONWP}`);
|
|
452
|
+
this.log.debug(`Chromedriver v. ${cdVersion} does not fully support ${_baseDriver.PROTOCOLS.W3C} protocol. ` + `Defaulting to ${_baseDriver.PROTOCOLS.MJSONWP}`);
|
|
449
453
|
return;
|
|
450
454
|
}
|
|
451
455
|
|
|
452
456
|
const chromeOptions = (0, _protocolHelpers.getCapValue)(this.capabilities, 'chromeOptions', {});
|
|
453
457
|
|
|
454
458
|
if (chromeOptions.w3c === false) {
|
|
455
|
-
log.info(`Chromedriver v. ${cdVersion} supports ${_baseDriver.PROTOCOLS.W3C} protocol, ` + `but ${_baseDriver.PROTOCOLS.MJSONWP} one has been explicitly requested`);
|
|
459
|
+
this.log.info(`Chromedriver v. ${cdVersion} supports ${_baseDriver.PROTOCOLS.W3C} protocol, ` + `but ${_baseDriver.PROTOCOLS.MJSONWP} one has been explicitly requested`);
|
|
456
460
|
return;
|
|
457
461
|
}
|
|
458
462
|
|
|
@@ -508,25 +512,25 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
508
512
|
|
|
509
513
|
if (match) {
|
|
510
514
|
webviewVersion = match[1];
|
|
511
|
-
log.debug(`Webview version: '${webviewVersion}'`);
|
|
515
|
+
this.log.debug(`Webview version: '${webviewVersion}'`);
|
|
512
516
|
}
|
|
513
517
|
|
|
514
518
|
match = /Starting ChromeDriver ([.\d]+)/.exec(out);
|
|
515
519
|
|
|
516
520
|
if (match) {
|
|
517
|
-
log.debug(`Chromedriver version: '${match[1]}'`);
|
|
521
|
+
this.log.debug(`Chromedriver version: '${match[1]}'`);
|
|
518
522
|
this.syncProtocol(match[1]);
|
|
519
523
|
}
|
|
520
524
|
|
|
521
525
|
if (this.verbose) {
|
|
522
526
|
for (let line of (stdout || '').trim().split('\n')) {
|
|
523
527
|
if (!line.trim().length) continue;
|
|
524
|
-
log.debug(`[STDOUT] ${line}`);
|
|
528
|
+
this.log.debug(`[STDOUT] ${line}`);
|
|
525
529
|
}
|
|
526
530
|
|
|
527
531
|
for (let line of (stderr || '').trim().split('\n')) {
|
|
528
532
|
if (!line.trim().length) continue;
|
|
529
|
-
log.error(`[STDERR] ${line}`);
|
|
533
|
+
this.log.error(`[STDERR] ${line}`);
|
|
530
534
|
}
|
|
531
535
|
}
|
|
532
536
|
});
|
|
@@ -534,17 +538,17 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
534
538
|
processIsAlive = false;
|
|
535
539
|
|
|
536
540
|
if (this.state !== Chromedriver.STATE_STOPPED && this.state !== Chromedriver.STATE_STOPPING && this.state !== Chromedriver.STATE_RESTARTING) {
|
|
537
|
-
|
|
538
|
-
log.error(msg);
|
|
541
|
+
const msg = `Chromedriver exited unexpectedly with code ${code}, signal ${signal}`;
|
|
542
|
+
this.log.error(msg);
|
|
539
543
|
this.changeState(Chromedriver.STATE_STOPPED);
|
|
540
544
|
}
|
|
541
545
|
});
|
|
542
|
-
log.info(`Spawning chromedriver with: ${this.chromedriver}
|
|
546
|
+
this.log.info(`Spawning chromedriver with: ${this.chromedriver} ${args.join(' ')}`);
|
|
543
547
|
await this.proc.start(startDetector);
|
|
544
548
|
await this.waitForOnline();
|
|
545
549
|
await this.startSession();
|
|
546
550
|
} catch (e) {
|
|
547
|
-
log.debug(e);
|
|
551
|
+
this.log.debug(e);
|
|
548
552
|
this.emit(Chromedriver.EVENT_ERROR, e);
|
|
549
553
|
|
|
550
554
|
if (processIsAlive) {
|
|
@@ -572,20 +576,16 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
572
576
|
}
|
|
573
577
|
|
|
574
578
|
message += e.message;
|
|
575
|
-
log.errorAndThrow(message);
|
|
579
|
+
this.log.errorAndThrow(message);
|
|
576
580
|
}
|
|
577
581
|
}
|
|
578
582
|
|
|
579
583
|
sessionId() {
|
|
580
|
-
|
|
581
|
-
return null;
|
|
582
|
-
}
|
|
583
|
-
|
|
584
|
-
return this.jwproxy.sessionId;
|
|
584
|
+
return this.state === Chromedriver.STATE_ONLINE ? this.jwproxy.sessionId : null;
|
|
585
585
|
}
|
|
586
586
|
|
|
587
587
|
async restart() {
|
|
588
|
-
log.info('Restarting chromedriver');
|
|
588
|
+
this.log.info('Restarting chromedriver');
|
|
589
589
|
|
|
590
590
|
if (this.state !== Chromedriver.STATE_ONLINE) {
|
|
591
591
|
throw new Error("Can't restart when we're not online");
|
|
@@ -624,8 +624,9 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
624
624
|
} : {
|
|
625
625
|
desiredCapabilities: this.capabilities
|
|
626
626
|
};
|
|
627
|
-
log.info(`Starting ${this.desiredProtocol} Chromedriver session with capabilities: ` + JSON.stringify(sessionCaps, null, 2));
|
|
627
|
+
this.log.info(`Starting ${this.desiredProtocol} Chromedriver session with capabilities: ` + JSON.stringify(sessionCaps, null, 2));
|
|
628
628
|
await this.jwproxy.command('/session', 'POST', sessionCaps);
|
|
629
|
+
this.log.prefix = (0, _utils.generateLogPrefix)(this, this.jwproxy.sessionId);
|
|
629
630
|
this.changeState(Chromedriver.STATE_ONLINE);
|
|
630
631
|
}
|
|
631
632
|
|
|
@@ -634,21 +635,27 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
634
635
|
this.changeState(Chromedriver.STATE_STOPPING);
|
|
635
636
|
}
|
|
636
637
|
|
|
637
|
-
|
|
638
|
-
|
|
639
|
-
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
this.
|
|
638
|
+
const runSafeStep = async f => {
|
|
639
|
+
try {
|
|
640
|
+
return await f();
|
|
641
|
+
} catch (e) {
|
|
642
|
+
this.log.warn(e.message);
|
|
643
|
+
this.log.debug(e.stack);
|
|
643
644
|
}
|
|
644
|
-
}
|
|
645
|
-
|
|
645
|
+
};
|
|
646
|
+
|
|
647
|
+
await runSafeStep(() => this.jwproxy.command('', 'DELETE'));
|
|
648
|
+
await runSafeStep(() => this.proc.stop('SIGTERM', 20000));
|
|
649
|
+
this.log.prefix = (0, _utils.generateLogPrefix)(this);
|
|
650
|
+
|
|
651
|
+
if (emitStates) {
|
|
652
|
+
this.changeState(Chromedriver.STATE_STOPPED);
|
|
646
653
|
}
|
|
647
654
|
}
|
|
648
655
|
|
|
649
656
|
changeState(state) {
|
|
650
657
|
this.state = state;
|
|
651
|
-
log.debug(`Changed state to '${state}'`);
|
|
658
|
+
this.log.debug(`Changed state to '${state}'`);
|
|
652
659
|
this.emit(Chromedriver.EVENT_CHANGED, {
|
|
653
660
|
state
|
|
654
661
|
});
|
|
@@ -664,13 +671,13 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
664
671
|
|
|
665
672
|
async killAll() {
|
|
666
673
|
let cmd = _support.system.isWindows() ? `wmic process where "commandline like '%chromedriver.exe%--port=${this.proxyPort}%'" delete` : `pkill -15 -f "${this.chromedriver}.*--port=${this.proxyPort}"`;
|
|
667
|
-
log.debug(`Killing any old chromedrivers, running: ${cmd}`);
|
|
674
|
+
this.log.debug(`Killing any old chromedrivers, running: ${cmd}`);
|
|
668
675
|
|
|
669
676
|
try {
|
|
670
677
|
await _bluebird.default.promisify(_child_process.default.exec)(cmd);
|
|
671
|
-
log.debug('Successfully cleaned up old chromedrivers');
|
|
678
|
+
this.log.debug('Successfully cleaned up old chromedrivers');
|
|
672
679
|
} catch (err) {
|
|
673
|
-
log.warn('No old chromedrivers seem to exist');
|
|
680
|
+
this.log.warn('No old chromedrivers seem to exist');
|
|
674
681
|
}
|
|
675
682
|
|
|
676
683
|
if (this.adb) {
|
|
@@ -678,9 +685,9 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
678
685
|
const udid = udidIndex > -1 ? this.adb.executable.defaultArgs[udidIndex + 1] : null;
|
|
679
686
|
|
|
680
687
|
if (udid) {
|
|
681
|
-
log.debug(`Cleaning this device's adb forwarded port socket connections: ${udid}`);
|
|
688
|
+
this.log.debug(`Cleaning this device's adb forwarded port socket connections: ${udid}`);
|
|
682
689
|
} else {
|
|
683
|
-
log.debug(`Cleaning any old adb forwarded port socket connections`);
|
|
690
|
+
this.log.debug(`Cleaning any old adb forwarded port socket connections`);
|
|
684
691
|
}
|
|
685
692
|
|
|
686
693
|
try {
|
|
@@ -696,7 +703,7 @@ class Chromedriver extends _events.default.EventEmitter {
|
|
|
696
703
|
}
|
|
697
704
|
}
|
|
698
705
|
} catch (err) {
|
|
699
|
-
log.warn(`Unable to clean forwarded ports. Error: '${err.message}'. Continuing.`);
|
|
706
|
+
this.log.warn(`Unable to clean forwarded ports. Error: '${err.message}'. Continuing.`);
|
|
700
707
|
}
|
|
701
708
|
}
|
|
702
709
|
}
|
|
@@ -724,4 +731,4 @@ var _default = Chromedriver;
|
|
|
724
731
|
exports.default = _default;require('source-map-support').install();
|
|
725
732
|
|
|
726
733
|
|
|
727
|
-
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/chromedriver.js"],"names":["log","logger","getLogger","NEW_CD_VERSION_FORMAT_MAJOR_VERSION","DEFAULT_HOST","MIN_CD_VERSION_WITH_W3C_SUPPORT","DEFAULT_PORT","CHROME_BUNDLE_ID","WEBVIEW_SHELL_BUNDLE_ID","WEBVIEW_BUNDLE_IDS","CHROMEDRIVER_TUTORIAL","VERSION_PATTERN","CD_VERSION_TIMEOUT","Chromedriver","events","EventEmitter","constructor","args","host","port","useSystemExecutable","executable","executableDir","bundleId","mappingPath","cmdArgs","adb","verbose","logPath","disableBuildCheck","details","isAutodownloadEnabled","proxyHost","proxyPort","proc","chromedriver","executableVerified","state","STATE_STOPPED","jwproxy","JWProxy","server","storageClient","ChromedriverStorageClient","chromedriverDir","capabilities","desiredProtocol","PROTOCOLS","MJSONWP","getDriversMapping","mapping","_","cloneDeep","CHROMEDRIVER_CHROME_MAPPING","debug","fs","exists","warn","info","JSON","parse","readFile","err","message","cdVersion","chromeVersion","toPairs","coercedVersion","semver","coerce","version","getChromedrivers","executables","glob","util","pluralize","length","cds","mapChromedriver","logError","stdout","stderr","errMsg","path","basename","timeout","includes","match","exec","minChromeVersion","major","minor","filter","cd","sort","a","b","isEmpty","getChromeVersion","Browser","versionMatch","apiLevel","getApiLevel","updateDriversMapping","newMapping","shouldUpdateStaticMapping","writeFile","stringify","e","Object","assign","getCompatibleChromedriver","values","didStorageSync","syncChromedrivers","retrievedMapping","retrieveMapping","driverKeys","syncDrivers","minBrowserVersion","synchronizedDriversMapping","reduce","acc","x","missingVersions","coercedVer","size","errorAndThrow","matchingDrivers","minChromeVersionS","gte","CD_CDN","stack","autodownloadSuggestion","Error","binPath","initChromedriverPath","syncProtocol","W3C","chromeOptions","w3c","start","caps","emitStartingState","loggingPrefs","browser","changeState","STATE_STARTING","adbPort","push","isArray","startDetector","startsWith","processIsAlive","webviewVersion","killAll","SubProcess","on","out","line","trim","split","error","code","signal","STATE_STOPPING","STATE_RESTARTING","msg","join","waitForOnline","startSession","emit","EVENT_ERROR","stop","versionsSupportedByDriver","sessionId","STATE_ONLINE","restart","chromedriverStopped","getStatus","command","sessionCaps","alwaysMatch","desiredCapabilities","emitStates","EVENT_CHANGED","sendCommand","url","method","body","proxyReq","req","res","proxyReqRes","cmd","system","isWindows","B","promisify","cp","udidIndex","defaultArgs","findIndex","item","udid","conn","getForwardList","params","removePortForward","replace","hasWorkingWebview"],"mappings":"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","sourcesContent":["// transpile:main\n\nimport events from 'events';\nimport { JWProxy, PROTOCOLS } from '@appium/base-driver';\nimport cp from 'child_process';\nimport { system, fs, logger, util } from '@appium/support';\nimport { retryInterval, asyncmap } from 'asyncbox';\nimport { SubProcess, exec } from 'teen_process';\nimport B from 'bluebird';\nimport {\n  getChromeVersion, getChromedriverDir, CHROMEDRIVER_CHROME_MAPPING,\n  getChromedriverBinaryPath, CD_CDN,\n} from './utils';\nimport semver from 'semver';\nimport _ from 'lodash';\nimport path from 'path';\nimport compareVersions from 'compare-versions';\nimport ChromedriverStorageClient from './storage-client';\nimport { toW3cCapNames, getCapValue } from './protocol-helpers';\n\n\nconst log = logger.getLogger('Chromedriver');\n\nconst NEW_CD_VERSION_FORMAT_MAJOR_VERSION = 73;\nconst DEFAULT_HOST = '127.0.0.1';\nconst MIN_CD_VERSION_WITH_W3C_SUPPORT = 75;\nconst DEFAULT_PORT = 9515;\nconst CHROME_BUNDLE_ID = 'com.android.chrome';\nconst WEBVIEW_SHELL_BUNDLE_ID = 'org.chromium.webview_shell';\nconst WEBVIEW_BUNDLE_IDS = [\n  'com.google.android.webview',\n  'com.android.webview',\n];\nconst CHROMEDRIVER_TUTORIAL = 'https://github.com/appium/appium/blob/master/docs/en/writing-running-appium/web/chromedriver.md';\nconst VERSION_PATTERN = /([\\d.]+)/;\n\nconst CD_VERSION_TIMEOUT = 5000;\n\nclass Chromedriver extends events.EventEmitter {\n  constructor (args = {}) {\n    super();\n\n    const {\n      host = DEFAULT_HOST,\n      port = DEFAULT_PORT,\n      useSystemExecutable = false,\n      executable,\n      executableDir = getChromedriverDir(),\n      bundleId,\n      mappingPath,\n      cmdArgs,\n      adb,\n      verbose,\n      logPath,\n      disableBuildCheck,\n      details,\n      isAutodownloadEnabled = false,\n    } = args;\n\n    this.proxyHost = host;\n    this.proxyPort = port;\n    this.adb = adb;\n    this.cmdArgs = cmdArgs;\n    this.proc = null;\n    this.useSystemExecutable = useSystemExecutable;\n    this.chromedriver = executable;\n    this.executableDir = executableDir;\n    this.mappingPath = mappingPath;\n    this.bundleId = bundleId;\n    this.executableVerified = false;\n    this.state = Chromedriver.STATE_STOPPED;\n    this.jwproxy = new JWProxy({\n      server: this.proxyHost,\n      port: this.proxyPort\n    });\n    this.verbose = verbose;\n    this.logPath = logPath;\n    this.disableBuildCheck = !!disableBuildCheck;\n    this.storageClient = isAutodownloadEnabled\n      ? new ChromedriverStorageClient({ chromedriverDir: this.executableDir })\n      : null;\n    this.details = details;\n    this.capabilities = {};\n    this.desiredProtocol = PROTOCOLS.MJSONWP;\n  }\n\n  async getDriversMapping () {\n    let mapping = _.cloneDeep(CHROMEDRIVER_CHROME_MAPPING);\n    if (this.mappingPath) {\n      log.debug(`Attempting to use Chromedriver->Chrome mapping from '${this.mappingPath}'`);\n      if (!await fs.exists(this.mappingPath)) {\n        log.warn(`No file found at '${this.mappingPath}'`);\n        log.info('Defaulting to the static Chromedriver->Chrome mapping');\n      } else {\n        try {\n          mapping = JSON.parse(await fs.readFile(this.mappingPath, 'utf8'));\n        } catch (err) {\n          log.warn(`Error parsing mapping from '${this.mappingPath}': ${err.message}`);\n          log.info('Defaulting to the static Chromedriver->Chrome mapping');\n        }\n      }\n    } else {\n      log.debug('Using the static Chromedriver->Chrome mapping');\n    }\n\n    // make sure that the values for minimum chrome version are semver compliant\n    for (const [cdVersion, chromeVersion] of _.toPairs(mapping)) {\n      const coercedVersion = semver.coerce(chromeVersion);\n      if (coercedVersion) {\n        mapping[cdVersion] = coercedVersion.version;\n      } else {\n        log.info(`'${chromeVersion}' is not a valid version number. Skipping it`);\n      }\n    }\n    return mapping;\n  }\n\n  async getChromedrivers (mapping) {\n    // go through the versions available\n    const executables = await fs.glob(`${this.executableDir}/*`);\n    log.debug(`Found ${util.pluralize('executable', executables.length, true)} ` +\n      `in '${this.executableDir}'`);\n    const cds = (await asyncmap(executables, async function mapChromedriver (executable) {\n      const logError = ({message, stdout = null, stderr = null}) => {\n        let errMsg = `Cannot retrieve version number from '${path.basename(executable)}' Chromedriver binary. ` +\n          `Make sure it returns a valid version string in response to '--version' command line argument. ${message}`;\n        if (stdout) {\n          errMsg += `\\nStdout: ${stdout}`;\n        }\n        if (stderr) {\n          errMsg += `\\nStderr: ${stderr}`;\n        }\n        log.warn(errMsg);\n        return null;\n      };\n\n      let stdout;\n      let stderr;\n      try {\n        ({stdout, stderr} = await exec(executable, ['--version'], {\n          timeout: CD_VERSION_TIMEOUT,\n        }));\n      } catch (err) {\n        if (!(err.message || '').includes('timed out') && !(err.stdout || '').includes('Starting ChromeDriver')) {\n          return logError(err);\n        }\n\n        // if this has timed out, it has actually started Chromedriver,\n        // in which case there will also be the version string in the output\n        stdout = err.stdout;\n      }\n\n      const match = /ChromeDriver\\s+\\(?v?([\\d.]+)\\)?/i.exec(stdout); // https://regex101.com/r/zpj5wA/1\n      if (!match) {\n        return logError({message: 'Cannot parse the version string', stdout, stderr});\n      }\n      let version = match[1];\n      let minChromeVersion = mapping[version];\n      const coercedVersion = semver.coerce(version);\n      if (coercedVersion) {\n        // before 2019-03-06 versions were of the form major.minor\n        if (coercedVersion.major < NEW_CD_VERSION_FORMAT_MAJOR_VERSION) {\n          version = `${coercedVersion.major}.${coercedVersion.minor}`;\n          minChromeVersion = mapping[version];\n        }\n        if (!minChromeVersion && coercedVersion.major >= NEW_CD_VERSION_FORMAT_MAJOR_VERSION) {\n          // Assume the major Chrome version is the same as the corresponding driver major version\n          minChromeVersion = `${coercedVersion.major}`;\n        }\n      }\n      return {\n        executable,\n        version,\n        minChromeVersion,\n      };\n    }))\n      .filter((cd) => !!cd)\n      .sort((a, b) => compareVersions(b.version, a.version));\n    if (_.isEmpty(cds)) {\n      log.info(`No Chromedrivers were found in '${this.executableDir}'`);\n      return cds;\n    }\n    log.debug(`The following Chromedriver executables were found:`);\n    for (const cd of cds) {\n      log.debug(`    '${cd.executable}' (version '${cd.version}', minimum Chrome version '${cd.minChromeVersion ? cd.minChromeVersion : 'Unknown'}')`);\n    }\n    return cds;\n  }\n\n  async getChromeVersion () {\n    // Try to retrieve the version from `details` property if it is set\n    // The `info` item must contain the output of /json/version CDP command\n    // where `Browser` field looks like `Chrome/72.0.3601.0``\n    if (this.details?.info) {\n      log.debug(`Browser version in the supplied details: ${this.details?.info?.Browser}`);\n    }\n    const versionMatch = VERSION_PATTERN.exec(this.details?.info?.Browser);\n    if (versionMatch) {\n      const coercedVersion = semver.coerce(versionMatch[1]);\n      if (coercedVersion) {\n        return coercedVersion;\n      }\n    }\n\n    let chromeVersion;\n\n    // in case of WebView Browser Tester, simply try to find the underlying webview\n    if (this.bundleId === WEBVIEW_SHELL_BUNDLE_ID) {\n      for (const bundleId of WEBVIEW_BUNDLE_IDS) {\n        chromeVersion = await getChromeVersion(this.adb, bundleId);\n        if (chromeVersion) {\n          this.bundleId = bundleId;\n          return semver.coerce(chromeVersion);\n        }\n      }\n      return null;\n    }\n\n    // on Android 7-9 webviews are backed by the main Chrome, not the system webview\n    if (this.adb) {\n      const apiLevel = await this.adb.getApiLevel();\n      if (apiLevel >= 24 && apiLevel <= 28 &&\n          [WEBVIEW_SHELL_BUNDLE_ID, ...WEBVIEW_BUNDLE_IDS].includes(this.bundleId)) {\n        this.bundleId = CHROME_BUNDLE_ID;\n      }\n    }\n\n    // try out webviews when no bundle id is sent in\n    if (!this.bundleId) {\n      // default to the generic Chrome bundle\n      this.bundleId = CHROME_BUNDLE_ID;\n\n      // we have a webview of some sort, so try to find the bundle version\n      for (const bundleId of WEBVIEW_BUNDLE_IDS) {\n        chromeVersion = await getChromeVersion(this.adb, bundleId);\n        if (chromeVersion) {\n          this.bundleId = bundleId;\n          break;\n        }\n      }\n    }\n\n    // if we do not have a chrome version, it must not be a webview\n    if (!chromeVersion) {\n      chromeVersion = await getChromeVersion(this.adb, this.bundleId);\n    }\n\n    // make sure it is semver, so later checks won't fail\n    return chromeVersion ? semver.coerce(chromeVersion) : null;\n  }\n\n  async updateDriversMapping (newMapping) {\n    let shouldUpdateStaticMapping = true;\n    if (await fs.exists(this.mappingPath)) {\n      try {\n        await fs.writeFile(this.mappingPath, JSON.stringify(newMapping, null, 2), 'utf8');\n        shouldUpdateStaticMapping = false;\n      } catch (e) {\n        log.warn(`Cannot store the updated chromedrivers mapping into '${this.mappingPath}'. ` +\n          `This may reduce the performance of further executions. Original error: ${e.message}`);\n      }\n    }\n    if (shouldUpdateStaticMapping) {\n      Object.assign(CHROMEDRIVER_CHROME_MAPPING, newMapping);\n    }\n  }\n\n  async getCompatibleChromedriver () {\n    if (!this.adb) {\n      return await getChromedriverBinaryPath();\n    }\n\n    const mapping = await this.getDriversMapping();\n    if (!_.isEmpty(mapping)) {\n      log.debug(`The most recent known Chrome version: ${_.values(mapping)[0]}`);\n    }\n\n    let didStorageSync = false;\n    const syncChromedrivers = async (chromeVersion) => {\n      didStorageSync = true;\n      const retrievedMapping = await this.storageClient.retrieveMapping();\n      log.debug('Got chromedrivers mapping from the storage: ' +\n        JSON.stringify(retrievedMapping, null, 2));\n      const driverKeys = await this.storageClient.syncDrivers({\n        minBrowserVersion: chromeVersion.major,\n      });\n      if (_.isEmpty(driverKeys)) {\n        return false;\n      }\n      const synchronizedDriversMapping = driverKeys.reduce((acc, x) => {\n        const {version, minBrowserVersion} = retrievedMapping[x];\n        acc[version] = minBrowserVersion;\n        return acc;\n      }, {});\n      Object.assign(mapping, synchronizedDriversMapping);\n      await this.updateDriversMapping(mapping);\n      return true;\n    };\n\n    do {\n      const cds = await this.getChromedrivers(mapping);\n\n      const missingVersions = {};\n      for (const {version, minChromeVersion} of cds) {\n        if (!minChromeVersion || mapping[version]) {\n          continue;\n        }\n        const coercedVer = semver.coerce(version);\n        if (!coercedVer || coercedVer.major < NEW_CD_VERSION_FORMAT_MAJOR_VERSION) {\n          continue;\n        }\n\n        missingVersions[version] = minChromeVersion;\n      }\n      if (!_.isEmpty(missingVersions)) {\n        log.info(`Found ${util.pluralize('Chromedriver', _.size(missingVersions), true)}, ` +\n          `which ${_.size(missingVersions) === 1 ? 'is' : 'are'} missing in the list of known versions: ` +\n          JSON.stringify(missingVersions));\n        await this.updateDriversMapping(Object.assign(mapping, missingVersions));\n      }\n\n      if (this.disableBuildCheck) {\n        if (_.isEmpty(cds)) {\n          log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` +\n            `'chromedriverDisableBuildCheck' capability is set to 'true'`);\n        }\n        const {version, executable} = cds[0];\n        log.warn(`Chrome build check disabled. Using most recent Chromedriver version (${version}, at '${executable}')`);\n        log.warn(`If this is wrong, set 'chromedriverDisableBuildCheck' capability to 'false'`);\n        return executable;\n      }\n\n      const chromeVersion = await this.getChromeVersion();\n      if (!chromeVersion) {\n        // unable to get the chrome version\n        if (_.isEmpty(cds)) {\n          log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` +\n            `the current Chrome version cannot be determined`);\n        }\n        const {version, executable} = cds[0];\n        log.warn(`Unable to discover Chrome version. Using Chromedriver ${version} at '${executable}'`);\n        return executable;\n      }\n      log.debug(`Found Chrome bundle '${this.bundleId}' version '${chromeVersion}'`);\n\n      const matchingDrivers = cds.filter(({minChromeVersion}) => {\n        const minChromeVersionS = minChromeVersion && semver.coerce(minChromeVersion);\n        if (!minChromeVersionS) {\n          return false;\n        }\n\n        return chromeVersion.major > NEW_CD_VERSION_FORMAT_MAJOR_VERSION\n          ? minChromeVersionS.major === chromeVersion.major\n          : semver.gte(chromeVersion, minChromeVersionS);\n      });\n      if (_.isEmpty(matchingDrivers)) {\n        if (this.storageClient && !didStorageSync) {\n          try {\n            if (await syncChromedrivers(chromeVersion)) {\n              continue;\n            }\n          } catch (e) {\n            log.warn(`Cannot synchronize local chromedrivers with the remote storage at ${CD_CDN}: ` +\n              e.message);\n            log.debug(e.stack);\n          }\n        }\n        const autodownloadSuggestion =\n          'You could also try to enable automated chromedrivers download server feature';\n        throw new Error(`No Chromedriver found that can automate Chrome '${chromeVersion}'. ` +\n          (!this.storageClient ? `${autodownloadSuggestion}. ` : '') +\n          `See ${CHROMEDRIVER_TUTORIAL} for more details`);\n      }\n\n      const binPath = matchingDrivers[0].executable;\n      log.debug(`Found ${util.pluralize('executable', matchingDrivers.length, true)} ` +\n        `capable of automating Chrome '${chromeVersion}'.\\nChoosing the most recent, '${binPath}'.`);\n      log.debug('If a specific version is required, specify it with the `chromedriverExecutable`' +\n        'desired capability.');\n      return binPath;\n    // eslint-disable-next-line no-constant-condition\n    } while (true);\n  }\n\n  async initChromedriverPath () {\n    if (this.executableVerified) return; //eslint-disable-line curly\n\n    // the executable might be set (if passed in)\n    // or we might want to use the basic one installed with this driver\n    // or we want to figure out the best one\n    if (!this.chromedriver) {\n      this.chromedriver = this.useSystemExecutable\n        ? await getChromedriverBinaryPath()\n        : await this.getCompatibleChromedriver();\n    }\n\n    if (!await fs.exists(this.chromedriver)) {\n      throw new Error(`Trying to use a chromedriver binary at the path ` +\n                      `${this.chromedriver}, but it doesn't exist!`);\n    }\n    this.executableVerified = true;\n    log.info(`Set chromedriver binary as: ${this.chromedriver}`);\n  }\n\n  syncProtocol (cdVersion = null) {\n    const coercedVersion = semver.coerce(cdVersion);\n    if (!coercedVersion || coercedVersion.major < MIN_CD_VERSION_WITH_W3C_SUPPORT) {\n      log.debug(`Chromedriver v. ${cdVersion} does not fully support ${PROTOCOLS.W3C} protocol. ` +\n        `Defaulting to ${PROTOCOLS.MJSONWP}`);\n      return;\n    }\n    const chromeOptions = getCapValue(this.capabilities, 'chromeOptions', {});\n    if (chromeOptions.w3c === false) {\n      log.info(`Chromedriver v. ${cdVersion} supports ${PROTOCOLS.W3C} protocol, ` +\n        `but ${PROTOCOLS.MJSONWP} one has been explicitly requested`);\n      return;\n    }\n    this.desiredProtocol = PROTOCOLS.W3C;\n    // given caps might not be properly prefixed\n    // so we try to fix them in order to properly init\n    // the new W3C session\n    this.capabilities = toW3cCapNames(this.capabilities);\n  }\n\n  async start (caps, emitStartingState = true) {\n    this.capabilities = _.cloneDeep(caps);\n\n    // set the logging preferences to ALL the console logs\n    this.capabilities.loggingPrefs = _.cloneDeep(getCapValue(caps, 'loggingPrefs', {}));\n    if (_.isEmpty(this.capabilities.loggingPrefs.browser)) {\n      this.capabilities.loggingPrefs.browser = 'ALL';\n    }\n\n    if (emitStartingState) {\n      this.changeState(Chromedriver.STATE_STARTING);\n    }\n\n    const args = [`--port=${this.proxyPort}`];\n    if (this.adb && this.adb.adbPort) {\n      args.push(`--adb-port=${this.adb.adbPort}`);\n    }\n    if (_.isArray(this.cmdArgs)) {\n      args.push(...this.cmdArgs);\n    }\n    if (this.logPath) {\n      args.push(`--log-path=${this.logPath}`);\n    }\n    if (this.disableBuildCheck) {\n      args.push('--disable-build-check');\n    }\n    args.push('--verbose');\n    // what are the process stdout/stderr conditions wherein we know that\n    // the process has started to our satisfaction?\n    const startDetector = (stdout) => stdout.startsWith('Starting ');\n\n    let processIsAlive = false;\n    let webviewVersion;\n    try {\n      await this.initChromedriverPath();\n      await this.killAll();\n\n      // set up our subprocess object\n      this.proc = new SubProcess(this.chromedriver, args);\n      processIsAlive = true;\n\n      // handle log output\n      this.proc.on('output', (stdout, stderr) => {\n        // if the cd output is not printed, find the chrome version and print\n        // will get a response like\n        //   DevTools response: {\n        //      \"Android-Package\": \"io.appium.sampleapp\",\n        //      \"Browser\": \"Chrome/55.0.2883.91\",\n        //      \"Protocol-Version\": \"1.2\",\n        //      \"User-Agent\": \"...\",\n        //      \"WebKit-Version\": \"537.36\"\n        //   }\n        const out = stdout + stderr;\n        let match = /\"Browser\": \"(.*)\"/.exec(out);\n        if (match) {\n          webviewVersion = match[1];\n          log.debug(`Webview version: '${webviewVersion}'`);\n        }\n\n        // also print chromedriver version to logs\n        // will output something like\n        //  Starting ChromeDriver 2.33.506106 (8a06c39c4582fbfbab6966dbb1c38a9173bfb1a2) on port 9515\n        match = /Starting ChromeDriver ([.\\d]+)/.exec(out);\n        if (match) {\n          log.debug(`Chromedriver version: '${match[1]}'`);\n          this.syncProtocol(match[1]);\n        }\n\n        // give the output if it is requested\n        if (this.verbose) {\n          for (let line of (stdout || '').trim().split('\\n')) {\n            if (!line.trim().length) continue; // eslint-disable-line curly\n            log.debug(`[STDOUT] ${line}`);\n          }\n          for (let line of (stderr || '').trim().split('\\n')) {\n            if (!line.trim().length) continue; // eslint-disable-line curly\n            log.error(`[STDERR] ${line}`);\n          }\n        }\n      });\n\n      // handle out-of-bound exit by simply emitting a stopped state\n      this.proc.on('exit', (code, signal) => {\n        processIsAlive = false;\n        if (this.state !== Chromedriver.STATE_STOPPED &&\n            this.state !== Chromedriver.STATE_STOPPING &&\n            this.state !== Chromedriver.STATE_RESTARTING) {\n          let msg = `Chromedriver exited unexpectedly with code ${code}, ` +\n                    `signal ${signal}`;\n          log.error(msg);\n          this.changeState(Chromedriver.STATE_STOPPED);\n        }\n      });\n      log.info(`Spawning chromedriver with: ${this.chromedriver} ` +\n               `${args.join(' ')}`);\n      // start subproc and wait for startDetector\n      await this.proc.start(startDetector);\n      await this.waitForOnline();\n      await this.startSession();\n    } catch (e) {\n      log.debug(e);\n      this.emit(Chromedriver.EVENT_ERROR, e);\n      // just because we had an error doesn't mean the chromedriver process\n      // finished; we should clean up if necessary\n      if (processIsAlive) {\n        await this.proc.stop();\n      }\n\n      let message = '';\n      // often the user's Chrome version is not supported by the version of Chromedriver\n      if (e.message.includes('Chrome version must be')) {\n        message += 'Unable to automate Chrome version because it is not supported by this version of Chromedriver.\\n';\n        if (webviewVersion) {\n          message += `Chrome version on the device: ${webviewVersion}\\n`;\n        }\n        const versionsSupportedByDriver = /Chrome version must be (.+)/.exec(e.message)?.[1] || '';\n        if (versionsSupportedByDriver) {\n          message += `Chromedriver supports Chrome version(s): ${versionsSupportedByDriver}\\n`;\n        }\n        message += `Visit '${CHROMEDRIVER_TUTORIAL}' to troubleshoot the problem.\\n`;\n      }\n\n      message += e.message;\n      log.errorAndThrow(message);\n    }\n  }\n\n  sessionId () {\n    if (this.state !== Chromedriver.STATE_ONLINE) {\n      return null;\n    }\n\n    return this.jwproxy.sessionId;\n  }\n\n  async restart () {\n    log.info('Restarting chromedriver');\n    if (this.state !== Chromedriver.STATE_ONLINE) {\n      throw new Error(\"Can't restart when we're not online\");\n    }\n    this.changeState(Chromedriver.STATE_RESTARTING);\n    await this.stop(false);\n    await this.start(this.capabilities, false);\n  }\n\n  async waitForOnline () {\n    // we need to make sure that CD hasn't crashed\n    let chromedriverStopped = false;\n    await retryInterval(20, 200, async () => {\n      if (this.state === Chromedriver.STATE_STOPPED) {\n        // we are either stopped or stopping, so something went wrong\n        chromedriverStopped = true;\n        return;\n      }\n      await this.getStatus();\n    });\n    if (chromedriverStopped) {\n      throw new Error('ChromeDriver crashed during startup.');\n    }\n  }\n\n  async getStatus () {\n    return await this.jwproxy.command('/status', 'GET');\n  }\n\n  async startSession () {\n    const sessionCaps = this.desiredProtocol === PROTOCOLS.W3C\n      ? {capabilities: {alwaysMatch: this.capabilities}}\n      : {desiredCapabilities: this.capabilities};\n    log.info(`Starting ${this.desiredProtocol} Chromedriver session with capabilities: ` +\n      JSON.stringify(sessionCaps, null, 2));\n    await this.jwproxy.command('/session', 'POST', sessionCaps);\n    this.changeState(Chromedriver.STATE_ONLINE);\n  }\n\n  async stop (emitStates = true) {\n    if (emitStates) {\n      this.changeState(Chromedriver.STATE_STOPPING);\n    }\n    try {\n      await this.jwproxy.command('', 'DELETE');\n      await this.proc.stop('SIGTERM', 20000);\n      if (emitStates) {\n        this.changeState(Chromedriver.STATE_STOPPED);\n      }\n    } catch (e) {\n      log.error(e);\n    }\n  }\n\n  changeState (state) {\n    this.state = state;\n    log.debug(`Changed state to '${state}'`);\n    this.emit(Chromedriver.EVENT_CHANGED, {state});\n  }\n\n  async sendCommand (url, method, body) {\n    return await this.jwproxy.command(url, method, body);\n  }\n\n  async proxyReq (req, res) {\n    return await this.jwproxy.proxyReqRes(req, res);\n  }\n\n  async killAll () {\n    let cmd = system.isWindows()\n      ? `wmic process where \"commandline like '%chromedriver.exe%--port=${this.proxyPort}%'\" delete`\n      : `pkill -15 -f \"${this.chromedriver}.*--port=${this.proxyPort}\"`;\n    log.debug(`Killing any old chromedrivers, running: ${cmd}`);\n    try {\n      await (B.promisify(cp.exec))(cmd);\n      log.debug('Successfully cleaned up old chromedrivers');\n    } catch (err) {\n      log.warn('No old chromedrivers seem to exist');\n    }\n\n    if (this.adb) {\n      const udidIndex = this.adb.executable.defaultArgs.findIndex((item) => item === '-s');\n      const udid = udidIndex > -1 ? this.adb.executable.defaultArgs[udidIndex + 1] : null;\n\n      if (udid) {\n        log.debug(`Cleaning this device's adb forwarded port socket connections: ${udid}`);\n      } else {\n        log.debug(`Cleaning any old adb forwarded port socket connections`);\n      }\n\n      try {\n        for (let conn of await this.adb.getForwardList()) {\n          // chromedriver will ask ADB to forward a port like \"deviceId tcp:port localabstract:webview_devtools_remote_port\"\n          if (!(conn.includes('webview_devtools') && (!udid || conn.includes(udid)))) {\n            continue;\n          }\n\n          let params = conn.split(/\\s+/);\n          if (params.length > 1) {\n            await this.adb.removePortForward(params[1].replace(/[\\D]*/, ''));\n          }\n        }\n      } catch (err) {\n        log.warn(`Unable to clean forwarded ports. Error: '${err.message}'. Continuing.`);\n      }\n    }\n  }\n\n  async hasWorkingWebview () {\n    // sometimes chromedriver stops automating webviews. this method runs a\n    // simple command to determine our state, and responds accordingly\n    try {\n      await this.jwproxy.command('/url', 'GET');\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n}\n\nChromedriver.EVENT_ERROR = 'chromedriver_error';\nChromedriver.EVENT_CHANGED = 'stateChanged';\nChromedriver.STATE_STOPPED = 'stopped';\nChromedriver.STATE_STARTING = 'starting';\nChromedriver.STATE_ONLINE = 'online';\nChromedriver.STATE_STOPPING = 'stopping';\nChromedriver.STATE_RESTARTING = 'restarting';\n\nexport { Chromedriver };\nexport default Chromedriver;\n"],"file":"lib/chromedriver.js","sourceRoot":"../.."}
|
|
734
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/chromedriver.js"],"names":["NEW_CD_VERSION_FORMAT_MAJOR_VERSION","DEFAULT_HOST","MIN_CD_VERSION_WITH_W3C_SUPPORT","DEFAULT_PORT","CHROME_BUNDLE_ID","WEBVIEW_SHELL_BUNDLE_ID","WEBVIEW_BUNDLE_IDS","CHROMEDRIVER_TUTORIAL","VERSION_PATTERN","CD_VERSION_TIMEOUT","Chromedriver","events","EventEmitter","constructor","args","host","port","useSystemExecutable","executable","executableDir","bundleId","mappingPath","cmdArgs","adb","verbose","logPath","disableBuildCheck","details","isAutodownloadEnabled","_log","logger","getLogger","proxyHost","proxyPort","proc","chromedriver","executableVerified","state","STATE_STOPPED","jwproxy","JWProxy","server","log","storageClient","ChromedriverStorageClient","chromedriverDir","capabilities","desiredProtocol","PROTOCOLS","MJSONWP","getDriversMapping","mapping","_","cloneDeep","CHROMEDRIVER_CHROME_MAPPING","debug","fs","exists","warn","info","JSON","parse","readFile","err","message","cdVersion","chromeVersion","toPairs","coercedVersion","semver","coerce","version","getChromedrivers","executables","glob","util","pluralize","length","cds","mapChromedriver","logError","stdout","stderr","errMsg","path","basename","timeout","includes","match","exec","minChromeVersion","major","minor","filter","cd","sort","a","b","isEmpty","getChromeVersion","Browser","versionMatch","apiLevel","getApiLevel","updateDriversMapping","newMapping","shouldUpdateStaticMapping","writeFile","stringify","e","Object","assign","getCompatibleChromedriver","values","didStorageSync","syncChromedrivers","retrievedMapping","retrieveMapping","driverKeys","syncDrivers","minBrowserVersion","synchronizedDriversMapping","reduce","acc","x","missingVersions","coercedVer","size","errorAndThrow","matchingDrivers","minChromeVersionS","gte","CD_CDN","stack","autodownloadSuggestion","Error","binPath","initChromedriverPath","syncProtocol","W3C","chromeOptions","w3c","start","caps","emitStartingState","loggingPrefs","browser","changeState","STATE_STARTING","adbPort","push","isArray","startDetector","startsWith","processIsAlive","webviewVersion","killAll","SubProcess","on","out","line","trim","split","error","code","signal","STATE_STOPPING","STATE_RESTARTING","msg","join","waitForOnline","startSession","emit","EVENT_ERROR","stop","versionsSupportedByDriver","sessionId","STATE_ONLINE","restart","chromedriverStopped","getStatus","command","sessionCaps","alwaysMatch","desiredCapabilities","prefix","emitStates","runSafeStep","f","EVENT_CHANGED","sendCommand","url","method","body","proxyReq","req","res","proxyReqRes","cmd","system","isWindows","B","promisify","cp","udidIndex","defaultArgs","findIndex","item","udid","conn","getForwardList","params","removePortForward","replace","hasWorkingWebview"],"mappings":"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","sourcesContent":["// transpile:main\n\nimport events from 'events';\nimport { JWProxy, PROTOCOLS } from '@appium/base-driver';\nimport cp from 'child_process';\nimport { system, fs, logger, util } from '@appium/support';\nimport { retryInterval, asyncmap } from 'asyncbox';\nimport { SubProcess, exec } from 'teen_process';\nimport B from 'bluebird';\nimport {\n  getChromeVersion, getChromedriverDir, CHROMEDRIVER_CHROME_MAPPING,\n  getChromedriverBinaryPath, CD_CDN, generateLogPrefix\n} from './utils';\nimport semver from 'semver';\nimport _ from 'lodash';\nimport path from 'path';\nimport compareVersions from 'compare-versions';\nimport ChromedriverStorageClient from './storage-client';\nimport { toW3cCapNames, getCapValue } from './protocol-helpers';\n\nconst NEW_CD_VERSION_FORMAT_MAJOR_VERSION = 73;\nconst DEFAULT_HOST = '127.0.0.1';\nconst MIN_CD_VERSION_WITH_W3C_SUPPORT = 75;\nconst DEFAULT_PORT = 9515;\nconst CHROME_BUNDLE_ID = 'com.android.chrome';\nconst WEBVIEW_SHELL_BUNDLE_ID = 'org.chromium.webview_shell';\nconst WEBVIEW_BUNDLE_IDS = [\n  'com.google.android.webview',\n  'com.android.webview',\n];\nconst CHROMEDRIVER_TUTORIAL = 'https://github.com/appium/appium/blob/master/docs/en/writing-running-appium/web/chromedriver.md';\nconst VERSION_PATTERN = /([\\d.]+)/;\n\nconst CD_VERSION_TIMEOUT = 5000;\n\nclass Chromedriver extends events.EventEmitter {\n  constructor (args = {}) {\n    super();\n\n    const {\n      host = DEFAULT_HOST,\n      port = DEFAULT_PORT,\n      useSystemExecutable = false,\n      executable,\n      executableDir = getChromedriverDir(),\n      bundleId,\n      mappingPath,\n      cmdArgs,\n      adb,\n      verbose,\n      logPath,\n      disableBuildCheck,\n      details,\n      isAutodownloadEnabled = false,\n    } = args;\n    this._log = logger.getLogger(generateLogPrefix(this));\n\n    this.proxyHost = host;\n    this.proxyPort = port;\n    this.adb = adb;\n    this.cmdArgs = cmdArgs;\n    this.proc = null;\n    this.useSystemExecutable = useSystemExecutable;\n    this.chromedriver = executable;\n    this.executableDir = executableDir;\n    this.mappingPath = mappingPath;\n    this.bundleId = bundleId;\n    this.executableVerified = false;\n    this.state = Chromedriver.STATE_STOPPED;\n    this.jwproxy = new JWProxy({\n      server: this.proxyHost,\n      port: this.proxyPort,\n      log: this._log,\n    });\n    this.verbose = verbose;\n    this.logPath = logPath;\n    this.disableBuildCheck = !!disableBuildCheck;\n    this.storageClient = isAutodownloadEnabled\n      ? new ChromedriverStorageClient({ chromedriverDir: this.executableDir })\n      : null;\n    this.details = details;\n    this.capabilities = {};\n    this.desiredProtocol = PROTOCOLS.MJSONWP;\n  }\n\n  get log () {\n    return this._log;\n  }\n\n  async getDriversMapping () {\n    let mapping = _.cloneDeep(CHROMEDRIVER_CHROME_MAPPING);\n    if (this.mappingPath) {\n      this.log.debug(`Attempting to use Chromedriver->Chrome mapping from '${this.mappingPath}'`);\n      if (!await fs.exists(this.mappingPath)) {\n        this.log.warn(`No file found at '${this.mappingPath}'`);\n        this.log.info('Defaulting to the static Chromedriver->Chrome mapping');\n      } else {\n        try {\n          mapping = JSON.parse(await fs.readFile(this.mappingPath, 'utf8'));\n        } catch (err) {\n          this.log.warn(`Error parsing mapping from '${this.mappingPath}': ${err.message}`);\n          this.log.info('Defaulting to the static Chromedriver->Chrome mapping');\n        }\n      }\n    } else {\n      this.log.debug('Using the static Chromedriver->Chrome mapping');\n    }\n\n    // make sure that the values for minimum chrome version are semver compliant\n    for (const [cdVersion, chromeVersion] of _.toPairs(mapping)) {\n      const coercedVersion = semver.coerce(chromeVersion);\n      if (coercedVersion) {\n        mapping[cdVersion] = coercedVersion.version;\n      } else {\n        this.log.info(`'${chromeVersion}' is not a valid version number. Skipping it`);\n      }\n    }\n    return mapping;\n  }\n\n  async getChromedrivers (mapping) {\n    // go through the versions available\n    const executables = await fs.glob(`${this.executableDir}/*`);\n    this.log.debug(`Found ${util.pluralize('executable', executables.length, true)} ` +\n      `in '${this.executableDir}'`);\n    const cds = (await asyncmap(executables, async function mapChromedriver (executable) {\n      const logError = ({message, stdout = null, stderr = null}) => {\n        let errMsg = `Cannot retrieve version number from '${path.basename(executable)}' Chromedriver binary. ` +\n          `Make sure it returns a valid version string in response to '--version' command line argument. ${message}`;\n        if (stdout) {\n          errMsg += `\\nStdout: ${stdout}`;\n        }\n        if (stderr) {\n          errMsg += `\\nStderr: ${stderr}`;\n        }\n        this.log.warn(errMsg);\n        return null;\n      };\n\n      let stdout;\n      let stderr;\n      try {\n        ({stdout, stderr} = await exec(executable, ['--version'], {\n          timeout: CD_VERSION_TIMEOUT,\n        }));\n      } catch (err) {\n        if (!(err.message || '').includes('timed out') && !(err.stdout || '').includes('Starting ChromeDriver')) {\n          return logError(err);\n        }\n\n        // if this has timed out, it has actually started Chromedriver,\n        // in which case there will also be the version string in the output\n        stdout = err.stdout;\n      }\n\n      const match = /ChromeDriver\\s+\\(?v?([\\d.]+)\\)?/i.exec(stdout); // https://regex101.com/r/zpj5wA/1\n      if (!match) {\n        return logError({message: 'Cannot parse the version string', stdout, stderr});\n      }\n      let version = match[1];\n      let minChromeVersion = mapping[version];\n      const coercedVersion = semver.coerce(version);\n      if (coercedVersion) {\n        // before 2019-03-06 versions were of the form major.minor\n        if (coercedVersion.major < NEW_CD_VERSION_FORMAT_MAJOR_VERSION) {\n          version = `${coercedVersion.major}.${coercedVersion.minor}`;\n          minChromeVersion = mapping[version];\n        }\n        if (!minChromeVersion && coercedVersion.major >= NEW_CD_VERSION_FORMAT_MAJOR_VERSION) {\n          // Assume the major Chrome version is the same as the corresponding driver major version\n          minChromeVersion = `${coercedVersion.major}`;\n        }\n      }\n      return {\n        executable,\n        version,\n        minChromeVersion,\n      };\n    }))\n      .filter((cd) => !!cd)\n      .sort((a, b) => compareVersions(b.version, a.version));\n    if (_.isEmpty(cds)) {\n      this.log.info(`No Chromedrivers were found in '${this.executableDir}'`);\n      return cds;\n    }\n    this.log.debug(`The following Chromedriver executables were found:`);\n    for (const cd of cds) {\n      this.log.debug(`    '${cd.executable}' (version '${cd.version}', minimum Chrome version '${cd.minChromeVersion ? cd.minChromeVersion : 'Unknown'}')`);\n    }\n    return cds;\n  }\n\n  async getChromeVersion () {\n    // Try to retrieve the version from `details` property if it is set\n    // The `info` item must contain the output of /json/version CDP command\n    // where `Browser` field looks like `Chrome/72.0.3601.0``\n    if (this.details?.info) {\n      this.log.debug(`Browser version in the supplied details: ${this.details?.info?.Browser}`);\n    }\n    const versionMatch = VERSION_PATTERN.exec(this.details?.info?.Browser);\n    if (versionMatch) {\n      const coercedVersion = semver.coerce(versionMatch[1]);\n      if (coercedVersion) {\n        return coercedVersion;\n      }\n    }\n\n    let chromeVersion;\n\n    // in case of WebView Browser Tester, simply try to find the underlying webview\n    if (this.bundleId === WEBVIEW_SHELL_BUNDLE_ID) {\n      for (const bundleId of WEBVIEW_BUNDLE_IDS) {\n        chromeVersion = await getChromeVersion(this.adb, bundleId);\n        if (chromeVersion) {\n          this.bundleId = bundleId;\n          return semver.coerce(chromeVersion);\n        }\n      }\n      return null;\n    }\n\n    // on Android 7-9 webviews are backed by the main Chrome, not the system webview\n    if (this.adb) {\n      const apiLevel = await this.adb.getApiLevel();\n      if (apiLevel >= 24 && apiLevel <= 28 &&\n          [WEBVIEW_SHELL_BUNDLE_ID, ...WEBVIEW_BUNDLE_IDS].includes(this.bundleId)) {\n        this.bundleId = CHROME_BUNDLE_ID;\n      }\n    }\n\n    // try out webviews when no bundle id is sent in\n    if (!this.bundleId) {\n      // default to the generic Chrome bundle\n      this.bundleId = CHROME_BUNDLE_ID;\n\n      // we have a webview of some sort, so try to find the bundle version\n      for (const bundleId of WEBVIEW_BUNDLE_IDS) {\n        chromeVersion = await getChromeVersion(this.adb, bundleId);\n        if (chromeVersion) {\n          this.bundleId = bundleId;\n          break;\n        }\n      }\n    }\n\n    // if we do not have a chrome version, it must not be a webview\n    if (!chromeVersion) {\n      chromeVersion = await getChromeVersion(this.adb, this.bundleId);\n    }\n\n    // make sure it is semver, so later checks won't fail\n    return chromeVersion ? semver.coerce(chromeVersion) : null;\n  }\n\n  async updateDriversMapping (newMapping) {\n    let shouldUpdateStaticMapping = true;\n    if (await fs.exists(this.mappingPath)) {\n      try {\n        await fs.writeFile(this.mappingPath, JSON.stringify(newMapping, null, 2), 'utf8');\n        shouldUpdateStaticMapping = false;\n      } catch (e) {\n        this.log.warn(`Cannot store the updated chromedrivers mapping into '${this.mappingPath}'. ` +\n          `This may reduce the performance of further executions. Original error: ${e.message}`);\n      }\n    }\n    if (shouldUpdateStaticMapping) {\n      Object.assign(CHROMEDRIVER_CHROME_MAPPING, newMapping);\n    }\n  }\n\n  async getCompatibleChromedriver () {\n    if (!this.adb) {\n      return await getChromedriverBinaryPath();\n    }\n\n    const mapping = await this.getDriversMapping();\n    if (!_.isEmpty(mapping)) {\n      this.log.debug(`The most recent known Chrome version: ${_.values(mapping)[0]}`);\n    }\n\n    let didStorageSync = false;\n    const syncChromedrivers = async (chromeVersion) => {\n      didStorageSync = true;\n      const retrievedMapping = await this.storageClient.retrieveMapping();\n      this.log.debug('Got chromedrivers mapping from the storage: ' +\n        JSON.stringify(retrievedMapping, null, 2));\n      const driverKeys = await this.storageClient.syncDrivers({\n        minBrowserVersion: chromeVersion.major,\n      });\n      if (_.isEmpty(driverKeys)) {\n        return false;\n      }\n      const synchronizedDriversMapping = driverKeys.reduce((acc, x) => {\n        const {version, minBrowserVersion} = retrievedMapping[x];\n        acc[version] = minBrowserVersion;\n        return acc;\n      }, {});\n      Object.assign(mapping, synchronizedDriversMapping);\n      await this.updateDriversMapping(mapping);\n      return true;\n    };\n\n    do {\n      const cds = await this.getChromedrivers(mapping);\n\n      const missingVersions = {};\n      for (const {version, minChromeVersion} of cds) {\n        if (!minChromeVersion || mapping[version]) {\n          continue;\n        }\n        const coercedVer = semver.coerce(version);\n        if (!coercedVer || coercedVer.major < NEW_CD_VERSION_FORMAT_MAJOR_VERSION) {\n          continue;\n        }\n\n        missingVersions[version] = minChromeVersion;\n      }\n      if (!_.isEmpty(missingVersions)) {\n        this.log.info(`Found ${util.pluralize('Chromedriver', _.size(missingVersions), true)}, ` +\n          `which ${_.size(missingVersions) === 1 ? 'is' : 'are'} missing in the list of known versions: ` +\n          JSON.stringify(missingVersions));\n        await this.updateDriversMapping(Object.assign(mapping, missingVersions));\n      }\n\n      if (this.disableBuildCheck) {\n        if (_.isEmpty(cds)) {\n          this.log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` +\n            `'chromedriverDisableBuildCheck' capability is set to 'true'`);\n        }\n        const {version, executable} = cds[0];\n        this.log.warn(`Chrome build check disabled. Using most recent Chromedriver version (${version}, at '${executable}')`);\n        this.log.warn(`If this is wrong, set 'chromedriverDisableBuildCheck' capability to 'false'`);\n        return executable;\n      }\n\n      const chromeVersion = await this.getChromeVersion();\n      if (!chromeVersion) {\n        // unable to get the chrome version\n        if (_.isEmpty(cds)) {\n          this.log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` +\n            `the current Chrome version cannot be determined`);\n        }\n        const {version, executable} = cds[0];\n        this.log.warn(`Unable to discover Chrome version. Using Chromedriver ${version} at '${executable}'`);\n        return executable;\n      }\n      this.log.debug(`Found Chrome bundle '${this.bundleId}' version '${chromeVersion}'`);\n\n      const matchingDrivers = cds.filter(({minChromeVersion}) => {\n        const minChromeVersionS = minChromeVersion && semver.coerce(minChromeVersion);\n        if (!minChromeVersionS) {\n          return false;\n        }\n\n        return chromeVersion.major > NEW_CD_VERSION_FORMAT_MAJOR_VERSION\n          ? minChromeVersionS.major === chromeVersion.major\n          : semver.gte(chromeVersion, minChromeVersionS);\n      });\n      if (_.isEmpty(matchingDrivers)) {\n        if (this.storageClient && !didStorageSync) {\n          try {\n            if (await syncChromedrivers(chromeVersion)) {\n              continue;\n            }\n          } catch (e) {\n            this.log.warn(`Cannot synchronize local chromedrivers with the remote storage at ${CD_CDN}: ` +\n              e.message);\n            this.log.debug(e.stack);\n          }\n        }\n        const autodownloadSuggestion =\n          'You could also try to enable automated chromedrivers download server feature';\n        throw new Error(`No Chromedriver found that can automate Chrome '${chromeVersion}'. ` +\n          (!this.storageClient ? `${autodownloadSuggestion}. ` : '') +\n          `See ${CHROMEDRIVER_TUTORIAL} for more details`);\n      }\n\n      const binPath = matchingDrivers[0].executable;\n      this.log.debug(`Found ${util.pluralize('executable', matchingDrivers.length, true)} ` +\n        `capable of automating Chrome '${chromeVersion}'.\\nChoosing the most recent, '${binPath}'.`);\n      this.log.debug('If a specific version is required, specify it with the `chromedriverExecutable`' +\n        'desired capability.');\n      return binPath;\n    // eslint-disable-next-line no-constant-condition\n    } while (true);\n  }\n\n  async initChromedriverPath () {\n    if (this.executableVerified) return; //eslint-disable-line curly\n\n    // the executable might be set (if passed in)\n    // or we might want to use the basic one installed with this driver\n    // or we want to figure out the best one\n    if (!this.chromedriver) {\n      this.chromedriver = this.useSystemExecutable\n        ? await getChromedriverBinaryPath()\n        : await this.getCompatibleChromedriver();\n    }\n\n    if (!await fs.exists(this.chromedriver)) {\n      throw new Error(`Trying to use a chromedriver binary at the path ` +\n                      `${this.chromedriver}, but it doesn't exist!`);\n    }\n    this.executableVerified = true;\n    this.log.info(`Set chromedriver binary as: ${this.chromedriver}`);\n  }\n\n  syncProtocol (cdVersion = null) {\n    const coercedVersion = semver.coerce(cdVersion);\n    if (!coercedVersion || coercedVersion.major < MIN_CD_VERSION_WITH_W3C_SUPPORT) {\n      this.log.debug(`Chromedriver v. ${cdVersion} does not fully support ${PROTOCOLS.W3C} protocol. ` +\n        `Defaulting to ${PROTOCOLS.MJSONWP}`);\n      return;\n    }\n    const chromeOptions = getCapValue(this.capabilities, 'chromeOptions', {});\n    if (chromeOptions.w3c === false) {\n      this.log.info(`Chromedriver v. ${cdVersion} supports ${PROTOCOLS.W3C} protocol, ` +\n        `but ${PROTOCOLS.MJSONWP} one has been explicitly requested`);\n      return;\n    }\n    this.desiredProtocol = PROTOCOLS.W3C;\n    // given caps might not be properly prefixed\n    // so we try to fix them in order to properly init\n    // the new W3C session\n    this.capabilities = toW3cCapNames(this.capabilities);\n  }\n\n  async start (caps, emitStartingState = true) {\n    this.capabilities = _.cloneDeep(caps);\n\n    // set the logging preferences to ALL the console logs\n    this.capabilities.loggingPrefs = _.cloneDeep(getCapValue(caps, 'loggingPrefs', {}));\n    if (_.isEmpty(this.capabilities.loggingPrefs.browser)) {\n      this.capabilities.loggingPrefs.browser = 'ALL';\n    }\n\n    if (emitStartingState) {\n      this.changeState(Chromedriver.STATE_STARTING);\n    }\n\n    const args = [`--port=${this.proxyPort}`];\n    if (this.adb && this.adb.adbPort) {\n      args.push(`--adb-port=${this.adb.adbPort}`);\n    }\n    if (_.isArray(this.cmdArgs)) {\n      args.push(...this.cmdArgs);\n    }\n    if (this.logPath) {\n      args.push(`--log-path=${this.logPath}`);\n    }\n    if (this.disableBuildCheck) {\n      args.push('--disable-build-check');\n    }\n    args.push('--verbose');\n    // what are the process stdout/stderr conditions wherein we know that\n    // the process has started to our satisfaction?\n    const startDetector = (stdout) => stdout.startsWith('Starting ');\n\n    let processIsAlive = false;\n    let webviewVersion;\n    try {\n      await this.initChromedriverPath();\n      await this.killAll();\n\n      // set up our subprocess object\n      this.proc = new SubProcess(this.chromedriver, args);\n      processIsAlive = true;\n\n      // handle log output\n      this.proc.on('output', (stdout, stderr) => {\n        // if the cd output is not printed, find the chrome version and print\n        // will get a response like\n        //   DevTools response: {\n        //      \"Android-Package\": \"io.appium.sampleapp\",\n        //      \"Browser\": \"Chrome/55.0.2883.91\",\n        //      \"Protocol-Version\": \"1.2\",\n        //      \"User-Agent\": \"...\",\n        //      \"WebKit-Version\": \"537.36\"\n        //   }\n        const out = stdout + stderr;\n        let match = /\"Browser\": \"(.*)\"/.exec(out);\n        if (match) {\n          webviewVersion = match[1];\n          this.log.debug(`Webview version: '${webviewVersion}'`);\n        }\n\n        // also print chromedriver version to logs\n        // will output something like\n        //  Starting ChromeDriver 2.33.506106 (8a06c39c4582fbfbab6966dbb1c38a9173bfb1a2) on port 9515\n        match = /Starting ChromeDriver ([.\\d]+)/.exec(out);\n        if (match) {\n          this.log.debug(`Chromedriver version: '${match[1]}'`);\n          this.syncProtocol(match[1]);\n        }\n\n        // give the output if it is requested\n        if (this.verbose) {\n          for (let line of (stdout || '').trim().split('\\n')) {\n            if (!line.trim().length) continue; // eslint-disable-line curly\n            this.log.debug(`[STDOUT] ${line}`);\n          }\n          for (let line of (stderr || '').trim().split('\\n')) {\n            if (!line.trim().length) continue; // eslint-disable-line curly\n            this.log.error(`[STDERR] ${line}`);\n          }\n        }\n      });\n\n      // handle out-of-bound exit by simply emitting a stopped state\n      this.proc.on('exit', (code, signal) => {\n        processIsAlive = false;\n        if (this.state !== Chromedriver.STATE_STOPPED &&\n            this.state !== Chromedriver.STATE_STOPPING &&\n            this.state !== Chromedriver.STATE_RESTARTING) {\n          const msg = `Chromedriver exited unexpectedly with code ${code}, signal ${signal}`;\n          this.log.error(msg);\n          this.changeState(Chromedriver.STATE_STOPPED);\n        }\n      });\n      this.log.info(`Spawning chromedriver with: ${this.chromedriver} ${args.join(' ')}`);\n      // start subproc and wait for startDetector\n      await this.proc.start(startDetector);\n      await this.waitForOnline();\n      await this.startSession();\n    } catch (e) {\n      this.log.debug(e);\n      this.emit(Chromedriver.EVENT_ERROR, e);\n      // just because we had an error doesn't mean the chromedriver process\n      // finished; we should clean up if necessary\n      if (processIsAlive) {\n        await this.proc.stop();\n      }\n\n      let message = '';\n      // often the user's Chrome version is not supported by the version of Chromedriver\n      if (e.message.includes('Chrome version must be')) {\n        message += 'Unable to automate Chrome version because it is not supported by this version of Chromedriver.\\n';\n        if (webviewVersion) {\n          message += `Chrome version on the device: ${webviewVersion}\\n`;\n        }\n        const versionsSupportedByDriver = /Chrome version must be (.+)/.exec(e.message)?.[1] || '';\n        if (versionsSupportedByDriver) {\n          message += `Chromedriver supports Chrome version(s): ${versionsSupportedByDriver}\\n`;\n        }\n        message += `Visit '${CHROMEDRIVER_TUTORIAL}' to troubleshoot the problem.\\n`;\n      }\n\n      message += e.message;\n      this.log.errorAndThrow(message);\n    }\n  }\n\n  sessionId () {\n    return this.state === Chromedriver.STATE_ONLINE ? this.jwproxy.sessionId : null;\n  }\n\n  async restart () {\n    this.log.info('Restarting chromedriver');\n    if (this.state !== Chromedriver.STATE_ONLINE) {\n      throw new Error(\"Can't restart when we're not online\");\n    }\n    this.changeState(Chromedriver.STATE_RESTARTING);\n    await this.stop(false);\n    await this.start(this.capabilities, false);\n  }\n\n  async waitForOnline () {\n    // we need to make sure that CD hasn't crashed\n    let chromedriverStopped = false;\n    await retryInterval(20, 200, async () => {\n      if (this.state === Chromedriver.STATE_STOPPED) {\n        // we are either stopped or stopping, so something went wrong\n        chromedriverStopped = true;\n        return;\n      }\n      await this.getStatus();\n    });\n    if (chromedriverStopped) {\n      throw new Error('ChromeDriver crashed during startup.');\n    }\n  }\n\n  async getStatus () {\n    return await this.jwproxy.command('/status', 'GET');\n  }\n\n  async startSession () {\n    const sessionCaps = this.desiredProtocol === PROTOCOLS.W3C\n      ? {capabilities: {alwaysMatch: this.capabilities}}\n      : {desiredCapabilities: this.capabilities};\n    this.log.info(`Starting ${this.desiredProtocol} Chromedriver session with capabilities: ` +\n      JSON.stringify(sessionCaps, null, 2));\n    await this.jwproxy.command('/session', 'POST', sessionCaps);\n    this.log.prefix = generateLogPrefix(this, this.jwproxy.sessionId);\n    this.changeState(Chromedriver.STATE_ONLINE);\n  }\n\n  async stop (emitStates = true) {\n    if (emitStates) {\n      this.changeState(Chromedriver.STATE_STOPPING);\n    }\n    const runSafeStep = async (f) => {\n      try {\n        return await f();\n      } catch (e) {\n        this.log.warn(e.message);\n        this.log.debug(e.stack);\n      }\n    };\n    await runSafeStep(() => this.jwproxy.command('', 'DELETE'));\n    await runSafeStep(() => this.proc.stop('SIGTERM', 20000));\n    this.log.prefix = generateLogPrefix(this);\n    if (emitStates) {\n      this.changeState(Chromedriver.STATE_STOPPED);\n    }\n  }\n\n  changeState (state) {\n    this.state = state;\n    this.log.debug(`Changed state to '${state}'`);\n    this.emit(Chromedriver.EVENT_CHANGED, {state});\n  }\n\n  async sendCommand (url, method, body) {\n    return await this.jwproxy.command(url, method, body);\n  }\n\n  async proxyReq (req, res) {\n    return await this.jwproxy.proxyReqRes(req, res);\n  }\n\n  async killAll () {\n    let cmd = system.isWindows()\n      ? `wmic process where \"commandline like '%chromedriver.exe%--port=${this.proxyPort}%'\" delete`\n      : `pkill -15 -f \"${this.chromedriver}.*--port=${this.proxyPort}\"`;\n    this.log.debug(`Killing any old chromedrivers, running: ${cmd}`);\n    try {\n      await (B.promisify(cp.exec))(cmd);\n      this.log.debug('Successfully cleaned up old chromedrivers');\n    } catch (err) {\n      this.log.warn('No old chromedrivers seem to exist');\n    }\n\n    if (this.adb) {\n      const udidIndex = this.adb.executable.defaultArgs.findIndex((item) => item === '-s');\n      const udid = udidIndex > -1 ? this.adb.executable.defaultArgs[udidIndex + 1] : null;\n\n      if (udid) {\n        this.log.debug(`Cleaning this device's adb forwarded port socket connections: ${udid}`);\n      } else {\n        this.log.debug(`Cleaning any old adb forwarded port socket connections`);\n      }\n\n      try {\n        for (let conn of await this.adb.getForwardList()) {\n          // chromedriver will ask ADB to forward a port like \"deviceId tcp:port localabstract:webview_devtools_remote_port\"\n          if (!(conn.includes('webview_devtools') && (!udid || conn.includes(udid)))) {\n            continue;\n          }\n\n          let params = conn.split(/\\s+/);\n          if (params.length > 1) {\n            await this.adb.removePortForward(params[1].replace(/[\\D]*/, ''));\n          }\n        }\n      } catch (err) {\n        this.log.warn(`Unable to clean forwarded ports. Error: '${err.message}'. Continuing.`);\n      }\n    }\n  }\n\n  async hasWorkingWebview () {\n    // sometimes chromedriver stops automating webviews. this method runs a\n    // simple command to determine our state, and responds accordingly\n    try {\n      await this.jwproxy.command('/url', 'GET');\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n}\n\nChromedriver.EVENT_ERROR = 'chromedriver_error';\nChromedriver.EVENT_CHANGED = 'stateChanged';\nChromedriver.STATE_STOPPED = 'stopped';\nChromedriver.STATE_STARTING = 'starting';\nChromedriver.STATE_ONLINE = 'online';\nChromedriver.STATE_STOPPING = 'stopping';\nChromedriver.STATE_RESTARTING = 'restarting';\n\nexport { Chromedriver };\nexport default Chromedriver;\n"],"file":"lib/chromedriver.js","sourceRoot":"../.."}
|
package/build/lib/utils.js
CHANGED
|
@@ -6,6 +6,7 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
6
6
|
value: true
|
|
7
7
|
});
|
|
8
8
|
exports.X86 = exports.X64 = exports.OS = exports.M1_ARCH_SUFFIX = exports.CHROMEDRIVER_CHROME_MAPPING = exports.CD_VER = exports.CD_CDN = exports.CD_BASE_DIR = void 0;
|
|
9
|
+
exports.generateLogPrefix = generateLogPrefix;
|
|
9
10
|
exports.getChromeVersion = getChromeVersion;
|
|
10
11
|
exports.getChromedriverBinaryPath = getChromedriverBinaryPath;
|
|
11
12
|
exports.getChromedriverDir = getChromedriverDir;
|
|
@@ -19,6 +20,8 @@ var _lodash = _interopRequireDefault(require("lodash"));
|
|
|
19
20
|
|
|
20
21
|
var _support = require("@appium/support");
|
|
21
22
|
|
|
23
|
+
var _baseDriver = require("@appium/base-driver");
|
|
24
|
+
|
|
22
25
|
var _path = _interopRequireDefault(require("path"));
|
|
23
26
|
|
|
24
27
|
var _compareVersions = _interopRequireDefault(require("compare-versions"));
|
|
@@ -121,7 +124,13 @@ const getOsInfo = _lodash.default.memoize(async function getOsInfo() {
|
|
|
121
124
|
};
|
|
122
125
|
});
|
|
123
126
|
|
|
124
|
-
exports.getOsInfo = getOsInfo;
|
|
127
|
+
exports.getOsInfo = getOsInfo;
|
|
128
|
+
|
|
129
|
+
const getBaseDriverInstance = _lodash.default.memoize(() => new _baseDriver.BaseDriver({}, false));
|
|
130
|
+
|
|
131
|
+
function generateLogPrefix(obj, sessionId = null) {
|
|
132
|
+
return getBaseDriverInstance().helpers.generateDriverLogPrefix(obj, sessionId);
|
|
133
|
+
}require('source-map-support').install();
|
|
125
134
|
|
|
126
135
|
|
|
127
|
-
//# sourceMappingURL=data:application/json;charset=utf8;base64,
|
|
136
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/utils.js"],"names":["rootDir","path","basename","__dirname","resolve","process","env","NO_PRECOMPILE","CHROMEDRIVER_CHROME_MAPPING","require","CD_VER","npm_config_chromedriver_version","CHROMEDRIVER_VERSION","getMostRecentChromedriver","CD_BASE_DIR","CD_CDN","npm_config_chromedriver_cdnurl","CHROMEDRIVER_CDNURL","OS","linux","windows","mac","X64","X86","M1_ARCH_SUFFIX","CD_EXECUTABLE_PREFIX","mapping","_","isEmpty","Error","last","keys","sort","compareVersions","getChromeVersion","adb","bundleId","versionName","getPackageInfo","getChromedriverDir","osName","getOsName","getChromedriverBinaryPath","pathSuffix","paths","fs","glob","cwd","absolute","nocase","nodir","strict","first","retrieveData","url","headers","opts","timeout","responseType","data","memoize","system","isWindows","isMac","getOsInfo","name","arch","hardwareName","trim","getBaseDriverInstance","BaseDriver","generateLogPrefix","obj","sessionId","helpers","generateDriverLogPrefix"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport { system, fs } from '@appium/support';\nimport { BaseDriver } from '@appium/base-driver';\nimport path from 'path';\nimport compareVersions from 'compare-versions';\nimport axios from 'axios';\nimport { exec } from 'teen_process';\n\n\nconst rootDir = path.basename(__dirname) === 'lib'\n  ? path.resolve(__dirname, process.env.NO_PRECOMPILE ? '..' : '../..')\n  : __dirname;\n// Chromedriver version: minimum Chrome version\nconst CHROMEDRIVER_CHROME_MAPPING = require(path.resolve(rootDir, 'config', 'mapping.json'));\n\nconst CD_VER = process.env.npm_config_chromedriver_version\n  || process.env.CHROMEDRIVER_VERSION\n  || getMostRecentChromedriver();\nconst CD_BASE_DIR = path.resolve(__dirname, '..', '..', 'chromedriver');\nconst CD_CDN = process.env.npm_config_chromedriver_cdnurl\n  || process.env.CHROMEDRIVER_CDNURL\n  || 'https://chromedriver.storage.googleapis.com';\nconst OS = {\n  linux: 'linux',\n  windows: 'win',\n  mac: 'mac'\n};\nconst X64 = '64';\nconst X86 = '32';\nconst M1_ARCH_SUFFIX = '_m1';\nconst CD_EXECUTABLE_PREFIX = 'chromedriver';\n\n\nfunction getMostRecentChromedriver (mapping = CHROMEDRIVER_CHROME_MAPPING) {\n  if (_.isEmpty(mapping)) {\n    throw new Error('Unable to get most recent Chromedriver version from empty mapping');\n  }\n  return _.last(_.keys(mapping).sort(compareVersions));\n}\n\nasync function getChromeVersion (adb, bundleId) {\n  const {versionName} = await adb.getPackageInfo(bundleId);\n  return versionName;\n}\n\nfunction getChromedriverDir (osName = getOsName()) {\n  return path.resolve(CD_BASE_DIR, osName);\n}\n\nasync function getChromedriverBinaryPath (osName = getOsName()) {\n  const rootDir = getChromedriverDir(osName);\n  const pathSuffix = osName === OS.windows ? '.exe' : '';\n  const paths = await fs.glob(`${CD_EXECUTABLE_PREFIX}*${pathSuffix}`, {\n    cwd: rootDir,\n    absolute: true,\n    nocase: true,\n    nodir: true,\n    strict: false,\n  });\n  return _.isEmpty(paths)\n    ? path.resolve(rootDir, `${CD_EXECUTABLE_PREFIX}${pathSuffix}`)\n    : _.first(paths);\n}\n\nasync function retrieveData (url, headers, opts = {}) {\n  const {\n    timeout = 5000,\n    responseType = 'text',\n  } = opts;\n  return (await axios({\n    url,\n    headers,\n    timeout,\n    responseType,\n  })).data;\n}\n\nconst getOsName = _.memoize(function getOsName () {\n  if (system.isWindows()) {\n    return OS.windows;\n  }\n  if (system.isMac()) {\n    return OS.mac;\n  }\n  return OS.linux;\n});\n\nconst getOsInfo = _.memoize(async function getOsInfo () {\n  return {\n    name: getOsName(),\n    arch: await system.arch(),\n    hardwareName: system.isWindows() ? null : _.trim(await exec('uname', ['-m'])),\n  };\n});\n\nconst getBaseDriverInstance = _.memoize(() => new BaseDriver({}, false));\n\n/**\n * Generates log prefix string\n *\n * @param {object} obj log owner instance\n * @param {string?} sessionId Optional session identifier\n * @returns {string}\n */\nfunction generateLogPrefix (obj, sessionId = null) {\n  return getBaseDriverInstance().helpers.generateDriverLogPrefix(obj, sessionId);\n}\n\n\nexport {\n  getChromeVersion, getChromedriverDir, getChromedriverBinaryPath, getOsName,\n  CD_BASE_DIR, CD_CDN, CD_VER, CHROMEDRIVER_CHROME_MAPPING, getMostRecentChromedriver,\n  retrieveData, getOsInfo, OS, X64, X86, M1_ARCH_SUFFIX, generateLogPrefix,\n};\n"],"file":"lib/utils.js","sourceRoot":"../.."}
|
package/config/mapping.json
CHANGED
package/lib/chromedriver.js
CHANGED
|
@@ -9,7 +9,7 @@ import { SubProcess, exec } from 'teen_process';
|
|
|
9
9
|
import B from 'bluebird';
|
|
10
10
|
import {
|
|
11
11
|
getChromeVersion, getChromedriverDir, CHROMEDRIVER_CHROME_MAPPING,
|
|
12
|
-
getChromedriverBinaryPath, CD_CDN,
|
|
12
|
+
getChromedriverBinaryPath, CD_CDN, generateLogPrefix
|
|
13
13
|
} from './utils';
|
|
14
14
|
import semver from 'semver';
|
|
15
15
|
import _ from 'lodash';
|
|
@@ -18,9 +18,6 @@ import compareVersions from 'compare-versions';
|
|
|
18
18
|
import ChromedriverStorageClient from './storage-client';
|
|
19
19
|
import { toW3cCapNames, getCapValue } from './protocol-helpers';
|
|
20
20
|
|
|
21
|
-
|
|
22
|
-
const log = logger.getLogger('Chromedriver');
|
|
23
|
-
|
|
24
21
|
const NEW_CD_VERSION_FORMAT_MAJOR_VERSION = 73;
|
|
25
22
|
const DEFAULT_HOST = '127.0.0.1';
|
|
26
23
|
const MIN_CD_VERSION_WITH_W3C_SUPPORT = 75;
|
|
@@ -56,6 +53,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
56
53
|
details,
|
|
57
54
|
isAutodownloadEnabled = false,
|
|
58
55
|
} = args;
|
|
56
|
+
this._log = logger.getLogger(generateLogPrefix(this));
|
|
59
57
|
|
|
60
58
|
this.proxyHost = host;
|
|
61
59
|
this.proxyPort = port;
|
|
@@ -71,7 +69,8 @@ class Chromedriver extends events.EventEmitter {
|
|
|
71
69
|
this.state = Chromedriver.STATE_STOPPED;
|
|
72
70
|
this.jwproxy = new JWProxy({
|
|
73
71
|
server: this.proxyHost,
|
|
74
|
-
port: this.proxyPort
|
|
72
|
+
port: this.proxyPort,
|
|
73
|
+
log: this._log,
|
|
75
74
|
});
|
|
76
75
|
this.verbose = verbose;
|
|
77
76
|
this.logPath = logPath;
|
|
@@ -84,23 +83,27 @@ class Chromedriver extends events.EventEmitter {
|
|
|
84
83
|
this.desiredProtocol = PROTOCOLS.MJSONWP;
|
|
85
84
|
}
|
|
86
85
|
|
|
86
|
+
get log () {
|
|
87
|
+
return this._log;
|
|
88
|
+
}
|
|
89
|
+
|
|
87
90
|
async getDriversMapping () {
|
|
88
91
|
let mapping = _.cloneDeep(CHROMEDRIVER_CHROME_MAPPING);
|
|
89
92
|
if (this.mappingPath) {
|
|
90
|
-
log.debug(`Attempting to use Chromedriver->Chrome mapping from '${this.mappingPath}'`);
|
|
93
|
+
this.log.debug(`Attempting to use Chromedriver->Chrome mapping from '${this.mappingPath}'`);
|
|
91
94
|
if (!await fs.exists(this.mappingPath)) {
|
|
92
|
-
log.warn(`No file found at '${this.mappingPath}'`);
|
|
93
|
-
log.info('Defaulting to the static Chromedriver->Chrome mapping');
|
|
95
|
+
this.log.warn(`No file found at '${this.mappingPath}'`);
|
|
96
|
+
this.log.info('Defaulting to the static Chromedriver->Chrome mapping');
|
|
94
97
|
} else {
|
|
95
98
|
try {
|
|
96
99
|
mapping = JSON.parse(await fs.readFile(this.mappingPath, 'utf8'));
|
|
97
100
|
} catch (err) {
|
|
98
|
-
log.warn(`Error parsing mapping from '${this.mappingPath}': ${err.message}`);
|
|
99
|
-
log.info('Defaulting to the static Chromedriver->Chrome mapping');
|
|
101
|
+
this.log.warn(`Error parsing mapping from '${this.mappingPath}': ${err.message}`);
|
|
102
|
+
this.log.info('Defaulting to the static Chromedriver->Chrome mapping');
|
|
100
103
|
}
|
|
101
104
|
}
|
|
102
105
|
} else {
|
|
103
|
-
log.debug('Using the static Chromedriver->Chrome mapping');
|
|
106
|
+
this.log.debug('Using the static Chromedriver->Chrome mapping');
|
|
104
107
|
}
|
|
105
108
|
|
|
106
109
|
// make sure that the values for minimum chrome version are semver compliant
|
|
@@ -109,7 +112,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
109
112
|
if (coercedVersion) {
|
|
110
113
|
mapping[cdVersion] = coercedVersion.version;
|
|
111
114
|
} else {
|
|
112
|
-
log.info(`'${chromeVersion}' is not a valid version number. Skipping it`);
|
|
115
|
+
this.log.info(`'${chromeVersion}' is not a valid version number. Skipping it`);
|
|
113
116
|
}
|
|
114
117
|
}
|
|
115
118
|
return mapping;
|
|
@@ -118,7 +121,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
118
121
|
async getChromedrivers (mapping) {
|
|
119
122
|
// go through the versions available
|
|
120
123
|
const executables = await fs.glob(`${this.executableDir}/*`);
|
|
121
|
-
log.debug(`Found ${util.pluralize('executable', executables.length, true)} ` +
|
|
124
|
+
this.log.debug(`Found ${util.pluralize('executable', executables.length, true)} ` +
|
|
122
125
|
`in '${this.executableDir}'`);
|
|
123
126
|
const cds = (await asyncmap(executables, async function mapChromedriver (executable) {
|
|
124
127
|
const logError = ({message, stdout = null, stderr = null}) => {
|
|
@@ -130,7 +133,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
130
133
|
if (stderr) {
|
|
131
134
|
errMsg += `\nStderr: ${stderr}`;
|
|
132
135
|
}
|
|
133
|
-
log.warn(errMsg);
|
|
136
|
+
this.log.warn(errMsg);
|
|
134
137
|
return null;
|
|
135
138
|
};
|
|
136
139
|
|
|
@@ -177,12 +180,12 @@ class Chromedriver extends events.EventEmitter {
|
|
|
177
180
|
.filter((cd) => !!cd)
|
|
178
181
|
.sort((a, b) => compareVersions(b.version, a.version));
|
|
179
182
|
if (_.isEmpty(cds)) {
|
|
180
|
-
log.info(`No Chromedrivers were found in '${this.executableDir}'`);
|
|
183
|
+
this.log.info(`No Chromedrivers were found in '${this.executableDir}'`);
|
|
181
184
|
return cds;
|
|
182
185
|
}
|
|
183
|
-
log.debug(`The following Chromedriver executables were found:`);
|
|
186
|
+
this.log.debug(`The following Chromedriver executables were found:`);
|
|
184
187
|
for (const cd of cds) {
|
|
185
|
-
log.debug(` '${cd.executable}' (version '${cd.version}', minimum Chrome version '${cd.minChromeVersion ? cd.minChromeVersion : 'Unknown'}')`);
|
|
188
|
+
this.log.debug(` '${cd.executable}' (version '${cd.version}', minimum Chrome version '${cd.minChromeVersion ? cd.minChromeVersion : 'Unknown'}')`);
|
|
186
189
|
}
|
|
187
190
|
return cds;
|
|
188
191
|
}
|
|
@@ -192,7 +195,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
192
195
|
// The `info` item must contain the output of /json/version CDP command
|
|
193
196
|
// where `Browser` field looks like `Chrome/72.0.3601.0``
|
|
194
197
|
if (this.details?.info) {
|
|
195
|
-
log.debug(`Browser version in the supplied details: ${this.details?.info?.Browser}`);
|
|
198
|
+
this.log.debug(`Browser version in the supplied details: ${this.details?.info?.Browser}`);
|
|
196
199
|
}
|
|
197
200
|
const versionMatch = VERSION_PATTERN.exec(this.details?.info?.Browser);
|
|
198
201
|
if (versionMatch) {
|
|
@@ -256,7 +259,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
256
259
|
await fs.writeFile(this.mappingPath, JSON.stringify(newMapping, null, 2), 'utf8');
|
|
257
260
|
shouldUpdateStaticMapping = false;
|
|
258
261
|
} catch (e) {
|
|
259
|
-
log.warn(`Cannot store the updated chromedrivers mapping into '${this.mappingPath}'. ` +
|
|
262
|
+
this.log.warn(`Cannot store the updated chromedrivers mapping into '${this.mappingPath}'. ` +
|
|
260
263
|
`This may reduce the performance of further executions. Original error: ${e.message}`);
|
|
261
264
|
}
|
|
262
265
|
}
|
|
@@ -272,14 +275,14 @@ class Chromedriver extends events.EventEmitter {
|
|
|
272
275
|
|
|
273
276
|
const mapping = await this.getDriversMapping();
|
|
274
277
|
if (!_.isEmpty(mapping)) {
|
|
275
|
-
log.debug(`The most recent known Chrome version: ${_.values(mapping)[0]}`);
|
|
278
|
+
this.log.debug(`The most recent known Chrome version: ${_.values(mapping)[0]}`);
|
|
276
279
|
}
|
|
277
280
|
|
|
278
281
|
let didStorageSync = false;
|
|
279
282
|
const syncChromedrivers = async (chromeVersion) => {
|
|
280
283
|
didStorageSync = true;
|
|
281
284
|
const retrievedMapping = await this.storageClient.retrieveMapping();
|
|
282
|
-
log.debug('Got chromedrivers mapping from the storage: ' +
|
|
285
|
+
this.log.debug('Got chromedrivers mapping from the storage: ' +
|
|
283
286
|
JSON.stringify(retrievedMapping, null, 2));
|
|
284
287
|
const driverKeys = await this.storageClient.syncDrivers({
|
|
285
288
|
minBrowserVersion: chromeVersion.major,
|
|
@@ -313,7 +316,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
313
316
|
missingVersions[version] = minChromeVersion;
|
|
314
317
|
}
|
|
315
318
|
if (!_.isEmpty(missingVersions)) {
|
|
316
|
-
log.info(`Found ${util.pluralize('Chromedriver', _.size(missingVersions), true)}, ` +
|
|
319
|
+
this.log.info(`Found ${util.pluralize('Chromedriver', _.size(missingVersions), true)}, ` +
|
|
317
320
|
`which ${_.size(missingVersions) === 1 ? 'is' : 'are'} missing in the list of known versions: ` +
|
|
318
321
|
JSON.stringify(missingVersions));
|
|
319
322
|
await this.updateDriversMapping(Object.assign(mapping, missingVersions));
|
|
@@ -321,12 +324,12 @@ class Chromedriver extends events.EventEmitter {
|
|
|
321
324
|
|
|
322
325
|
if (this.disableBuildCheck) {
|
|
323
326
|
if (_.isEmpty(cds)) {
|
|
324
|
-
log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` +
|
|
327
|
+
this.log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` +
|
|
325
328
|
`'chromedriverDisableBuildCheck' capability is set to 'true'`);
|
|
326
329
|
}
|
|
327
330
|
const {version, executable} = cds[0];
|
|
328
|
-
log.warn(`Chrome build check disabled. Using most recent Chromedriver version (${version}, at '${executable}')`);
|
|
329
|
-
log.warn(`If this is wrong, set 'chromedriverDisableBuildCheck' capability to 'false'`);
|
|
331
|
+
this.log.warn(`Chrome build check disabled. Using most recent Chromedriver version (${version}, at '${executable}')`);
|
|
332
|
+
this.log.warn(`If this is wrong, set 'chromedriverDisableBuildCheck' capability to 'false'`);
|
|
330
333
|
return executable;
|
|
331
334
|
}
|
|
332
335
|
|
|
@@ -334,14 +337,14 @@ class Chromedriver extends events.EventEmitter {
|
|
|
334
337
|
if (!chromeVersion) {
|
|
335
338
|
// unable to get the chrome version
|
|
336
339
|
if (_.isEmpty(cds)) {
|
|
337
|
-
log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` +
|
|
340
|
+
this.log.errorAndThrow(`There must be at least one Chromedriver executable available for use if ` +
|
|
338
341
|
`the current Chrome version cannot be determined`);
|
|
339
342
|
}
|
|
340
343
|
const {version, executable} = cds[0];
|
|
341
|
-
log.warn(`Unable to discover Chrome version. Using Chromedriver ${version} at '${executable}'`);
|
|
344
|
+
this.log.warn(`Unable to discover Chrome version. Using Chromedriver ${version} at '${executable}'`);
|
|
342
345
|
return executable;
|
|
343
346
|
}
|
|
344
|
-
log.debug(`Found Chrome bundle '${this.bundleId}' version '${chromeVersion}'`);
|
|
347
|
+
this.log.debug(`Found Chrome bundle '${this.bundleId}' version '${chromeVersion}'`);
|
|
345
348
|
|
|
346
349
|
const matchingDrivers = cds.filter(({minChromeVersion}) => {
|
|
347
350
|
const minChromeVersionS = minChromeVersion && semver.coerce(minChromeVersion);
|
|
@@ -360,9 +363,9 @@ class Chromedriver extends events.EventEmitter {
|
|
|
360
363
|
continue;
|
|
361
364
|
}
|
|
362
365
|
} catch (e) {
|
|
363
|
-
log.warn(`Cannot synchronize local chromedrivers with the remote storage at ${CD_CDN}: ` +
|
|
366
|
+
this.log.warn(`Cannot synchronize local chromedrivers with the remote storage at ${CD_CDN}: ` +
|
|
364
367
|
e.message);
|
|
365
|
-
log.debug(e.stack);
|
|
368
|
+
this.log.debug(e.stack);
|
|
366
369
|
}
|
|
367
370
|
}
|
|
368
371
|
const autodownloadSuggestion =
|
|
@@ -373,9 +376,9 @@ class Chromedriver extends events.EventEmitter {
|
|
|
373
376
|
}
|
|
374
377
|
|
|
375
378
|
const binPath = matchingDrivers[0].executable;
|
|
376
|
-
log.debug(`Found ${util.pluralize('executable', matchingDrivers.length, true)} ` +
|
|
379
|
+
this.log.debug(`Found ${util.pluralize('executable', matchingDrivers.length, true)} ` +
|
|
377
380
|
`capable of automating Chrome '${chromeVersion}'.\nChoosing the most recent, '${binPath}'.`);
|
|
378
|
-
log.debug('If a specific version is required, specify it with the `chromedriverExecutable`' +
|
|
381
|
+
this.log.debug('If a specific version is required, specify it with the `chromedriverExecutable`' +
|
|
379
382
|
'desired capability.');
|
|
380
383
|
return binPath;
|
|
381
384
|
// eslint-disable-next-line no-constant-condition
|
|
@@ -399,19 +402,19 @@ class Chromedriver extends events.EventEmitter {
|
|
|
399
402
|
`${this.chromedriver}, but it doesn't exist!`);
|
|
400
403
|
}
|
|
401
404
|
this.executableVerified = true;
|
|
402
|
-
log.info(`Set chromedriver binary as: ${this.chromedriver}`);
|
|
405
|
+
this.log.info(`Set chromedriver binary as: ${this.chromedriver}`);
|
|
403
406
|
}
|
|
404
407
|
|
|
405
408
|
syncProtocol (cdVersion = null) {
|
|
406
409
|
const coercedVersion = semver.coerce(cdVersion);
|
|
407
410
|
if (!coercedVersion || coercedVersion.major < MIN_CD_VERSION_WITH_W3C_SUPPORT) {
|
|
408
|
-
log.debug(`Chromedriver v. ${cdVersion} does not fully support ${PROTOCOLS.W3C} protocol. ` +
|
|
411
|
+
this.log.debug(`Chromedriver v. ${cdVersion} does not fully support ${PROTOCOLS.W3C} protocol. ` +
|
|
409
412
|
`Defaulting to ${PROTOCOLS.MJSONWP}`);
|
|
410
413
|
return;
|
|
411
414
|
}
|
|
412
415
|
const chromeOptions = getCapValue(this.capabilities, 'chromeOptions', {});
|
|
413
416
|
if (chromeOptions.w3c === false) {
|
|
414
|
-
log.info(`Chromedriver v. ${cdVersion} supports ${PROTOCOLS.W3C} protocol, ` +
|
|
417
|
+
this.log.info(`Chromedriver v. ${cdVersion} supports ${PROTOCOLS.W3C} protocol, ` +
|
|
415
418
|
`but ${PROTOCOLS.MJSONWP} one has been explicitly requested`);
|
|
416
419
|
return;
|
|
417
420
|
}
|
|
@@ -478,7 +481,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
478
481
|
let match = /"Browser": "(.*)"/.exec(out);
|
|
479
482
|
if (match) {
|
|
480
483
|
webviewVersion = match[1];
|
|
481
|
-
log.debug(`Webview version: '${webviewVersion}'`);
|
|
484
|
+
this.log.debug(`Webview version: '${webviewVersion}'`);
|
|
482
485
|
}
|
|
483
486
|
|
|
484
487
|
// also print chromedriver version to logs
|
|
@@ -486,7 +489,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
486
489
|
// Starting ChromeDriver 2.33.506106 (8a06c39c4582fbfbab6966dbb1c38a9173bfb1a2) on port 9515
|
|
487
490
|
match = /Starting ChromeDriver ([.\d]+)/.exec(out);
|
|
488
491
|
if (match) {
|
|
489
|
-
log.debug(`Chromedriver version: '${match[1]}'`);
|
|
492
|
+
this.log.debug(`Chromedriver version: '${match[1]}'`);
|
|
490
493
|
this.syncProtocol(match[1]);
|
|
491
494
|
}
|
|
492
495
|
|
|
@@ -494,11 +497,11 @@ class Chromedriver extends events.EventEmitter {
|
|
|
494
497
|
if (this.verbose) {
|
|
495
498
|
for (let line of (stdout || '').trim().split('\n')) {
|
|
496
499
|
if (!line.trim().length) continue; // eslint-disable-line curly
|
|
497
|
-
log.debug(`[STDOUT] ${line}`);
|
|
500
|
+
this.log.debug(`[STDOUT] ${line}`);
|
|
498
501
|
}
|
|
499
502
|
for (let line of (stderr || '').trim().split('\n')) {
|
|
500
503
|
if (!line.trim().length) continue; // eslint-disable-line curly
|
|
501
|
-
log.error(`[STDERR] ${line}`);
|
|
504
|
+
this.log.error(`[STDERR] ${line}`);
|
|
502
505
|
}
|
|
503
506
|
}
|
|
504
507
|
});
|
|
@@ -509,20 +512,18 @@ class Chromedriver extends events.EventEmitter {
|
|
|
509
512
|
if (this.state !== Chromedriver.STATE_STOPPED &&
|
|
510
513
|
this.state !== Chromedriver.STATE_STOPPING &&
|
|
511
514
|
this.state !== Chromedriver.STATE_RESTARTING) {
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
log.error(msg);
|
|
515
|
+
const msg = `Chromedriver exited unexpectedly with code ${code}, signal ${signal}`;
|
|
516
|
+
this.log.error(msg);
|
|
515
517
|
this.changeState(Chromedriver.STATE_STOPPED);
|
|
516
518
|
}
|
|
517
519
|
});
|
|
518
|
-
log.info(`Spawning chromedriver with: ${this.chromedriver} `
|
|
519
|
-
`${args.join(' ')}`);
|
|
520
|
+
this.log.info(`Spawning chromedriver with: ${this.chromedriver} ${args.join(' ')}`);
|
|
520
521
|
// start subproc and wait for startDetector
|
|
521
522
|
await this.proc.start(startDetector);
|
|
522
523
|
await this.waitForOnline();
|
|
523
524
|
await this.startSession();
|
|
524
525
|
} catch (e) {
|
|
525
|
-
log.debug(e);
|
|
526
|
+
this.log.debug(e);
|
|
526
527
|
this.emit(Chromedriver.EVENT_ERROR, e);
|
|
527
528
|
// just because we had an error doesn't mean the chromedriver process
|
|
528
529
|
// finished; we should clean up if necessary
|
|
@@ -545,20 +546,16 @@ class Chromedriver extends events.EventEmitter {
|
|
|
545
546
|
}
|
|
546
547
|
|
|
547
548
|
message += e.message;
|
|
548
|
-
log.errorAndThrow(message);
|
|
549
|
+
this.log.errorAndThrow(message);
|
|
549
550
|
}
|
|
550
551
|
}
|
|
551
552
|
|
|
552
553
|
sessionId () {
|
|
553
|
-
|
|
554
|
-
return null;
|
|
555
|
-
}
|
|
556
|
-
|
|
557
|
-
return this.jwproxy.sessionId;
|
|
554
|
+
return this.state === Chromedriver.STATE_ONLINE ? this.jwproxy.sessionId : null;
|
|
558
555
|
}
|
|
559
556
|
|
|
560
557
|
async restart () {
|
|
561
|
-
log.info('Restarting chromedriver');
|
|
558
|
+
this.log.info('Restarting chromedriver');
|
|
562
559
|
if (this.state !== Chromedriver.STATE_ONLINE) {
|
|
563
560
|
throw new Error("Can't restart when we're not online");
|
|
564
561
|
}
|
|
@@ -591,9 +588,10 @@ class Chromedriver extends events.EventEmitter {
|
|
|
591
588
|
const sessionCaps = this.desiredProtocol === PROTOCOLS.W3C
|
|
592
589
|
? {capabilities: {alwaysMatch: this.capabilities}}
|
|
593
590
|
: {desiredCapabilities: this.capabilities};
|
|
594
|
-
log.info(`Starting ${this.desiredProtocol} Chromedriver session with capabilities: ` +
|
|
591
|
+
this.log.info(`Starting ${this.desiredProtocol} Chromedriver session with capabilities: ` +
|
|
595
592
|
JSON.stringify(sessionCaps, null, 2));
|
|
596
593
|
await this.jwproxy.command('/session', 'POST', sessionCaps);
|
|
594
|
+
this.log.prefix = generateLogPrefix(this, this.jwproxy.sessionId);
|
|
597
595
|
this.changeState(Chromedriver.STATE_ONLINE);
|
|
598
596
|
}
|
|
599
597
|
|
|
@@ -601,20 +599,25 @@ class Chromedriver extends events.EventEmitter {
|
|
|
601
599
|
if (emitStates) {
|
|
602
600
|
this.changeState(Chromedriver.STATE_STOPPING);
|
|
603
601
|
}
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
this.
|
|
602
|
+
const runSafeStep = async (f) => {
|
|
603
|
+
try {
|
|
604
|
+
return await f();
|
|
605
|
+
} catch (e) {
|
|
606
|
+
this.log.warn(e.message);
|
|
607
|
+
this.log.debug(e.stack);
|
|
609
608
|
}
|
|
610
|
-
}
|
|
611
|
-
|
|
609
|
+
};
|
|
610
|
+
await runSafeStep(() => this.jwproxy.command('', 'DELETE'));
|
|
611
|
+
await runSafeStep(() => this.proc.stop('SIGTERM', 20000));
|
|
612
|
+
this.log.prefix = generateLogPrefix(this);
|
|
613
|
+
if (emitStates) {
|
|
614
|
+
this.changeState(Chromedriver.STATE_STOPPED);
|
|
612
615
|
}
|
|
613
616
|
}
|
|
614
617
|
|
|
615
618
|
changeState (state) {
|
|
616
619
|
this.state = state;
|
|
617
|
-
log.debug(`Changed state to '${state}'`);
|
|
620
|
+
this.log.debug(`Changed state to '${state}'`);
|
|
618
621
|
this.emit(Chromedriver.EVENT_CHANGED, {state});
|
|
619
622
|
}
|
|
620
623
|
|
|
@@ -630,12 +633,12 @@ class Chromedriver extends events.EventEmitter {
|
|
|
630
633
|
let cmd = system.isWindows()
|
|
631
634
|
? `wmic process where "commandline like '%chromedriver.exe%--port=${this.proxyPort}%'" delete`
|
|
632
635
|
: `pkill -15 -f "${this.chromedriver}.*--port=${this.proxyPort}"`;
|
|
633
|
-
log.debug(`Killing any old chromedrivers, running: ${cmd}`);
|
|
636
|
+
this.log.debug(`Killing any old chromedrivers, running: ${cmd}`);
|
|
634
637
|
try {
|
|
635
638
|
await (B.promisify(cp.exec))(cmd);
|
|
636
|
-
log.debug('Successfully cleaned up old chromedrivers');
|
|
639
|
+
this.log.debug('Successfully cleaned up old chromedrivers');
|
|
637
640
|
} catch (err) {
|
|
638
|
-
log.warn('No old chromedrivers seem to exist');
|
|
641
|
+
this.log.warn('No old chromedrivers seem to exist');
|
|
639
642
|
}
|
|
640
643
|
|
|
641
644
|
if (this.adb) {
|
|
@@ -643,9 +646,9 @@ class Chromedriver extends events.EventEmitter {
|
|
|
643
646
|
const udid = udidIndex > -1 ? this.adb.executable.defaultArgs[udidIndex + 1] : null;
|
|
644
647
|
|
|
645
648
|
if (udid) {
|
|
646
|
-
log.debug(`Cleaning this device's adb forwarded port socket connections: ${udid}`);
|
|
649
|
+
this.log.debug(`Cleaning this device's adb forwarded port socket connections: ${udid}`);
|
|
647
650
|
} else {
|
|
648
|
-
log.debug(`Cleaning any old adb forwarded port socket connections`);
|
|
651
|
+
this.log.debug(`Cleaning any old adb forwarded port socket connections`);
|
|
649
652
|
}
|
|
650
653
|
|
|
651
654
|
try {
|
|
@@ -661,7 +664,7 @@ class Chromedriver extends events.EventEmitter {
|
|
|
661
664
|
}
|
|
662
665
|
}
|
|
663
666
|
} catch (err) {
|
|
664
|
-
log.warn(`Unable to clean forwarded ports. Error: '${err.message}'. Continuing.`);
|
|
667
|
+
this.log.warn(`Unable to clean forwarded ports. Error: '${err.message}'. Continuing.`);
|
|
665
668
|
}
|
|
666
669
|
}
|
|
667
670
|
}
|
package/lib/utils.js
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import _ from 'lodash';
|
|
2
2
|
import { system, fs } from '@appium/support';
|
|
3
|
+
import { BaseDriver } from '@appium/base-driver';
|
|
3
4
|
import path from 'path';
|
|
4
5
|
import compareVersions from 'compare-versions';
|
|
5
6
|
import axios from 'axios';
|
|
@@ -92,9 +93,22 @@ const getOsInfo = _.memoize(async function getOsInfo () {
|
|
|
92
93
|
};
|
|
93
94
|
});
|
|
94
95
|
|
|
96
|
+
const getBaseDriverInstance = _.memoize(() => new BaseDriver({}, false));
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* Generates log prefix string
|
|
100
|
+
*
|
|
101
|
+
* @param {object} obj log owner instance
|
|
102
|
+
* @param {string?} sessionId Optional session identifier
|
|
103
|
+
* @returns {string}
|
|
104
|
+
*/
|
|
105
|
+
function generateLogPrefix (obj, sessionId = null) {
|
|
106
|
+
return getBaseDriverInstance().helpers.generateDriverLogPrefix(obj, sessionId);
|
|
107
|
+
}
|
|
108
|
+
|
|
95
109
|
|
|
96
110
|
export {
|
|
97
111
|
getChromeVersion, getChromedriverDir, getChromedriverBinaryPath, getOsName,
|
|
98
112
|
CD_BASE_DIR, CD_CDN, CD_VER, CHROMEDRIVER_CHROME_MAPPING, getMostRecentChromedriver,
|
|
99
|
-
retrieveData, getOsInfo, OS, X64, X86, M1_ARCH_SUFFIX,
|
|
113
|
+
retrieveData, getOsInfo, OS, X64, X86, M1_ARCH_SUFFIX, generateLogPrefix,
|
|
100
114
|
};
|
package/package.json
CHANGED
|
@@ -6,7 +6,7 @@
|
|
|
6
6
|
"chrome",
|
|
7
7
|
"android"
|
|
8
8
|
],
|
|
9
|
-
"version": "5.0.
|
|
9
|
+
"version": "5.0.3",
|
|
10
10
|
"author": "appium",
|
|
11
11
|
"license": "Apache-2.0",
|
|
12
12
|
"repository": {
|
|
@@ -35,14 +35,14 @@
|
|
|
35
35
|
"config/mapping.json"
|
|
36
36
|
],
|
|
37
37
|
"dependencies": {
|
|
38
|
-
"@appium/base-driver": "^8.
|
|
38
|
+
"@appium/base-driver": "^8.4.0",
|
|
39
39
|
"@appium/support": "^2.55.3",
|
|
40
40
|
"@babel/runtime": "^7.0.0",
|
|
41
41
|
"@xmldom/xmldom": "^0.x",
|
|
42
42
|
"asyncbox": "^2.0.2",
|
|
43
43
|
"axios": "^0.x",
|
|
44
44
|
"bluebird": "^3.5.1",
|
|
45
|
-
"compare-versions": "^
|
|
45
|
+
"compare-versions": "^4.1.3",
|
|
46
46
|
"fancy-log": "^2.0.0",
|
|
47
47
|
"lodash": "^4.17.4",
|
|
48
48
|
"semver": "^7.0.0",
|
|
@@ -76,15 +76,13 @@
|
|
|
76
76
|
"@appium/test-support": "^1.0.0",
|
|
77
77
|
"@babel/core": "^7.16.0",
|
|
78
78
|
"@babel/eslint-parser": "^7.16.3",
|
|
79
|
+
"@semantic-release/git": "^10.0.1",
|
|
79
80
|
"chai": "^4.1.2",
|
|
80
81
|
"chai-as-promised": "^7.1.1",
|
|
81
|
-
"eslint": "^7.32.0",
|
|
82
|
-
"eslint-plugin-import": "^2.25.3",
|
|
83
|
-
"eslint-plugin-mocha": "^9.0.0",
|
|
84
|
-
"eslint-plugin-promise": "^6.0.0",
|
|
85
82
|
"gulp": "^4.0.0",
|
|
86
83
|
"mocha": "^9.0.0",
|
|
87
84
|
"pre-commit": "^1.1.3",
|
|
88
|
-
"
|
|
85
|
+
"semantic-release": "^19.0.2",
|
|
86
|
+
"sinon": "^13.0.0"
|
|
89
87
|
}
|
|
90
88
|
}
|