@appium/base-driver 8.2.2 → 8.3.0

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.
Files changed (61) hide show
  1. package/build/lib/basedriver/capabilities.js +3 -1
  2. package/build/lib/basedriver/commands/find.js +4 -11
  3. package/build/lib/basedriver/commands/log.js +3 -6
  4. package/build/lib/basedriver/commands/session.js +18 -27
  5. package/build/lib/basedriver/commands/settings.js +4 -8
  6. package/build/lib/basedriver/commands/timeout.js +10 -15
  7. package/build/lib/basedriver/device-settings.js +14 -2
  8. package/build/lib/basedriver/driver.js +25 -23
  9. package/build/lib/basedriver/helpers.js +140 -84
  10. package/build/lib/express/express-logging.js +2 -2
  11. package/build/lib/express/idempotency.js +2 -2
  12. package/build/lib/helpers/capabilities.js +39 -0
  13. package/build/lib/index.js +7 -7
  14. package/build/lib/jsonwp-proxy/protocol-converter.js +19 -16
  15. package/build/lib/jsonwp-proxy/proxy.js +20 -16
  16. package/build/lib/protocol/errors.js +4 -2
  17. package/build/lib/protocol/helpers.js +3 -20
  18. package/build/lib/protocol/protocol.js +44 -45
  19. package/build/lib/protocol/routes.js +67 -1
  20. package/build/test/basedriver/capabilities-specs.js +43 -1
  21. package/build/test/basedriver/capability-specs.js +126 -167
  22. package/build/test/basedriver/commands/log-specs.js +12 -5
  23. package/build/test/basedriver/driver-tests.js +11 -14
  24. package/build/test/basedriver/helpers-specs.js +5 -1
  25. package/build/test/basedriver/timeout-specs.js +7 -9
  26. package/build/test/express/server-e2e-specs.js +10 -5
  27. package/build/test/express/server-specs.js +22 -16
  28. package/build/test/express/static-specs.js +10 -5
  29. package/build/test/jsonwp-proxy/proxy-e2e-specs.js +1 -2
  30. package/build/test/jsonwp-proxy/proxy-specs.js +1 -6
  31. package/build/test/protocol/fake-driver.js +12 -15
  32. package/build/test/protocol/protocol-e2e-specs.js +49 -103
  33. package/build/test/protocol/routes-specs.js +2 -2
  34. package/lib/basedriver/capabilities.js +3 -0
  35. package/lib/basedriver/commands/find.js +3 -6
  36. package/lib/basedriver/commands/log.js +2 -4
  37. package/lib/basedriver/commands/session.js +21 -22
  38. package/lib/basedriver/commands/settings.js +3 -5
  39. package/lib/basedriver/commands/timeout.js +9 -10
  40. package/lib/basedriver/device-settings.js +10 -1
  41. package/lib/basedriver/driver.js +29 -16
  42. package/lib/basedriver/helpers.js +201 -83
  43. package/lib/express/express-logging.js +1 -1
  44. package/lib/express/idempotency.js +1 -1
  45. package/lib/helpers/capabilities.js +25 -0
  46. package/lib/index.js +6 -4
  47. package/lib/jsonwp-proxy/protocol-converter.js +15 -18
  48. package/lib/jsonwp-proxy/proxy.js +17 -15
  49. package/lib/protocol/errors.js +1 -1
  50. package/lib/protocol/helpers.js +5 -25
  51. package/lib/protocol/protocol.js +43 -54
  52. package/lib/protocol/routes.js +60 -1
  53. package/package.json +29 -22
  54. package/test/basedriver/capabilities-specs.js +34 -2
  55. package/test/basedriver/capability-specs.js +120 -146
  56. package/test/basedriver/commands/log-specs.js +12 -3
  57. package/test/basedriver/driver-tests.js +12 -7
  58. package/test/basedriver/helpers-specs.js +4 -0
  59. package/test/basedriver/timeout-specs.js +6 -11
  60. package/build/lib/protocol/sessions-cache.js +0 -88
  61. package/lib/protocol/sessions-cache.js +0 -74
@@ -13,6 +13,12 @@ var _lodash = _interopRequireDefault(require("lodash"));
13
13
 
14
14
  var _logger = _interopRequireDefault(require("./logger"));
15
15
 
16
+ var _support = require("@appium/support");
17
+
18
+ var _errors = require("../protocol/errors");
19
+
20
+ const MAX_SETTINGS_SIZE = 20 * 1024 * 1024;
21
+
16
22
  class DeviceSettings {
17
23
  constructor(defaultSettings = {}, onSettingsUpdate = null) {
18
24
  this._settings = Object.assign({}, defaultSettings);
@@ -21,7 +27,13 @@ class DeviceSettings {
21
27
 
22
28
  async update(newSettings) {
23
29
  if (!_lodash.default.isPlainObject(newSettings)) {
24
- throw new Error(`Settings update should be called with valid JSON. Got ` + `${JSON.stringify(newSettings)} instead`);
30
+ throw new _errors.errors.InvalidArgumentError(`Settings update should be called with valid JSON. Got ` + `${JSON.stringify(newSettings)} instead`);
31
+ }
32
+
33
+ if (_support.node.getObjectSize({ ...this._settings,
34
+ ...newSettings
35
+ }) >= MAX_SETTINGS_SIZE) {
36
+ throw new _errors.errors.InvalidArgumentError(`New settings cannot be applied, because the overall ` + `object size exceeds the allowed limit of ${_support.util.toReadableSizeString(MAX_SETTINGS_SIZE)}`);
25
37
  }
26
38
 
27
39
  for (const prop of _lodash.default.keys(newSettings)) {
@@ -53,4 +65,4 @@ var _default = DeviceSettings;
53
65
  exports.default = _default;require('source-map-support').install();
54
66
 
55
67
 
56
- //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9iYXNlZHJpdmVyL2RldmljZS1zZXR0aW5ncy5qcyJdLCJuYW1lcyI6WyJEZXZpY2VTZXR0aW5ncyIsImNvbnN0cnVjdG9yIiwiZGVmYXVsdFNldHRpbmdzIiwib25TZXR0aW5nc1VwZGF0ZSIsIl9zZXR0aW5ncyIsIk9iamVjdCIsImFzc2lnbiIsInVwZGF0ZSIsIm5ld1NldHRpbmdzIiwiXyIsImlzUGxhaW5PYmplY3QiLCJFcnJvciIsIkpTT04iLCJzdHJpbmdpZnkiLCJwcm9wIiwia2V5cyIsImlzVW5kZWZpbmVkIiwibG9nIiwiZGVidWciLCJpc0Z1bmN0aW9uIiwiZXJyb3JBbmRUaHJvdyIsIm5hbWUiLCJnZXRTZXR0aW5ncyJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFFQSxNQUFNQSxjQUFOLENBQXFCO0FBRW5CQyxFQUFBQSxXQUFXLENBQUVDLGVBQWUsR0FBRyxFQUFwQixFQUF3QkMsZ0JBQWdCLEdBQUcsSUFBM0MsRUFBaUQ7QUFDMUQsU0FBS0MsU0FBTCxHQUFpQkMsTUFBTSxDQUFDQyxNQUFQLENBQWMsRUFBZCxFQUFrQkosZUFBbEIsQ0FBakI7QUFDQSxTQUFLQyxnQkFBTCxHQUF3QkEsZ0JBQXhCO0FBQ0Q7O0FBR1csUUFBTkksTUFBTSxDQUFFQyxXQUFGLEVBQWU7QUFDekIsUUFBSSxDQUFDQyxnQkFBRUMsYUFBRixDQUFnQkYsV0FBaEIsQ0FBTCxFQUFtQztBQUNqQyxZQUFNLElBQUlHLEtBQUosQ0FBVyx3REFBRCxHQUNiLEdBQUVDLElBQUksQ0FBQ0MsU0FBTCxDQUFlTCxXQUFmLENBQTRCLFVBRDNCLENBQU47QUFFRDs7QUFDRCxTQUFLLE1BQU1NLElBQVgsSUFBbUJMLGdCQUFFTSxJQUFGLENBQU9QLFdBQVAsQ0FBbkIsRUFBd0M7QUFDdEMsVUFBSSxDQUFDQyxnQkFBRU8sV0FBRixDQUFjLEtBQUtaLFNBQUwsQ0FBZVUsSUFBZixDQUFkLENBQUwsRUFBMEM7QUFDeEMsWUFBSSxLQUFLVixTQUFMLENBQWVVLElBQWYsTUFBeUJOLFdBQVcsQ0FBQ00sSUFBRCxDQUF4QyxFQUFnRDtBQUM5Q0csMEJBQUlDLEtBQUosQ0FBVyxpQkFBZ0JKLElBQUssc0RBQWhDOztBQUNBO0FBQ0Q7QUFDRjs7QUFFRCxVQUFJTCxnQkFBRVUsVUFBRixDQUFhLEtBQUtoQixnQkFBbEIsQ0FBSixFQUF5QztBQUN2QyxjQUFNLEtBQUtBLGdCQUFMLENBQXNCVyxJQUF0QixFQUE0Qk4sV0FBVyxDQUFDTSxJQUFELENBQXZDLEVBQStDLEtBQUtWLFNBQUwsQ0FBZVUsSUFBZixDQUEvQyxDQUFOO0FBQ0EsYUFBS1YsU0FBTCxDQUFlVSxJQUFmLElBQXVCTixXQUFXLENBQUNNLElBQUQsQ0FBbEM7QUFDRCxPQUhELE1BR087QUFDTEcsd0JBQUlHLGFBQUosQ0FBbUIsNkJBQUQsR0FDZix5Q0FBd0MsS0FBS25CLFdBQUwsQ0FBaUJvQixJQUFLLEdBRGpFO0FBRUQ7QUFDRjtBQUNGOztBQUVEQyxFQUFBQSxXQUFXLEdBQUk7QUFDYixXQUFPLEtBQUtsQixTQUFaO0FBQ0Q7O0FBakNrQjs7O2VBb0NOSixjIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IF8gZnJvbSAnbG9kYXNoJztcbmltcG9ydCBsb2cgZnJvbSAnLi9sb2dnZXInO1xuXG5jbGFzcyBEZXZpY2VTZXR0aW5ncyB7XG5cbiAgY29uc3RydWN0b3IgKGRlZmF1bHRTZXR0aW5ncyA9IHt9LCBvblNldHRpbmdzVXBkYXRlID0gbnVsbCkge1xuICAgIHRoaXMuX3NldHRpbmdzID0gT2JqZWN0LmFzc2lnbih7fSwgZGVmYXVsdFNldHRpbmdzKTtcbiAgICB0aGlzLm9uU2V0dGluZ3NVcGRhdGUgPSBvblNldHRpbmdzVXBkYXRlO1xuICB9XG5cbiAgLy8gY2FsbHMgdXBkYXRlU2V0dGluZ3MgZnJvbSBpbXBsZW1lbnRpbmcgZHJpdmVyIGV2ZXJ5IHRpbWUgYSBzZXR0aW5nIGlzIGNoYW5nZWQuXG4gIGFzeW5jIHVwZGF0ZSAobmV3U2V0dGluZ3MpIHtcbiAgICBpZiAoIV8uaXNQbGFpbk9iamVjdChuZXdTZXR0aW5ncykpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgU2V0dGluZ3MgdXBkYXRlIHNob3VsZCBiZSBjYWxsZWQgd2l0aCB2YWxpZCBKU09OLiBHb3QgYCArXG4gICAgICAgIGAke0pTT04uc3RyaW5naWZ5KG5ld1NldHRpbmdzKX0gaW5zdGVhZGApO1xuICAgIH1cbiAgICBmb3IgKGNvbnN0IHByb3Agb2YgXy5rZXlzKG5ld1NldHRpbmdzKSkge1xuICAgICAgaWYgKCFfLmlzVW5kZWZpbmVkKHRoaXMuX3NldHRpbmdzW3Byb3BdKSkge1xuICAgICAgICBpZiAodGhpcy5fc2V0dGluZ3NbcHJvcF0gPT09IG5ld1NldHRpbmdzW3Byb3BdKSB7XG4gICAgICAgICAgbG9nLmRlYnVnKGBUaGUgdmFsdWUgb2YgJyR7cHJvcH0nIHNldHRpbmcgZGlkIG5vdCBjaGFuZ2UuIFNraXBwaW5nIHRoZSB1cGRhdGUgZm9yIGl0YCk7XG4gICAgICAgICAgY29udGludWU7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIC8vIHVwZGF0ZSBzZXR0aW5nIG9ubHkgd2hlbiB0aGVyZSBpcyB1cGRhdGVTZXR0aW5ncyBkZWZpbmVkLlxuICAgICAgaWYgKF8uaXNGdW5jdGlvbih0aGlzLm9uU2V0dGluZ3NVcGRhdGUpKSB7XG4gICAgICAgIGF3YWl0IHRoaXMub25TZXR0aW5nc1VwZGF0ZShwcm9wLCBuZXdTZXR0aW5nc1twcm9wXSwgdGhpcy5fc2V0dGluZ3NbcHJvcF0pO1xuICAgICAgICB0aGlzLl9zZXR0aW5nc1twcm9wXSA9IG5ld1NldHRpbmdzW3Byb3BdO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgbG9nLmVycm9yQW5kVGhyb3coYFVuYWJsZSB0byB1cGRhdGUgc2V0dGluZ3M7IGAgK1xuICAgICAgICAgIGBvblNldHRpbmdzVXBkYXRlIG1ldGhvZCBub3QgZm91bmQgb24gJyR7dGhpcy5jb25zdHJ1Y3Rvci5uYW1lfSdgKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICBnZXRTZXR0aW5ncyAoKSB7XG4gICAgcmV0dXJuIHRoaXMuX3NldHRpbmdzO1xuICB9XG59XG5cbmV4cG9ydCBkZWZhdWx0IERldmljZVNldHRpbmdzO1xuZXhwb3J0IHsgRGV2aWNlU2V0dGluZ3MgfTtcbiJdLCJmaWxlIjoibGliL2Jhc2Vkcml2ZXIvZGV2aWNlLXNldHRpbmdzLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uIn0=
68
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9iYXNlZHJpdmVyL2RldmljZS1zZXR0aW5ncy5qcyJdLCJuYW1lcyI6WyJNQVhfU0VUVElOR1NfU0laRSIsIkRldmljZVNldHRpbmdzIiwiY29uc3RydWN0b3IiLCJkZWZhdWx0U2V0dGluZ3MiLCJvblNldHRpbmdzVXBkYXRlIiwiX3NldHRpbmdzIiwiT2JqZWN0IiwiYXNzaWduIiwidXBkYXRlIiwibmV3U2V0dGluZ3MiLCJfIiwiaXNQbGFpbk9iamVjdCIsImVycm9ycyIsIkludmFsaWRBcmd1bWVudEVycm9yIiwiSlNPTiIsInN0cmluZ2lmeSIsIm5vZGUiLCJnZXRPYmplY3RTaXplIiwidXRpbCIsInRvUmVhZGFibGVTaXplU3RyaW5nIiwicHJvcCIsImtleXMiLCJpc1VuZGVmaW5lZCIsImxvZyIsImRlYnVnIiwiaXNGdW5jdGlvbiIsImVycm9yQW5kVGhyb3ciLCJuYW1lIiwiZ2V0U2V0dGluZ3MiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBRUEsTUFBTUEsaUJBQWlCLEdBQUcsS0FBSyxJQUFMLEdBQVksSUFBdEM7O0FBRUEsTUFBTUMsY0FBTixDQUFxQjtBQUVuQkMsRUFBQUEsV0FBVyxDQUFFQyxlQUFlLEdBQUcsRUFBcEIsRUFBd0JDLGdCQUFnQixHQUFHLElBQTNDLEVBQWlEO0FBQzFELFNBQUtDLFNBQUwsR0FBaUJDLE1BQU0sQ0FBQ0MsTUFBUCxDQUFjLEVBQWQsRUFBa0JKLGVBQWxCLENBQWpCO0FBQ0EsU0FBS0MsZ0JBQUwsR0FBd0JBLGdCQUF4QjtBQUNEOztBQUdXLFFBQU5JLE1BQU0sQ0FBRUMsV0FBRixFQUFlO0FBQ3pCLFFBQUksQ0FBQ0MsZ0JBQUVDLGFBQUYsQ0FBZ0JGLFdBQWhCLENBQUwsRUFBbUM7QUFDakMsWUFBTSxJQUFJRyxlQUFPQyxvQkFBWCxDQUFpQyx3REFBRCxHQUNuQyxHQUFFQyxJQUFJLENBQUNDLFNBQUwsQ0FBZU4sV0FBZixDQUE0QixVQUQzQixDQUFOO0FBRUQ7O0FBQ0QsUUFBSU8sY0FBS0MsYUFBTCxDQUFtQixFQUFDLEdBQUcsS0FBS1osU0FBVDtBQUFvQixTQUFHSTtBQUF2QixLQUFuQixLQUEyRFQsaUJBQS9ELEVBQWtGO0FBQ2hGLFlBQU0sSUFBSVksZUFBT0Msb0JBQVgsQ0FBaUMsc0RBQUQsR0FDbkMsNENBQTJDSyxjQUFLQyxvQkFBTCxDQUEwQm5CLGlCQUExQixDQUE2QyxFQURyRixDQUFOO0FBRUQ7O0FBRUQsU0FBSyxNQUFNb0IsSUFBWCxJQUFtQlYsZ0JBQUVXLElBQUYsQ0FBT1osV0FBUCxDQUFuQixFQUF3QztBQUN0QyxVQUFJLENBQUNDLGdCQUFFWSxXQUFGLENBQWMsS0FBS2pCLFNBQUwsQ0FBZWUsSUFBZixDQUFkLENBQUwsRUFBMEM7QUFDeEMsWUFBSSxLQUFLZixTQUFMLENBQWVlLElBQWYsTUFBeUJYLFdBQVcsQ0FBQ1csSUFBRCxDQUF4QyxFQUFnRDtBQUM5Q0csMEJBQUlDLEtBQUosQ0FBVyxpQkFBZ0JKLElBQUssc0RBQWhDOztBQUNBO0FBQ0Q7QUFDRjs7QUFFRCxVQUFJVixnQkFBRWUsVUFBRixDQUFhLEtBQUtyQixnQkFBbEIsQ0FBSixFQUF5QztBQUN2QyxjQUFNLEtBQUtBLGdCQUFMLENBQXNCZ0IsSUFBdEIsRUFBNEJYLFdBQVcsQ0FBQ1csSUFBRCxDQUF2QyxFQUErQyxLQUFLZixTQUFMLENBQWVlLElBQWYsQ0FBL0MsQ0FBTjtBQUNBLGFBQUtmLFNBQUwsQ0FBZWUsSUFBZixJQUF1QlgsV0FBVyxDQUFDVyxJQUFELENBQWxDO0FBQ0QsT0FIRCxNQUdPO0FBQ0xHLHdCQUFJRyxhQUFKLENBQW1CLDZCQUFELEdBQ2YseUNBQXdDLEtBQUt4QixXQUFMLENBQWlCeUIsSUFBSyxHQURqRTtBQUVEO0FBQ0Y7QUFDRjs7QUFFREMsRUFBQUEsV0FBVyxHQUFJO0FBQ2IsV0FBTyxLQUFLdkIsU0FBWjtBQUNEOztBQXRDa0I7OztlQXlDTkosYyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgbG9nIGZyb20gJy4vbG9nZ2VyJztcbmltcG9ydCB7IG5vZGUsIHV0aWwgfSBmcm9tICdAYXBwaXVtL3N1cHBvcnQnO1xuaW1wb3J0IHsgZXJyb3JzIH0gZnJvbSAnLi4vcHJvdG9jb2wvZXJyb3JzJztcblxuY29uc3QgTUFYX1NFVFRJTkdTX1NJWkUgPSAyMCAqIDEwMjQgKiAxMDI0OyAvLyAyMCBNQlxuXG5jbGFzcyBEZXZpY2VTZXR0aW5ncyB7XG5cbiAgY29uc3RydWN0b3IgKGRlZmF1bHRTZXR0aW5ncyA9IHt9LCBvblNldHRpbmdzVXBkYXRlID0gbnVsbCkge1xuICAgIHRoaXMuX3NldHRpbmdzID0gT2JqZWN0LmFzc2lnbih7fSwgZGVmYXVsdFNldHRpbmdzKTtcbiAgICB0aGlzLm9uU2V0dGluZ3NVcGRhdGUgPSBvblNldHRpbmdzVXBkYXRlO1xuICB9XG5cbiAgLy8gY2FsbHMgdXBkYXRlU2V0dGluZ3MgZnJvbSBpbXBsZW1lbnRpbmcgZHJpdmVyIGV2ZXJ5IHRpbWUgYSBzZXR0aW5nIGlzIGNoYW5nZWQuXG4gIGFzeW5jIHVwZGF0ZSAobmV3U2V0dGluZ3MpIHtcbiAgICBpZiAoIV8uaXNQbGFpbk9iamVjdChuZXdTZXR0aW5ncykpIHtcbiAgICAgIHRocm93IG5ldyBlcnJvcnMuSW52YWxpZEFyZ3VtZW50RXJyb3IoYFNldHRpbmdzIHVwZGF0ZSBzaG91bGQgYmUgY2FsbGVkIHdpdGggdmFsaWQgSlNPTi4gR290IGAgK1xuICAgICAgICBgJHtKU09OLnN0cmluZ2lmeShuZXdTZXR0aW5ncyl9IGluc3RlYWRgKTtcbiAgICB9XG4gICAgaWYgKG5vZGUuZ2V0T2JqZWN0U2l6ZSh7Li4udGhpcy5fc2V0dGluZ3MsIC4uLm5ld1NldHRpbmdzfSkgPj0gTUFYX1NFVFRJTkdTX1NJWkUpIHtcbiAgICAgIHRocm93IG5ldyBlcnJvcnMuSW52YWxpZEFyZ3VtZW50RXJyb3IoYE5ldyBzZXR0aW5ncyBjYW5ub3QgYmUgYXBwbGllZCwgYmVjYXVzZSB0aGUgb3ZlcmFsbCBgICtcbiAgICAgICAgYG9iamVjdCBzaXplIGV4Y2VlZHMgdGhlIGFsbG93ZWQgbGltaXQgb2YgJHt1dGlsLnRvUmVhZGFibGVTaXplU3RyaW5nKE1BWF9TRVRUSU5HU19TSVpFKX1gKTtcbiAgICB9XG5cbiAgICBmb3IgKGNvbnN0IHByb3Agb2YgXy5rZXlzKG5ld1NldHRpbmdzKSkge1xuICAgICAgaWYgKCFfLmlzVW5kZWZpbmVkKHRoaXMuX3NldHRpbmdzW3Byb3BdKSkge1xuICAgICAgICBpZiAodGhpcy5fc2V0dGluZ3NbcHJvcF0gPT09IG5ld1NldHRpbmdzW3Byb3BdKSB7XG4gICAgICAgICAgbG9nLmRlYnVnKGBUaGUgdmFsdWUgb2YgJyR7cHJvcH0nIHNldHRpbmcgZGlkIG5vdCBjaGFuZ2UuIFNraXBwaW5nIHRoZSB1cGRhdGUgZm9yIGl0YCk7XG4gICAgICAgICAgY29udGludWU7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIC8vIHVwZGF0ZSBzZXR0aW5nIG9ubHkgd2hlbiB0aGVyZSBpcyB1cGRhdGVTZXR0aW5ncyBkZWZpbmVkLlxuICAgICAgaWYgKF8uaXNGdW5jdGlvbih0aGlzLm9uU2V0dGluZ3NVcGRhdGUpKSB7XG4gICAgICAgIGF3YWl0IHRoaXMub25TZXR0aW5nc1VwZGF0ZShwcm9wLCBuZXdTZXR0aW5nc1twcm9wXSwgdGhpcy5fc2V0dGluZ3NbcHJvcF0pO1xuICAgICAgICB0aGlzLl9zZXR0aW5nc1twcm9wXSA9IG5ld1NldHRpbmdzW3Byb3BdO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgbG9nLmVycm9yQW5kVGhyb3coYFVuYWJsZSB0byB1cGRhdGUgc2V0dGluZ3M7IGAgK1xuICAgICAgICAgIGBvblNldHRpbmdzVXBkYXRlIG1ldGhvZCBub3QgZm91bmQgb24gJyR7dGhpcy5jb25zdHJ1Y3Rvci5uYW1lfSdgKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICBnZXRTZXR0aW5ncyAoKSB7XG4gICAgcmV0dXJuIHRoaXMuX3NldHRpbmdzO1xuICB9XG59XG5cbmV4cG9ydCBkZWZhdWx0IERldmljZVNldHRpbmdzO1xuZXhwb3J0IHsgRGV2aWNlU2V0dGluZ3MgfTtcbiJdLCJmaWxlIjoibGliL2Jhc2Vkcml2ZXIvZGV2aWNlLXNldHRpbmdzLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uIn0=
@@ -7,8 +7,6 @@ Object.defineProperty(exports, "__esModule", {
7
7
  });
8
8
  exports.default = exports.BaseDriver = void 0;
9
9
 
10
- var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
11
-
12
10
  require("source-map-support/register");
13
11
 
14
12
  var _protocol = require("../protocol");
@@ -23,8 +21,6 @@ var _commands = _interopRequireDefault(require("./commands"));
23
21
 
24
22
  var helpers = _interopRequireWildcard(require("./helpers"));
25
23
 
26
- var _logger = _interopRequireDefault(require("./logger"));
27
-
28
24
  var _deviceSettings = _interopRequireDefault(require("./device-settings"));
29
25
 
30
26
  var _desiredCaps = require("./desired-caps");
@@ -59,6 +55,8 @@ const EVENT_SESSION_QUIT_DONE = 'quitSessionFinished';
59
55
  const ON_UNEXPECTED_SHUTDOWN_EVENT = 'onUnexpectedShutdown';
60
56
 
61
57
  class BaseDriver extends _protocol.Protocol {
58
+ static baseVersion = BASEDRIVER_VER;
59
+
62
60
  constructor(opts = {}, shouldValidateCaps = true) {
63
61
  super();
64
62
  this.sessionId = null;
@@ -66,6 +64,7 @@ class BaseDriver extends _protocol.Protocol {
66
64
  this.caps = null;
67
65
  this.originalCaps = null;
68
66
  this.helpers = helpers;
67
+ this._log = null;
69
68
  this.basePath = _constants.DEFAULT_BASE_PATH;
70
69
  this.relaxedSecurityEnabled = false;
71
70
  this.allowInsecure = [];
@@ -90,6 +89,15 @@ class BaseDriver extends _protocol.Protocol {
90
89
  this.protocol = null;
91
90
  }
92
91
 
92
+ get log() {
93
+ if (!this._log) {
94
+ const instanceName = `${this.constructor.name}@${_support.node.getObjectId(this).substring(0, 8)}`;
95
+ this._log = _support.logger.getLogger(() => this.sessionId ? `${instanceName} (${this.sessionId.substring(0, 8)})` : instanceName);
96
+ }
97
+
98
+ return this._log;
99
+ }
100
+
93
101
  onUnexpectedShutdown(handler) {
94
102
  this.eventEmitter.on(ON_UNEXPECTED_SHUTDOWN_EVENT, handler);
95
103
  }
@@ -124,7 +132,7 @@ class BaseDriver extends _protocol.Protocol {
124
132
 
125
133
  this._eventHistory[eventName].push(ts);
126
134
 
127
- _logger.default.debug(`Event '${eventName}' logged at ${ts} (${logTime})`);
135
+ this.log.debug(`Event '${eventName}' logged at ${ts} (${logTime})`);
128
136
  }
129
137
 
130
138
  async getStatus() {
@@ -160,10 +168,10 @@ class BaseDriver extends _protocol.Protocol {
160
168
  let extraCaps = _lodash.default.difference(_lodash.default.keys(caps), _lodash.default.keys(this._constraints));
161
169
 
162
170
  if (extraCaps.length) {
163
- _logger.default.warn(`The following capabilities were provided, but are not ` + `recognized by Appium:`);
171
+ this.log.warn(`The following capabilities were provided, but are not ` + `recognized by Appium:`);
164
172
 
165
173
  for (const cap of extraCaps) {
166
- _logger.default.warn(` ${cap}`);
174
+ this.log.warn(` ${cap}`);
167
175
  }
168
176
  }
169
177
  }
@@ -176,7 +184,7 @@ class BaseDriver extends _protocol.Protocol {
176
184
  try {
177
185
  (0, _capabilities.validateCaps)(caps, this._constraints);
178
186
  } catch (e) {
179
- _logger.default.errorAndThrow(new _protocol.errors.SessionNotCreatedError(`The desiredCapabilities object was not valid for the ` + `following reason(s): ${e.message}`));
187
+ this.log.errorAndThrow(new _protocol.errors.SessionNotCreatedError(`The desiredCapabilities object was not valid for the ` + `following reason(s): ${e.message}`));
180
188
  }
181
189
 
182
190
  this.logExtraCaps(caps);
@@ -225,9 +233,9 @@ class BaseDriver extends _protocol.Protocol {
225
233
  let startTime = Date.now();
226
234
 
227
235
  if (cmd === 'createSession') {
228
- this.protocol = (0, _protocol.determineProtocol)(...args);
236
+ this.protocol = (0, _protocol.determineProtocol)(args);
229
237
  this.logEvent(EVENT_SESSION_INIT);
230
- } else if (cmd === 'deleteSession') {
238
+ } else if (cmd === _protocol.DELETE_SESSION_COMMAND) {
231
239
  this.logEvent(EVENT_SESSION_QUIT_START);
232
240
  }
233
241
 
@@ -255,7 +263,7 @@ class BaseDriver extends _protocol.Protocol {
255
263
 
256
264
  const res = this.isCommandsQueueEnabled ? await this.commandsQueueGuard.acquire(BaseDriver.name, commandExecutor) : await commandExecutor();
257
265
 
258
- if (this.isCommandsQueueEnabled && cmd !== 'deleteSession') {
266
+ if (this.isCommandsQueueEnabled && cmd !== _protocol.DELETE_SESSION_COMMAND) {
259
267
  this.startNewCommandTimeout();
260
268
  }
261
269
 
@@ -269,7 +277,7 @@ class BaseDriver extends _protocol.Protocol {
269
277
 
270
278
  if (cmd === 'createSession') {
271
279
  this.logEvent(EVENT_SESSION_START);
272
- } else if (cmd === 'deleteSession') {
280
+ } else if (cmd === _protocol.DELETE_SESSION_COMMAND) {
273
281
  this.logEvent(EVENT_SESSION_QUIT_DONE);
274
282
  }
275
283
 
@@ -289,8 +297,7 @@ class BaseDriver extends _protocol.Protocol {
289
297
 
290
298
  validateLocatorStrategy(strategy, webContext = false) {
291
299
  let validStrategies = this.locatorStrategies;
292
-
293
- _logger.default.debug(`Valid locator strategies for this request: ${validStrategies.join(', ')}`);
300
+ this.log.debug(`Valid locator strategies for this request: ${validStrategies.join(', ')}`);
294
301
 
295
302
  if (webContext) {
296
303
  validStrategies = validStrategies.concat(this.webLocatorStrategies);
@@ -302,10 +309,8 @@ class BaseDriver extends _protocol.Protocol {
302
309
  }
303
310
 
304
311
  async reset() {
305
- _logger.default.debug('Resetting app mid-session');
306
-
307
- _logger.default.debug('Running generic full reset');
308
-
312
+ this.log.debug('Resetting app mid-session');
313
+ this.log.debug('Running generic full reset');
309
314
  let currentConfig = {};
310
315
 
311
316
  for (let property of ['implicitWaitMs', 'newCommandTimeoutMs', 'sessionId', 'resetOnUnexpectedShutdown']) {
@@ -316,9 +321,7 @@ class BaseDriver extends _protocol.Protocol {
316
321
 
317
322
  try {
318
323
  await this.deleteSession(this.sessionId);
319
-
320
- _logger.default.debug('Restarting app');
321
-
324
+ this.log.debug('Restarting app');
322
325
  await this.createSession(undefined, undefined, this.originalCaps);
323
326
  } finally {
324
327
  for (let [key, value] of _lodash.default.toPairs(currentConfig)) {
@@ -378,7 +381,6 @@ class BaseDriver extends _protocol.Protocol {
378
381
  }
379
382
 
380
383
  exports.BaseDriver = BaseDriver;
381
- (0, _defineProperty2.default)(BaseDriver, "baseVersion", BASEDRIVER_VER);
382
384
 
383
385
  for (let [cmd, fn] of _lodash.default.toPairs(_commands.default)) {
384
386
  BaseDriver.prototype[cmd] = fn;
@@ -388,4 +390,4 @@ var _default = BaseDriver;
388
390
  exports.default = _default;require('source-map-support').install();
389
391
 
390
392
 
391
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/basedriver/driver.js"],"names":["version","BASEDRIVER_VER","fs","readPackageJsonFrom","__dirname","B","config","cancellation","NEW_COMMAND_TIMEOUT_MS","EVENT_SESSION_INIT","EVENT_SESSION_START","EVENT_SESSION_QUIT_START","EVENT_SESSION_QUIT_DONE","ON_UNEXPECTED_SHUTDOWN_EVENT","BaseDriver","Protocol","constructor","opts","shouldValidateCaps","sessionId","caps","originalCaps","helpers","basePath","DEFAULT_BASE_PATH","relaxedSecurityEnabled","allowInsecure","denyInsecure","newCommandTimeoutMs","implicitWaitMs","_constraints","_","cloneDeep","desiredCapabilityConstraints","locatorStrategies","webLocatorStrategies","tmpDir","process","env","APPIUM_TMP_DIR","os","tmpdir","shutdownUnexpectedly","noCommandTimer","commandsQueueGuard","AsyncLock","settings","DeviceSettings","noop","initialOpts","managedDrivers","_eventHistory","commands","eventEmitter","EventEmitter","protocol","onUnexpectedShutdown","handler","on","driverData","isCommandsQueueEnabled","eventHistory","logEvent","eventName","Error","ts","Date","now","logTime","toTimeString","push","log","debug","getStatus","desiredCapConstraints","constraints","Object","assign","value","toPairs","presence","allowEmpty","sessionExists","driverForSession","logExtraCaps","extraCaps","difference","keys","length","warn","cap","validateDesiredCaps","e","errorAndThrow","errors","SessionNotCreatedError","message","isMjsonwpProtocol","PROTOCOLS","MJSONWP","isW3CProtocol","W3C","setProtocolMJSONWP","setProtocolW3C","isFeatureEnabled","name","includes","ensureFeatureEnabled","executeCommand","cmd","args","startTime","clearNewCommandTimeout","NoSuchDriverError","NotYetImplementedError","unexpectedShutdownListener","commandExecutor","race","resolve","reject","finally","removeListener","res","acquire","startNewCommandTimeout","endTime","startUnexpectedShutdown","err","emit","deleteSession","validateLocatorStrategy","strategy","webContext","validStrategies","join","concat","InvalidSelectorError","reset","currentConfig","property","resetOnUnexpectedShutdown","createSession","undefined","key","proxyActive","getProxyAvoidList","canProxy","proxyRouteIsAvoided","method","url","avoidSchema","isArray","avoidMethod","avoidPathRegex","isRegExp","normalizedUrl","replace","RegExp","escapeRegExp","test","addManagedDriver","driver","getManagedDrivers","fn","prototype"],"mappings":"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","sourcesContent":["import {\n  Protocol, errors, determineProtocol\n} from '../protocol';\nimport { fs } from '@appium/support';\nimport { PROTOCOLS, DEFAULT_BASE_PATH } from '../constants';\nimport os from 'os';\nimport commands from './commands';\nimport * as helpers from './helpers';\nimport log from './logger';\nimport DeviceSettings from './device-settings';\nimport { desiredCapabilityConstraints } from './desired-caps';\nimport { validateCaps } from './capabilities';\nimport B from 'bluebird';\nimport _ from 'lodash';\nimport AsyncLock from 'async-lock';\nimport { EventEmitter } from 'events';\n\n// for compat with running tests transpiled and in-place\nconst {version: BASEDRIVER_VER} = fs.readPackageJsonFrom(__dirname);\n\nB.config({\n  cancellation: true,\n});\n\nconst NEW_COMMAND_TIMEOUT_MS = 60 * 1000;\n\nconst EVENT_SESSION_INIT = 'newSessionRequested';\nconst EVENT_SESSION_START = 'newSessionStarted';\nconst EVENT_SESSION_QUIT_START = 'quitSessionRequested';\nconst EVENT_SESSION_QUIT_DONE = 'quitSessionFinished';\nconst ON_UNEXPECTED_SHUTDOWN_EVENT = 'onUnexpectedShutdown';\n\nclass BaseDriver extends Protocol {\n\n  /**\n   * Make the basedriver version available so for any driver which inherits from this package, we\n   * know which version of basedriver it inherited from\n   */\n  static baseVersion = BASEDRIVER_VER;\n\n  constructor (opts = {}, shouldValidateCaps = true) {\n    super();\n\n    // setup state\n    this.sessionId = null;\n    this.opts = opts;\n    this.caps = null;\n    this.originalCaps = null; // To give the original capabilities to reset\n    this.helpers = helpers;\n\n    // basePath is used for several purposes, for example in setting up\n    // proxying to other drivers, since we need to know what the base path\n    // of any incoming request might look like. We set it to the default\n    // initially but it is automatically updated during any actual program\n    // execution by the routeConfiguringFunction, which is necessarily run as\n    // the entrypoint for any Appium server\n    this.basePath = DEFAULT_BASE_PATH;\n\n    // initialize security modes\n    this.relaxedSecurityEnabled = false;\n    this.allowInsecure = [];\n    this.denyInsecure = [];\n\n    // timeout initialization\n    this.newCommandTimeoutMs = NEW_COMMAND_TIMEOUT_MS;\n    this.implicitWaitMs = 0;\n\n    this._constraints = _.cloneDeep(desiredCapabilityConstraints);\n    this.locatorStrategies = [];\n    this.webLocatorStrategies = [];\n\n    // use a custom tmp dir to avoid losing data and app when computer is\n    // restarted\n    this.opts.tmpDir = this.opts.tmpDir ||\n                       process.env.APPIUM_TMP_DIR ||\n                       os.tmpdir();\n\n    // base-driver internals\n    this.shutdownUnexpectedly = false;\n    this.noCommandTimer = null;\n    this.shouldValidateCaps = shouldValidateCaps;\n    this.commandsQueueGuard = new AsyncLock();\n\n    // settings should be instantiated by drivers which extend BaseDriver, but\n    // we set it to an empty DeviceSettings instance here to make sure that the\n    // default settings are applied even if an extending driver doesn't utilize\n    // the settings functionality itself\n    this.settings = new DeviceSettings({}, _.noop);\n\n    // keeping track of initial opts\n    this.initialOpts = _.cloneDeep(this.opts);\n\n    // allow subclasses to have internal drivers\n    this.managedDrivers = [];\n\n    // store event timings\n    this._eventHistory = {\n      commands: [] // commands get a special place\n    };\n\n    // used to handle driver events\n    this.eventEmitter = new EventEmitter();\n\n    this.protocol = null;\n  }\n\n  /**\n   * Set a callback handler if needed to execute a custom piece of code\n   * when the driver is shut down unexpectedly. Multiple calls to this method\n   * will cause the handler to be executed mutiple times\n   *\n   * @param {Function} handler The code to be executed on unexpected shutdown.\n   * The function may accept one argument, which is the actual error instance, which\n   * caused the driver to shut down.\n   */\n  onUnexpectedShutdown (handler) {\n    this.eventEmitter.on(ON_UNEXPECTED_SHUTDOWN_EVENT, handler);\n  }\n\n  /**\n   * This property is used by AppiumDriver to store the data of the\n   * specific driver sessions. This data can be later used to adjust\n   * properties for driver instances running in parallel.\n   * Override it in inherited driver classes if necessary.\n   *\n   * @return {object} Driver properties mapping\n   */\n  get driverData () {\n    return {};\n  }\n\n  /**\n   * This property controls the way {#executeCommand} method\n   * handles new driver commands received from the client.\n   * Override it for inherited classes only in special cases.\n   *\n   * @return {boolean} If the returned value is true (default) then all the commands\n   *   received by the particular driver instance are going to be put into the queue,\n   *   so each following command will not be executed until the previous command\n   *   execution is completed. False value disables that queue, so each driver command\n   *   is executed independently and does not wait for anything.\n   */\n  get isCommandsQueueEnabled () {\n    return true;\n  }\n\n  /*\n   * make eventHistory a property and return a cloned object so a consumer can't\n   * inadvertently change data outside of logEvent\n   */\n  get eventHistory () {\n    return _.cloneDeep(this._eventHistory);\n  }\n\n  /*\n   * API method for driver developers to log timings for important events\n   */\n  logEvent (eventName) {\n    if (eventName === 'commands') {\n      throw new Error('Cannot log commands directly');\n    }\n    if (typeof eventName !== 'string') {\n      throw new Error(`Invalid eventName ${eventName}`);\n    }\n    if (!this._eventHistory[eventName]) {\n      this._eventHistory[eventName] = [];\n    }\n    const ts = Date.now();\n    const logTime = (new Date(ts)).toTimeString();\n    this._eventHistory[eventName].push(ts);\n    log.debug(`Event '${eventName}' logged at ${ts} (${logTime})`);\n  }\n\n  /*\n   * Overridden in appium driver, but here so that individual drivers can be\n   * tested with clients that poll\n   */\n  async getStatus () { // eslint-disable-line require-await\n    return {};\n  }\n\n  // we only want subclasses to ever extend the contraints\n  set desiredCapConstraints (constraints) {\n    this._constraints = Object.assign(this._constraints, constraints);\n    // 'presence' means different things in different versions of the validator,\n    // when we say 'true' we mean that it should not be able to be empty\n    for (const [, value] of _.toPairs(this._constraints)) {\n      if (value && value.presence === true) {\n        value.presence = {\n          allowEmpty: false,\n        };\n      }\n    }\n  }\n\n  get desiredCapConstraints () {\n    return this._constraints;\n  }\n\n  // method required by MJSONWP in order to determine whether it should\n  // respond with an invalid session response\n  sessionExists (sessionId) {\n    if (!sessionId) return false; // eslint-disable-line curly\n    return sessionId === this.sessionId;\n  }\n\n  // method required by MJSONWP in order to determine if the command should\n  // be proxied directly to the driver\n  driverForSession (/*sessionId*/) {\n    return this;\n  }\n\n  logExtraCaps (caps) {\n    let extraCaps = _.difference(_.keys(caps),\n                                 _.keys(this._constraints));\n    if (extraCaps.length) {\n      log.warn(`The following capabilities were provided, but are not ` +\n               `recognized by Appium:`);\n      for (const cap of extraCaps) {\n        log.warn(`  ${cap}`);\n      }\n    }\n  }\n\n  validateDesiredCaps (caps) {\n    if (!this.shouldValidateCaps) {\n      return true;\n    }\n\n    try {\n      validateCaps(caps, this._constraints);\n    } catch (e) {\n      log.errorAndThrow(new errors.SessionNotCreatedError(`The desiredCapabilities object was not valid for the ` +\n                    `following reason(s): ${e.message}`));\n    }\n\n    this.logExtraCaps(caps);\n\n    return true;\n  }\n\n  isMjsonwpProtocol () {\n    return this.protocol === PROTOCOLS.MJSONWP;\n  }\n\n  isW3CProtocol () {\n    return this.protocol === PROTOCOLS.W3C;\n  }\n\n  setProtocolMJSONWP () {\n    this.protocol = PROTOCOLS.MJSONWP;\n  }\n\n  setProtocolW3C () {\n    this.protocol = PROTOCOLS.W3C;\n  }\n\n  /**\n   * Check whether a given feature is enabled via its name\n   *\n   * @param {string} name - name of feature/command\n   *\n   * @returns {Boolean}\n   */\n  isFeatureEnabled (name) {\n    // if we have explicitly denied this feature, return false immediately\n    if (this.denyInsecure && _.includes(this.denyInsecure, name)) {\n      return false;\n    }\n\n    // if we specifically have allowed the feature, return true\n    if (this.allowInsecure && _.includes(this.allowInsecure, name)) {\n      return true;\n    }\n\n    // otherwise, if we've globally allowed insecure features and not denied\n    // this one, return true\n    if (this.relaxedSecurityEnabled) {\n      return true;\n    }\n\n    // if we haven't allowed anything insecure, then reject\n    return false;\n  }\n\n  /**\n   * Assert that a given feature is enabled and throw a helpful error if it's\n   * not\n   *\n   * @param {string} name - name of feature/command\n   */\n  ensureFeatureEnabled (name) {\n    if (!this.isFeatureEnabled(name)) {\n      throw new Error(`Potentially insecure feature '${name}' has not been ` +\n                      `enabled. If you want to enable this feature and accept ` +\n                      `the security ramifications, please do so by following ` +\n                      `the documented instructions at https://github.com/appium` +\n                      `/appium/blob/master/docs/en/writing-running-appium/security.md`);\n    }\n  }\n\n  // This is the main command handler for the driver. It wraps command\n  // execution with timeout logic, checking that we have a valid session,\n  // and ensuring that we execute commands one at a time. This method is called\n  // by MJSONWP's express router.\n  async executeCommand (cmd, ...args) {\n    // get start time for this command, and log in special cases\n    let startTime = Date.now();\n\n    if (cmd === 'createSession') {\n      // If creating a session determine if W3C or MJSONWP protocol was requested and remember the choice\n      this.protocol = determineProtocol(...args);\n      this.logEvent(EVENT_SESSION_INIT);\n    } else if (cmd === 'deleteSession') {\n      this.logEvent(EVENT_SESSION_QUIT_START);\n    }\n\n    // if we had a command timer running, clear it now that we're starting\n    // a new command and so don't want to time out\n    this.clearNewCommandTimeout();\n\n    if (this.shutdownUnexpectedly) {\n      throw new errors.NoSuchDriverError('The driver was unexpectedly shut down!');\n    }\n\n    // If we don't have this command, it must not be implemented\n    if (!this[cmd]) {\n      throw new errors.NotYetImplementedError();\n    }\n\n    let unexpectedShutdownListener;\n    const commandExecutor = async () => await B.race([\n      this[cmd](...args),\n      new B((resolve, reject) => {\n        unexpectedShutdownListener = reject;\n        this.eventEmitter.on(ON_UNEXPECTED_SHUTDOWN_EVENT, unexpectedShutdownListener);\n      })\n    ]).finally(() => {\n      if (unexpectedShutdownListener) {\n        // This is needed to prevent memory leaks\n        this.eventEmitter.removeListener(ON_UNEXPECTED_SHUTDOWN_EVENT, unexpectedShutdownListener);\n        unexpectedShutdownListener = null;\n      }\n    });\n    const res = this.isCommandsQueueEnabled\n      ? await this.commandsQueueGuard.acquire(BaseDriver.name, commandExecutor)\n      : await commandExecutor();\n\n    // if we have set a new command timeout (which is the default), start a\n    // timer once we've finished executing this command. If we don't clear\n    // the timer (which is done when a new command comes in), we will trigger\n    // automatic session deletion in this.onCommandTimeout. Of course we don't\n    // want to trigger the timer when the user is shutting down the session\n    // intentionally\n    if (this.isCommandsQueueEnabled && cmd !== 'deleteSession') {\n      // resetting existing timeout\n      this.startNewCommandTimeout();\n    }\n\n    // log timing information about this command\n    const endTime = Date.now();\n    this._eventHistory.commands.push({cmd, startTime, endTime});\n    if (cmd === 'createSession') {\n      this.logEvent(EVENT_SESSION_START);\n    } else if (cmd === 'deleteSession') {\n      this.logEvent(EVENT_SESSION_QUIT_DONE);\n    }\n\n    return res;\n  }\n\n  async startUnexpectedShutdown (err = new errors.NoSuchDriverError('The driver was unexpectedly shut down!')) {\n    this.eventEmitter.emit(ON_UNEXPECTED_SHUTDOWN_EVENT, err); // allow others to listen for this\n    this.shutdownUnexpectedly = true;\n    try {\n      await this.deleteSession(this.sessionId);\n    } finally {\n      this.shutdownUnexpectedly = false;\n    }\n  }\n\n  validateLocatorStrategy (strategy, webContext = false) {\n    let validStrategies = this.locatorStrategies;\n    log.debug(`Valid locator strategies for this request: ${validStrategies.join(', ')}`);\n\n    if (webContext) {\n      validStrategies = validStrategies.concat(this.webLocatorStrategies);\n    }\n\n    if (!_.includes(validStrategies, strategy)) {\n      throw new errors.InvalidSelectorError(`Locator Strategy '${strategy}' is not supported for this session`);\n    }\n  }\n\n  /*\n   * Restart the session with the original caps,\n   * preserving the timeout config.\n   */\n  async reset () {\n    log.debug('Resetting app mid-session');\n    log.debug('Running generic full reset');\n\n    // preserving state\n    let currentConfig = {};\n    for (let property of ['implicitWaitMs', 'newCommandTimeoutMs', 'sessionId', 'resetOnUnexpectedShutdown']) {\n      currentConfig[property] = this[property];\n    }\n\n    // We also need to preserve the unexpected shutdown, and make sure it is not cancelled during reset.\n    this.resetOnUnexpectedShutdown = () => {};\n\n    try {\n      await this.deleteSession(this.sessionId);\n      log.debug('Restarting app');\n      await this.createSession(undefined, undefined, this.originalCaps);\n    } finally {\n      // always restore state.\n      for (let [key, value] of _.toPairs(currentConfig)) {\n        this[key] = value;\n      }\n    }\n    this.clearNewCommandTimeout();\n  }\n\n  proxyActive (/* sessionId */) {\n    return false;\n  }\n\n  getProxyAvoidList (/* sessionId */) {\n    return [];\n  }\n\n  canProxy (/* sessionId */) {\n    return false;\n  }\n\n  /**\n   * Whether a given command route (expressed as method and url) should not be\n   * proxied according to this driver\n   *\n   * @param {string} sessionId - the current sessionId (in case the driver runs\n   * multiple session ids and requires it). This is not used in this method but\n   * should be made available to overridden methods.\n   * @param {string} method - HTTP method of the route\n   * @param {string} url - url of the route\n   * @param {?*} body - webdriver request body\n   *\n   * @returns {boolean} - whether the route should be avoided\n   */\n  proxyRouteIsAvoided (sessionId, method, url/*, body*/) {\n    for (let avoidSchema of this.getProxyAvoidList(sessionId)) {\n      if (!_.isArray(avoidSchema) || avoidSchema.length !== 2) {\n        throw new Error('Proxy avoidance must be a list of pairs');\n      }\n      let [avoidMethod, avoidPathRegex] = avoidSchema;\n      if (!_.includes(['GET', 'POST', 'DELETE'], avoidMethod)) {\n        throw new Error(`Unrecognized proxy avoidance method '${avoidMethod}'`);\n      }\n      if (!_.isRegExp(avoidPathRegex)) {\n        throw new Error('Proxy avoidance path must be a regular expression');\n      }\n      let normalizedUrl = url.replace(new RegExp(`^${_.escapeRegExp(this.basePath)}`), '');\n      if (avoidMethod === method && avoidPathRegex.test(normalizedUrl)) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  addManagedDriver (driver) {\n    this.managedDrivers.push(driver);\n  }\n\n  getManagedDrivers () {\n    return this.managedDrivers;\n  }\n}\n\nfor (let [cmd, fn] of _.toPairs(commands)) {\n  BaseDriver.prototype[cmd] = fn;\n}\n\nexport { BaseDriver };\nexport default BaseDriver;\n"],"file":"lib/basedriver/driver.js","sourceRoot":"../../.."}
393
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/basedriver/driver.js"],"names":["version","BASEDRIVER_VER","fs","readPackageJsonFrom","__dirname","B","config","cancellation","NEW_COMMAND_TIMEOUT_MS","EVENT_SESSION_INIT","EVENT_SESSION_START","EVENT_SESSION_QUIT_START","EVENT_SESSION_QUIT_DONE","ON_UNEXPECTED_SHUTDOWN_EVENT","BaseDriver","Protocol","baseVersion","constructor","opts","shouldValidateCaps","sessionId","caps","originalCaps","helpers","_log","basePath","DEFAULT_BASE_PATH","relaxedSecurityEnabled","allowInsecure","denyInsecure","newCommandTimeoutMs","implicitWaitMs","_constraints","_","cloneDeep","desiredCapabilityConstraints","locatorStrategies","webLocatorStrategies","tmpDir","process","env","APPIUM_TMP_DIR","os","tmpdir","shutdownUnexpectedly","noCommandTimer","commandsQueueGuard","AsyncLock","settings","DeviceSettings","noop","initialOpts","managedDrivers","_eventHistory","commands","eventEmitter","EventEmitter","protocol","log","instanceName","name","node","getObjectId","substring","logger","getLogger","onUnexpectedShutdown","handler","on","driverData","isCommandsQueueEnabled","eventHistory","logEvent","eventName","Error","ts","Date","now","logTime","toTimeString","push","debug","getStatus","desiredCapConstraints","constraints","Object","assign","value","toPairs","presence","allowEmpty","sessionExists","driverForSession","logExtraCaps","extraCaps","difference","keys","length","warn","cap","validateDesiredCaps","e","errorAndThrow","errors","SessionNotCreatedError","message","isMjsonwpProtocol","PROTOCOLS","MJSONWP","isW3CProtocol","W3C","setProtocolMJSONWP","setProtocolW3C","isFeatureEnabled","includes","ensureFeatureEnabled","executeCommand","cmd","args","startTime","DELETE_SESSION_COMMAND","clearNewCommandTimeout","NoSuchDriverError","NotYetImplementedError","unexpectedShutdownListener","commandExecutor","race","resolve","reject","finally","removeListener","res","acquire","startNewCommandTimeout","endTime","startUnexpectedShutdown","err","emit","deleteSession","validateLocatorStrategy","strategy","webContext","validStrategies","join","concat","InvalidSelectorError","reset","currentConfig","property","resetOnUnexpectedShutdown","createSession","undefined","key","proxyActive","getProxyAvoidList","canProxy","proxyRouteIsAvoided","method","url","avoidSchema","isArray","avoidMethod","avoidPathRegex","isRegExp","normalizedUrl","replace","RegExp","escapeRegExp","test","addManagedDriver","driver","getManagedDrivers","fn","prototype"],"mappings":"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","sourcesContent":["import {\n  Protocol, errors, determineProtocol, DELETE_SESSION_COMMAND,\n} from '../protocol';\nimport { fs, logger, node } from '@appium/support';\nimport { PROTOCOLS, DEFAULT_BASE_PATH } from '../constants';\nimport os from 'os';\nimport commands from './commands';\nimport * as helpers from './helpers';\nimport DeviceSettings from './device-settings';\nimport { desiredCapabilityConstraints } from './desired-caps';\nimport { validateCaps } from './capabilities';\nimport B from 'bluebird';\nimport _ from 'lodash';\nimport AsyncLock from 'async-lock';\nimport { EventEmitter } from 'events';\n\n// for compat with running tests transpiled and in-place\nconst {version: BASEDRIVER_VER} = fs.readPackageJsonFrom(__dirname);\n\nB.config({\n  cancellation: true,\n});\n\nconst NEW_COMMAND_TIMEOUT_MS = 60 * 1000;\n\nconst EVENT_SESSION_INIT = 'newSessionRequested';\nconst EVENT_SESSION_START = 'newSessionStarted';\nconst EVENT_SESSION_QUIT_START = 'quitSessionRequested';\nconst EVENT_SESSION_QUIT_DONE = 'quitSessionFinished';\nconst ON_UNEXPECTED_SHUTDOWN_EVENT = 'onUnexpectedShutdown';\n\n\nclass BaseDriver extends Protocol {\n\n  /**\n   * Make the basedriver version available so for any driver which inherits from this package, we\n   * know which version of basedriver it inherited from\n   */\n  static baseVersion = BASEDRIVER_VER;\n\n  constructor (opts = {}, shouldValidateCaps = true) {\n    super();\n\n    // setup state\n    this.sessionId = null;\n    this.opts = opts;\n    this.caps = null;\n    this.originalCaps = null; // To give the original capabilities to reset\n    this.helpers = helpers;\n\n    this._log = null;\n\n    // basePath is used for several purposes, for example in setting up\n    // proxying to other drivers, since we need to know what the base path\n    // of any incoming request might look like. We set it to the default\n    // initially but it is automatically updated during any actual program\n    // execution by the routeConfiguringFunction, which is necessarily run as\n    // the entrypoint for any Appium server\n    this.basePath = DEFAULT_BASE_PATH;\n\n    // initialize security modes\n    this.relaxedSecurityEnabled = false;\n    this.allowInsecure = [];\n    this.denyInsecure = [];\n\n    // timeout initialization\n    this.newCommandTimeoutMs = NEW_COMMAND_TIMEOUT_MS;\n    this.implicitWaitMs = 0;\n\n    this._constraints = _.cloneDeep(desiredCapabilityConstraints);\n    this.locatorStrategies = [];\n    this.webLocatorStrategies = [];\n\n    // use a custom tmp dir to avoid losing data and app when computer is\n    // restarted\n    this.opts.tmpDir = this.opts.tmpDir ||\n                       process.env.APPIUM_TMP_DIR ||\n                       os.tmpdir();\n\n    // base-driver internals\n    this.shutdownUnexpectedly = false;\n    this.noCommandTimer = null;\n    this.shouldValidateCaps = shouldValidateCaps;\n    this.commandsQueueGuard = new AsyncLock();\n\n    // settings should be instantiated by drivers which extend BaseDriver, but\n    // we set it to an empty DeviceSettings instance here to make sure that the\n    // default settings are applied even if an extending driver doesn't utilize\n    // the settings functionality itself\n    this.settings = new DeviceSettings({}, _.noop);\n\n    // keeping track of initial opts\n    this.initialOpts = _.cloneDeep(this.opts);\n\n    // allow subclasses to have internal drivers\n    this.managedDrivers = [];\n\n    // store event timings\n    this._eventHistory = {\n      commands: [] // commands get a special place\n    };\n\n    // used to handle driver events\n    this.eventEmitter = new EventEmitter();\n\n    this.protocol = null;\n  }\n\n  get log () {\n    if (!this._log) {\n      const instanceName = `${this.constructor.name}@${node.getObjectId(this).substring(0, 8)}`;\n      this._log = logger.getLogger(() =>\n        this.sessionId ? `${instanceName} (${this.sessionId.substring(0, 8)})` : instanceName\n      );\n    }\n\n    return this._log;\n  }\n\n  /**\n   * Set a callback handler if needed to execute a custom piece of code\n   * when the driver is shut down unexpectedly. Multiple calls to this method\n   * will cause the handler to be executed mutiple times\n   *\n   * @param {Function} handler The code to be executed on unexpected shutdown.\n   * The function may accept one argument, which is the actual error instance, which\n   * caused the driver to shut down.\n   */\n  onUnexpectedShutdown (handler) {\n    this.eventEmitter.on(ON_UNEXPECTED_SHUTDOWN_EVENT, handler);\n  }\n\n  /**\n   * This property is used by AppiumDriver to store the data of the\n   * specific driver sessions. This data can be later used to adjust\n   * properties for driver instances running in parallel.\n   * Override it in inherited driver classes if necessary.\n   *\n   * @return {object} Driver properties mapping\n   */\n  get driverData () {\n    return {};\n  }\n\n  /**\n   * This property controls the way {#executeCommand} method\n   * handles new driver commands received from the client.\n   * Override it for inherited classes only in special cases.\n   *\n   * @return {boolean} If the returned value is true (default) then all the commands\n   *   received by the particular driver instance are going to be put into the queue,\n   *   so each following command will not be executed until the previous command\n   *   execution is completed. False value disables that queue, so each driver command\n   *   is executed independently and does not wait for anything.\n   */\n  get isCommandsQueueEnabled () {\n    return true;\n  }\n\n  /*\n   * make eventHistory a property and return a cloned object so a consumer can't\n   * inadvertently change data outside of logEvent\n   */\n  get eventHistory () {\n    return _.cloneDeep(this._eventHistory);\n  }\n\n  /*\n   * API method for driver developers to log timings for important events\n   */\n  logEvent (eventName) {\n    if (eventName === 'commands') {\n      throw new Error('Cannot log commands directly');\n    }\n    if (typeof eventName !== 'string') {\n      throw new Error(`Invalid eventName ${eventName}`);\n    }\n    if (!this._eventHistory[eventName]) {\n      this._eventHistory[eventName] = [];\n    }\n    const ts = Date.now();\n    const logTime = (new Date(ts)).toTimeString();\n    this._eventHistory[eventName].push(ts);\n    this.log.debug(`Event '${eventName}' logged at ${ts} (${logTime})`);\n  }\n\n  /*\n   * Overridden in appium driver, but here so that individual drivers can be\n   * tested with clients that poll\n   */\n  async getStatus () { // eslint-disable-line require-await\n    return {};\n  }\n\n  // we only want subclasses to ever extend the contraints\n  set desiredCapConstraints (constraints) {\n    this._constraints = Object.assign(this._constraints, constraints);\n    // 'presence' means different things in different versions of the validator,\n    // when we say 'true' we mean that it should not be able to be empty\n    for (const [, value] of _.toPairs(this._constraints)) {\n      if (value && value.presence === true) {\n        value.presence = {\n          allowEmpty: false,\n        };\n      }\n    }\n  }\n\n  get desiredCapConstraints () {\n    return this._constraints;\n  }\n\n  // method required by MJSONWP in order to determine whether it should\n  // respond with an invalid session response\n  sessionExists (sessionId) {\n    if (!sessionId) return false; // eslint-disable-line curly\n    return sessionId === this.sessionId;\n  }\n\n  // method required by MJSONWP in order to determine if the command should\n  // be proxied directly to the driver\n  driverForSession (/*sessionId*/) {\n    return this;\n  }\n\n  logExtraCaps (caps) {\n    let extraCaps = _.difference(_.keys(caps),\n                                 _.keys(this._constraints));\n    if (extraCaps.length) {\n      this.log.warn(`The following capabilities were provided, but are not ` +\n               `recognized by Appium:`);\n      for (const cap of extraCaps) {\n        this.log.warn(`  ${cap}`);\n      }\n    }\n  }\n\n  validateDesiredCaps (caps) {\n    if (!this.shouldValidateCaps) {\n      return true;\n    }\n\n    try {\n      validateCaps(caps, this._constraints);\n    } catch (e) {\n      this.log.errorAndThrow(new errors.SessionNotCreatedError(`The desiredCapabilities object was not valid for the ` +\n        `following reason(s): ${e.message}`));\n    }\n\n    this.logExtraCaps(caps);\n\n    return true;\n  }\n\n  isMjsonwpProtocol () {\n    return this.protocol === PROTOCOLS.MJSONWP;\n  }\n\n  isW3CProtocol () {\n    return this.protocol === PROTOCOLS.W3C;\n  }\n\n  setProtocolMJSONWP () {\n    this.protocol = PROTOCOLS.MJSONWP;\n  }\n\n  setProtocolW3C () {\n    this.protocol = PROTOCOLS.W3C;\n  }\n\n  /**\n   * Check whether a given feature is enabled via its name\n   *\n   * @param {string} name - name of feature/command\n   *\n   * @returns {Boolean}\n   */\n  isFeatureEnabled (name) {\n    // if we have explicitly denied this feature, return false immediately\n    if (this.denyInsecure && _.includes(this.denyInsecure, name)) {\n      return false;\n    }\n\n    // if we specifically have allowed the feature, return true\n    if (this.allowInsecure && _.includes(this.allowInsecure, name)) {\n      return true;\n    }\n\n    // otherwise, if we've globally allowed insecure features and not denied\n    // this one, return true\n    if (this.relaxedSecurityEnabled) {\n      return true;\n    }\n\n    // if we haven't allowed anything insecure, then reject\n    return false;\n  }\n\n  /**\n   * Assert that a given feature is enabled and throw a helpful error if it's\n   * not\n   *\n   * @param {string} name - name of feature/command\n   */\n  ensureFeatureEnabled (name) {\n    if (!this.isFeatureEnabled(name)) {\n      throw new Error(`Potentially insecure feature '${name}' has not been ` +\n                      `enabled. If you want to enable this feature and accept ` +\n                      `the security ramifications, please do so by following ` +\n                      `the documented instructions at https://github.com/appium` +\n                      `/appium/blob/master/docs/en/writing-running-appium/security.md`);\n    }\n  }\n\n  // This is the main command handler for the driver. It wraps command\n  // execution with timeout logic, checking that we have a valid session,\n  // and ensuring that we execute commands one at a time. This method is called\n  // by MJSONWP's express router.\n  async executeCommand (cmd, ...args) {\n    // get start time for this command, and log in special cases\n    let startTime = Date.now();\n\n    if (cmd === 'createSession') {\n      // If creating a session determine if W3C or MJSONWP protocol was requested and remember the choice\n      this.protocol = determineProtocol(args);\n      this.logEvent(EVENT_SESSION_INIT);\n    } else if (cmd === DELETE_SESSION_COMMAND) {\n      this.logEvent(EVENT_SESSION_QUIT_START);\n    }\n\n    // if we had a command timer running, clear it now that we're starting\n    // a new command and so don't want to time out\n    this.clearNewCommandTimeout();\n\n    if (this.shutdownUnexpectedly) {\n      throw new errors.NoSuchDriverError('The driver was unexpectedly shut down!');\n    }\n\n    // If we don't have this command, it must not be implemented\n    if (!this[cmd]) {\n      throw new errors.NotYetImplementedError();\n    }\n\n    let unexpectedShutdownListener;\n    const commandExecutor = async () => await B.race([\n      this[cmd](...args),\n      new B((resolve, reject) => {\n        unexpectedShutdownListener = reject;\n        this.eventEmitter.on(ON_UNEXPECTED_SHUTDOWN_EVENT, unexpectedShutdownListener);\n      })\n    ]).finally(() => {\n      if (unexpectedShutdownListener) {\n        // This is needed to prevent memory leaks\n        this.eventEmitter.removeListener(ON_UNEXPECTED_SHUTDOWN_EVENT, unexpectedShutdownListener);\n        unexpectedShutdownListener = null;\n      }\n    });\n    const res = this.isCommandsQueueEnabled\n      ? await this.commandsQueueGuard.acquire(BaseDriver.name, commandExecutor)\n      : await commandExecutor();\n\n    // if we have set a new command timeout (which is the default), start a\n    // timer once we've finished executing this command. If we don't clear\n    // the timer (which is done when a new command comes in), we will trigger\n    // automatic session deletion in this.onCommandTimeout. Of course we don't\n    // want to trigger the timer when the user is shutting down the session\n    // intentionally\n    if (this.isCommandsQueueEnabled && cmd !== DELETE_SESSION_COMMAND) {\n      // resetting existing timeout\n      this.startNewCommandTimeout();\n    }\n\n    // log timing information about this command\n    const endTime = Date.now();\n    this._eventHistory.commands.push({cmd, startTime, endTime});\n    if (cmd === 'createSession') {\n      this.logEvent(EVENT_SESSION_START);\n    } else if (cmd === DELETE_SESSION_COMMAND) {\n      this.logEvent(EVENT_SESSION_QUIT_DONE);\n    }\n\n    return res;\n  }\n\n  async startUnexpectedShutdown (err = new errors.NoSuchDriverError('The driver was unexpectedly shut down!')) {\n    this.eventEmitter.emit(ON_UNEXPECTED_SHUTDOWN_EVENT, err); // allow others to listen for this\n    this.shutdownUnexpectedly = true;\n    try {\n      await this.deleteSession(this.sessionId);\n    } finally {\n      this.shutdownUnexpectedly = false;\n    }\n  }\n\n  validateLocatorStrategy (strategy, webContext = false) {\n    let validStrategies = this.locatorStrategies;\n    this.log.debug(`Valid locator strategies for this request: ${validStrategies.join(', ')}`);\n\n    if (webContext) {\n      validStrategies = validStrategies.concat(this.webLocatorStrategies);\n    }\n\n    if (!_.includes(validStrategies, strategy)) {\n      throw new errors.InvalidSelectorError(`Locator Strategy '${strategy}' is not supported for this session`);\n    }\n  }\n\n  /*\n   * Restart the session with the original caps,\n   * preserving the timeout config.\n   */\n  async reset () {\n    this.log.debug('Resetting app mid-session');\n    this.log.debug('Running generic full reset');\n\n    // preserving state\n    let currentConfig = {};\n    for (let property of ['implicitWaitMs', 'newCommandTimeoutMs', 'sessionId', 'resetOnUnexpectedShutdown']) {\n      currentConfig[property] = this[property];\n    }\n\n    // We also need to preserve the unexpected shutdown, and make sure it is not cancelled during reset.\n    this.resetOnUnexpectedShutdown = () => {};\n\n    try {\n      await this.deleteSession(this.sessionId);\n      this.log.debug('Restarting app');\n      await this.createSession(undefined, undefined, this.originalCaps);\n    } finally {\n      // always restore state.\n      for (let [key, value] of _.toPairs(currentConfig)) {\n        this[key] = value;\n      }\n    }\n    this.clearNewCommandTimeout();\n  }\n\n  proxyActive (/* sessionId */) {\n    return false;\n  }\n\n  getProxyAvoidList (/* sessionId */) {\n    return [];\n  }\n\n  canProxy (/* sessionId */) {\n    return false;\n  }\n\n  /**\n   * Whether a given command route (expressed as method and url) should not be\n   * proxied according to this driver\n   *\n   * @param {string} sessionId - the current sessionId (in case the driver runs\n   * multiple session ids and requires it). This is not used in this method but\n   * should be made available to overridden methods.\n   * @param {string} method - HTTP method of the route\n   * @param {string} url - url of the route\n   * @param {?*} body - webdriver request body\n   *\n   * @returns {boolean} - whether the route should be avoided\n   */\n  proxyRouteIsAvoided (sessionId, method, url/*, body*/) {\n    for (let avoidSchema of this.getProxyAvoidList(sessionId)) {\n      if (!_.isArray(avoidSchema) || avoidSchema.length !== 2) {\n        throw new Error('Proxy avoidance must be a list of pairs');\n      }\n      let [avoidMethod, avoidPathRegex] = avoidSchema;\n      if (!_.includes(['GET', 'POST', 'DELETE'], avoidMethod)) {\n        throw new Error(`Unrecognized proxy avoidance method '${avoidMethod}'`);\n      }\n      if (!_.isRegExp(avoidPathRegex)) {\n        throw new Error('Proxy avoidance path must be a regular expression');\n      }\n      let normalizedUrl = url.replace(new RegExp(`^${_.escapeRegExp(this.basePath)}`), '');\n      if (avoidMethod === method && avoidPathRegex.test(normalizedUrl)) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  addManagedDriver (driver) {\n    this.managedDrivers.push(driver);\n  }\n\n  getManagedDrivers () {\n    return this.managedDrivers;\n  }\n}\n\nfor (let [cmd, fn] of _.toPairs(commands)) {\n  BaseDriver.prototype[cmd] = fn;\n}\n\nexport { BaseDriver };\nexport default BaseDriver;\n"],"file":"lib/basedriver/driver.js","sourceRoot":"../../.."}