@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
@@ -5,7 +5,6 @@ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefau
5
5
  Object.defineProperty(exports, "__esModule", {
6
6
  value: true
7
7
  });
8
- exports.JSONWP_SUCCESS_STATUS_CODE = void 0;
9
8
  Object.defineProperty(exports, "MJSONWP_ELEMENT_KEY", {
10
9
  enumerable: true,
11
10
  get: function () {
@@ -29,10 +28,6 @@ var _helpers = require("../basedriver/helpers");
29
28
 
30
29
  var _constants = require("../constants");
31
30
 
32
- const JSONWP_SUCCESS_STATUS_CODE = 0;
33
- exports.JSONWP_SUCCESS_STATUS_CODE = JSONWP_SUCCESS_STATUS_CODE;
34
- const JSONWP_UNKNOWN_ERROR_STATUS_CODE = 13;
35
-
36
31
  function formatResponseValue(resValue) {
37
32
  if (_lodash.default.isUndefined(resValue)) {
38
33
  return null;
@@ -41,21 +36,9 @@ function formatResponseValue(resValue) {
41
36
  return (0, _helpers.duplicateKeys)(resValue, _constants.MJSONWP_ELEMENT_KEY, _constants.W3C_ELEMENT_KEY);
42
37
  }
43
38
 
44
- function formatStatus(responseBody, responseCode = 200, protocol = null) {
45
- if (!_lodash.default.isPlainObject(responseBody)) {
46
- return responseBody;
47
- }
48
-
49
- const isError = _lodash.default.has(responseBody.value, 'error') || responseCode >= 400;
50
-
51
- if (protocol === _constants.PROTOCOLS.MJSONWP && !_lodash.default.isInteger(responseBody.status) || !protocol && !_lodash.default.has(responseBody, 'status')) {
52
- responseBody.status = isError ? JSONWP_UNKNOWN_ERROR_STATUS_CODE : JSONWP_SUCCESS_STATUS_CODE;
53
- } else if (protocol === _constants.PROTOCOLS.W3C && _lodash.default.has(responseBody, 'status')) {
54
- delete responseBody.status;
55
- }
56
-
57
- return responseBody;
39
+ function formatStatus(responseBody) {
40
+ return _lodash.default.isPlainObject(responseBody) ? _lodash.default.omit(responseBody, ['status']) : responseBody;
58
41
  }require('source-map-support').install();
59
42
 
60
43
 
61
- //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9wcm90b2NvbC9oZWxwZXJzLmpzIl0sIm5hbWVzIjpbIkpTT05XUF9TVUNDRVNTX1NUQVRVU19DT0RFIiwiSlNPTldQX1VOS05PV05fRVJST1JfU1RBVFVTX0NPREUiLCJmb3JtYXRSZXNwb25zZVZhbHVlIiwicmVzVmFsdWUiLCJfIiwiaXNVbmRlZmluZWQiLCJNSlNPTldQX0VMRU1FTlRfS0VZIiwiVzNDX0VMRU1FTlRfS0VZIiwiZm9ybWF0U3RhdHVzIiwicmVzcG9uc2VCb2R5IiwicmVzcG9uc2VDb2RlIiwicHJvdG9jb2wiLCJpc1BsYWluT2JqZWN0IiwiaXNFcnJvciIsImhhcyIsInZhbHVlIiwiUFJPVE9DT0xTIiwiTUpTT05XUCIsImlzSW50ZWdlciIsInN0YXR1cyIsIlczQyJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBOztBQUNBOztBQUNBOztBQUVBLE1BQU1BLDBCQUEwQixHQUFHLENBQW5DOztBQUNBLE1BQU1DLGdDQUFnQyxHQUFHLEVBQXpDOztBQVdBLFNBQVNDLG1CQUFULENBQThCQyxRQUE5QixFQUF3QztBQUN0QyxNQUFJQyxnQkFBRUMsV0FBRixDQUFjRixRQUFkLENBQUosRUFBNkI7QUFFM0IsV0FBTyxJQUFQO0FBQ0Q7O0FBR0QsU0FBTyw0QkFBY0EsUUFBZCxFQUF3QkcsOEJBQXhCLEVBQTZDQywwQkFBN0MsQ0FBUDtBQUNEOztBQWFELFNBQVNDLFlBQVQsQ0FBdUJDLFlBQXZCLEVBQXFDQyxZQUFZLEdBQUcsR0FBcEQsRUFBeURDLFFBQVEsR0FBRyxJQUFwRSxFQUEwRTtBQUN4RSxNQUFJLENBQUNQLGdCQUFFUSxhQUFGLENBQWdCSCxZQUFoQixDQUFMLEVBQW9DO0FBQ2xDLFdBQU9BLFlBQVA7QUFDRDs7QUFDRCxRQUFNSSxPQUFPLEdBQUdULGdCQUFFVSxHQUFGLENBQU1MLFlBQVksQ0FBQ00sS0FBbkIsRUFBMEIsT0FBMUIsS0FBc0NMLFlBQVksSUFBSSxHQUF0RTs7QUFDQSxNQUFLQyxRQUFRLEtBQUtLLHFCQUFVQyxPQUF2QixJQUFrQyxDQUFDYixnQkFBRWMsU0FBRixDQUFZVCxZQUFZLENBQUNVLE1BQXpCLENBQXBDLElBQ0UsQ0FBQ1IsUUFBRCxJQUFhLENBQUNQLGdCQUFFVSxHQUFGLENBQU1MLFlBQU4sRUFBb0IsUUFBcEIsQ0FEcEIsRUFDb0Q7QUFDbERBLElBQUFBLFlBQVksQ0FBQ1UsTUFBYixHQUFzQk4sT0FBTyxHQUN6QlosZ0NBRHlCLEdBRXpCRCwwQkFGSjtBQUdELEdBTEQsTUFLTyxJQUFJVyxRQUFRLEtBQUtLLHFCQUFVSSxHQUF2QixJQUE4QmhCLGdCQUFFVSxHQUFGLENBQU1MLFlBQU4sRUFBb0IsUUFBcEIsQ0FBbEMsRUFBaUU7QUFDdEUsV0FBT0EsWUFBWSxDQUFDVSxNQUFwQjtBQUNEOztBQUNELFNBQU9WLFlBQVA7QUFDRCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgeyBkdXBsaWNhdGVLZXlzIH0gZnJvbSAnLi4vYmFzZWRyaXZlci9oZWxwZXJzJztcbmltcG9ydCB7IE1KU09OV1BfRUxFTUVOVF9LRVksIFczQ19FTEVNRU5UX0tFWSwgUFJPVE9DT0xTIH0gZnJvbSAnLi4vY29uc3RhbnRzJztcblxuY29uc3QgSlNPTldQX1NVQ0NFU1NfU1RBVFVTX0NPREUgPSAwO1xuY29uc3QgSlNPTldQX1VOS05PV05fRVJST1JfU1RBVFVTX0NPREUgPSAxMztcblxuLyoqXG4gKiBQcmVwcm9jZXNzZXMgdGhlIHJlc3VsdGluZyB2YWx1ZSBmb3IgQVBJIHJlc3BvbnNlcyxcbiAqIHNvIHRoZXkgaGF2ZSBrZXlzIGZvciBib3RoIFczQyBhbmQgSlNPTldQIHByb3RvY29scy5cbiAqIFRoZSBhcmd1bWVudCB2YWx1ZSBpcyBOT1QgbXV0YXRlZFxuICpcbiAqIEBwYXJhbSB7P09iamVjdH0gcmVzVmFsdWUgVGhlIGFjdHVhbCByZXNwb25zZSB2YWx1ZVxuICogQHJldHVybnMgez9PYmplY3R9IEVpdGhlciBtb2RpZmllZCB2YWx1ZSBvciB0aGUgc2FtZSBvbmUgaWZcbiAqIG5vdGhpbmcgaGFzIGJlZW4gbW9kaWZpZWRcbiAqL1xuZnVuY3Rpb24gZm9ybWF0UmVzcG9uc2VWYWx1ZSAocmVzVmFsdWUpIHtcbiAgaWYgKF8uaXNVbmRlZmluZWQocmVzVmFsdWUpKSB7XG4gICAgLy8gY29udmVydCB1bmRlZmluZWQgdG8gbnVsbFxuICAgIHJldHVybiBudWxsO1xuICB9XG4gIC8vIElmIHRoZSBNSlNPTldQIGVsZW1lbnQga2V5IGZvcm1hdCAoRUxFTUVOVCkgd2FzIHByb3ZpZGVkLCBhZGQgYSBkdXBsaWNhdGUga2V5IChlbGVtZW50LTYwNjYtMTFlNC1hNTJlLTRmNzM1NDY2Y2VjZilcbiAgLy8gSWYgdGhlIFczQyBlbGVtZW50IGtleSBmb3JtYXQgKGVsZW1lbnQtNjA2Ni0xMWU0LWE1MmUtNGY3MzU0NjZjZWNmKSB3YXMgcHJvdmlkZWQsIGFkZCBhIGR1cGxpY2F0ZSBrZXkgKEVMRU1FTlQpXG4gIHJldHVybiBkdXBsaWNhdGVLZXlzKHJlc1ZhbHVlLCBNSlNPTldQX0VMRU1FTlRfS0VZLCBXM0NfRUxFTUVOVF9LRVkpO1xufVxuXG4vKipcbiAqIFByb3Blcmx5IGZvcm1hdHMgdGhlIHN0YXR1cyBmb3IgQVBJIHJlc3BvbnNlcyxcbiAqIHNvIHRoZXkgYXJlIGNvcnJlY3QgZm9yIGJvdGggVzNDIGFuZCBKU09OV1AgcHJvdG9jb2xzLlxuICogVGhpcyBtZXRob2QgRE9FUyBtdXRhdGUgdGhlIGByZXNwb25zZUJvZHlgIGFyZ3VtZW50IGlmIG5lZWRlZFxuICpcbiAqIEBwYXJhbSB7T2JqZWN0fSByZXNwb25zZUJvZHlcbiAqIEBwYXJhbSB7bnVtYmVyfSByZXNwb25zZUNvZGUgdGhlIEhUVFAgcmVzcG9uc2UgY29kZVxuICogQHBhcmFtIHs/c3RyaW5nfSBwcm90b2NvbCBUaGUgbmFtZSBvZiB0aGUgcHJvdG9jb2wsIGVpdGhlclxuICogYFBST1RPQ09MUy5XM0NgIG9yIGBQUk9UT0NPTFMuTUpTT05XUGBcbiAqIEByZXR1cm5zIHtPYmplY3R9IFRoZSBmaXhlZCByZXNwb25zZSBib2R5XG4gKi9cbmZ1bmN0aW9uIGZvcm1hdFN0YXR1cyAocmVzcG9uc2VCb2R5LCByZXNwb25zZUNvZGUgPSAyMDAsIHByb3RvY29sID0gbnVsbCkge1xuICBpZiAoIV8uaXNQbGFpbk9iamVjdChyZXNwb25zZUJvZHkpKSB7XG4gICAgcmV0dXJuIHJlc3BvbnNlQm9keTtcbiAgfVxuICBjb25zdCBpc0Vycm9yID0gXy5oYXMocmVzcG9uc2VCb2R5LnZhbHVlLCAnZXJyb3InKSB8fCByZXNwb25zZUNvZGUgPj0gNDAwO1xuICBpZiAoKHByb3RvY29sID09PSBQUk9UT0NPTFMuTUpTT05XUCAmJiAhXy5pc0ludGVnZXIocmVzcG9uc2VCb2R5LnN0YXR1cykpXG4gICAgfHwgKCFwcm90b2NvbCAmJiAhXy5oYXMocmVzcG9uc2VCb2R5LCAnc3RhdHVzJykpKSB7XG4gICAgcmVzcG9uc2VCb2R5LnN0YXR1cyA9IGlzRXJyb3JcbiAgICAgID8gSlNPTldQX1VOS05PV05fRVJST1JfU1RBVFVTX0NPREVcbiAgICAgIDogSlNPTldQX1NVQ0NFU1NfU1RBVFVTX0NPREU7XG4gIH0gZWxzZSBpZiAocHJvdG9jb2wgPT09IFBST1RPQ09MUy5XM0MgJiYgXy5oYXMocmVzcG9uc2VCb2R5LCAnc3RhdHVzJykpIHtcbiAgICBkZWxldGUgcmVzcG9uc2VCb2R5LnN0YXR1cztcbiAgfVxuICByZXR1cm4gcmVzcG9uc2VCb2R5O1xufVxuXG5cbmV4cG9ydCB7XG4gIE1KU09OV1BfRUxFTUVOVF9LRVksIFczQ19FTEVNRU5UX0tFWSwgZm9ybWF0UmVzcG9uc2VWYWx1ZSxcbiAgSlNPTldQX1NVQ0NFU1NfU1RBVFVTX0NPREUsIGZvcm1hdFN0YXR1cyxcbn07XG4iXSwiZmlsZSI6ImxpYi9wcm90b2NvbC9oZWxwZXJzLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uIn0=
44
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9wcm90b2NvbC9oZWxwZXJzLmpzIl0sIm5hbWVzIjpbImZvcm1hdFJlc3BvbnNlVmFsdWUiLCJyZXNWYWx1ZSIsIl8iLCJpc1VuZGVmaW5lZCIsIk1KU09OV1BfRUxFTUVOVF9LRVkiLCJXM0NfRUxFTUVOVF9LRVkiLCJmb3JtYXRTdGF0dXMiLCJyZXNwb25zZUJvZHkiLCJpc1BsYWluT2JqZWN0Iiwib21pdCJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUE7O0FBQ0E7O0FBQ0E7O0FBV0EsU0FBU0EsbUJBQVQsQ0FBOEJDLFFBQTlCLEVBQXdDO0FBQ3RDLE1BQUlDLGdCQUFFQyxXQUFGLENBQWNGLFFBQWQsQ0FBSixFQUE2QjtBQUUzQixXQUFPLElBQVA7QUFDRDs7QUFHRCxTQUFPLDRCQUFjQSxRQUFkLEVBQXdCRyw4QkFBeEIsRUFBNkNDLDBCQUE3QyxDQUFQO0FBQ0Q7O0FBU0QsU0FBU0MsWUFBVCxDQUF1QkMsWUFBdkIsRUFBcUM7QUFDbkMsU0FBT0wsZ0JBQUVNLGFBQUYsQ0FBZ0JELFlBQWhCLElBQWdDTCxnQkFBRU8sSUFBRixDQUFPRixZQUFQLEVBQXFCLENBQUMsUUFBRCxDQUFyQixDQUFoQyxHQUFtRUEsWUFBMUU7QUFDRCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgeyBkdXBsaWNhdGVLZXlzIH0gZnJvbSAnLi4vYmFzZWRyaXZlci9oZWxwZXJzJztcbmltcG9ydCB7IE1KU09OV1BfRUxFTUVOVF9LRVksIFczQ19FTEVNRU5UX0tFWSB9IGZyb20gJy4uL2NvbnN0YW50cyc7XG5cbi8qKlxuICogUHJlcHJvY2Vzc2VzIHRoZSByZXN1bHRpbmcgdmFsdWUgZm9yIEFQSSByZXNwb25zZXMsXG4gKiBzbyB0aGV5IGhhdmUga2V5cyBmb3IgYm90aCBXM0MgYW5kIEpTT05XUCBwcm90b2NvbHMuXG4gKiBUaGUgYXJndW1lbnQgdmFsdWUgaXMgTk9UIG11dGF0ZWRcbiAqXG4gKiBAcGFyYW0gez9PYmplY3R9IHJlc1ZhbHVlIFRoZSBhY3R1YWwgcmVzcG9uc2UgdmFsdWVcbiAqIEByZXR1cm5zIHs/T2JqZWN0fSBFaXRoZXIgbW9kaWZpZWQgdmFsdWUgb3IgdGhlIHNhbWUgb25lIGlmXG4gKiBub3RoaW5nIGhhcyBiZWVuIG1vZGlmaWVkXG4gKi9cbmZ1bmN0aW9uIGZvcm1hdFJlc3BvbnNlVmFsdWUgKHJlc1ZhbHVlKSB7XG4gIGlmIChfLmlzVW5kZWZpbmVkKHJlc1ZhbHVlKSkge1xuICAgIC8vIGNvbnZlcnQgdW5kZWZpbmVkIHRvIG51bGxcbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuICAvLyBJZiB0aGUgTUpTT05XUCBlbGVtZW50IGtleSBmb3JtYXQgKEVMRU1FTlQpIHdhcyBwcm92aWRlZCwgYWRkIGEgZHVwbGljYXRlIGtleSAoZWxlbWVudC02MDY2LTExZTQtYTUyZS00ZjczNTQ2NmNlY2YpXG4gIC8vIElmIHRoZSBXM0MgZWxlbWVudCBrZXkgZm9ybWF0IChlbGVtZW50LTYwNjYtMTFlNC1hNTJlLTRmNzM1NDY2Y2VjZikgd2FzIHByb3ZpZGVkLCBhZGQgYSBkdXBsaWNhdGUga2V5IChFTEVNRU5UKVxuICByZXR1cm4gZHVwbGljYXRlS2V5cyhyZXNWYWx1ZSwgTUpTT05XUF9FTEVNRU5UX0tFWSwgVzNDX0VMRU1FTlRfS0VZKTtcbn1cblxuLyoqXG4gKiBQcm9wZXJseSBmb3JtYXRzIHRoZSBzdGF0dXMgZm9yIEFQSSByZXNwb25zZXMsXG4gKiBzbyB0aGV5IGFyZSBjb3JyZWN0IGZvciB0aGUgVzNDIHByb3RvY29sLlxuICpcbiAqIEBwYXJhbSB7T2JqZWN0fSByZXNwb25zZUJvZHlcbiAqIEByZXR1cm5zIHtPYmplY3R9IFRoZSBmaXhlZCByZXNwb25zZSBib2R5XG4gKi9cbmZ1bmN0aW9uIGZvcm1hdFN0YXR1cyAocmVzcG9uc2VCb2R5KSB7XG4gIHJldHVybiBfLmlzUGxhaW5PYmplY3QocmVzcG9uc2VCb2R5KSA/IF8ub21pdChyZXNwb25zZUJvZHksIFsnc3RhdHVzJ10pIDogcmVzcG9uc2VCb2R5O1xufVxuXG5cbmV4cG9ydCB7XG4gIE1KU09OV1BfRUxFTUVOVF9LRVksIFczQ19FTEVNRU5UX0tFWSwgZm9ybWF0UmVzcG9uc2VWYWx1ZSwgZm9ybWF0U3RhdHVzXG59O1xuIl0sImZpbGUiOiJsaWIvcHJvdG9jb2wvaGVscGVycy5qcyIsInNvdXJjZVJvb3QiOiIuLi8uLi8uLiJ9
@@ -5,7 +5,7 @@ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefau
5
5
  Object.defineProperty(exports, "__esModule", {
6
6
  value: true
7
7
  });
8
- exports.Protocol = exports.DELETE_SESSION_COMMAND = exports.CREATE_SESSION_COMMAND = void 0;
8
+ exports.Protocol = exports.GET_STATUS_COMMAND = exports.DELETE_SESSION_COMMAND = exports.CREATE_SESSION_COMMAND = void 0;
9
9
  exports.determineProtocol = determineProtocol;
10
10
  exports.driverShouldDoJwpProxy = driverShouldDoJwpProxy;
11
11
  exports.isSessionCommand = isSessionCommand;
@@ -29,35 +29,57 @@ var _helpers = require("./helpers");
29
29
 
30
30
  var _constants = require("../constants");
31
31
 
32
- var _sessionsCache = _interopRequireDefault(require("./sessions-cache"));
32
+ var _capabilities = require("../helpers/capabilities");
33
33
 
34
34
  const CREATE_SESSION_COMMAND = 'createSession';
35
35
  exports.CREATE_SESSION_COMMAND = CREATE_SESSION_COMMAND;
36
36
  const DELETE_SESSION_COMMAND = 'deleteSession';
37
37
  exports.DELETE_SESSION_COMMAND = DELETE_SESSION_COMMAND;
38
+ const GET_STATUS_COMMAND = 'getStatus';
39
+ exports.GET_STATUS_COMMAND = GET_STATUS_COMMAND;
38
40
 
39
41
  class Protocol {}
40
42
 
41
43
  exports.Protocol = Protocol;
42
44
 
43
- function determineProtocol(desiredCapabilities, requiredCapabilities, capabilities) {
44
- return _lodash.default.isPlainObject(capabilities) ? _constants.PROTOCOLS.W3C : _constants.PROTOCOLS.MJSONWP;
45
+ function determineProtocol(createSessionArgs) {
46
+ return _lodash.default.some(createSessionArgs, _capabilities.isW3cCaps) ? _constants.PROTOCOLS.W3C : _constants.PROTOCOLS.MJSONWP;
45
47
  }
46
48
 
47
49
  function extractProtocol(driver, sessionId = null) {
50
+ var _dstDriver$protocol;
51
+
48
52
  const dstDriver = _lodash.default.isFunction(driver.driverForSession) ? driver.driverForSession(sessionId) : driver;
49
53
 
50
54
  if (dstDriver === driver) {
51
55
  return driver.protocol;
52
56
  }
53
57
 
54
- return dstDriver ? dstDriver.protocol : _sessionsCache.default.getProtocol(sessionId);
58
+ return (_dstDriver$protocol = dstDriver === null || dstDriver === void 0 ? void 0 : dstDriver.protocol) !== null && _dstDriver$protocol !== void 0 ? _dstDriver$protocol : _constants.PROTOCOLS.W3C;
55
59
  }
56
60
 
57
61
  function isSessionCommand(command) {
58
62
  return !_lodash.default.includes(_routes.NO_SESSION_ID_COMMANDS, command);
59
63
  }
60
64
 
65
+ function getLogger(driver, sessionId = null) {
66
+ var _driver$driverForSess, _dstDriver$log;
67
+
68
+ const dstDriver = sessionId && _lodash.default.isFunction(driver.driverForSession) ? (_driver$driverForSess = driver.driverForSession(sessionId)) !== null && _driver$driverForSess !== void 0 ? _driver$driverForSess : driver : driver;
69
+
70
+ if (_lodash.default.isFunction((_dstDriver$log = dstDriver.log) === null || _dstDriver$log === void 0 ? void 0 : _dstDriver$log.info)) {
71
+ return dstDriver.log;
72
+ }
73
+
74
+ let logPrefix = dstDriver.constructor ? `${dstDriver.constructor.name}@${_support.node.getObjectId(dstDriver).substring(0, 8)}` : 'AppiumDriver';
75
+
76
+ if (sessionId) {
77
+ logPrefix += ` (${sessionId.substring(0, 8)})`;
78
+ }
79
+
80
+ return _support.logger.getLogger(logPrefix);
81
+ }
82
+
61
83
  function wrapParams(paramSets, jsonObj) {
62
84
  let res = jsonObj;
63
85
 
@@ -164,11 +186,11 @@ function makeArgs(requestParams, jsonObj, payloadParams, protocol) {
164
186
 
165
187
  function routeConfiguringFunction(driver) {
166
188
  if (!driver.sessionExists) {
167
- throw new Error('Drivers used with MJSONWP must implement `sessionExists`');
189
+ throw new Error('Drivers must implement `sessionExists` property');
168
190
  }
169
191
 
170
192
  if (!(driver.executeCommand || driver.execute)) {
171
- throw new Error('Drivers used with MJSONWP must implement `executeCommand` or `execute`');
193
+ throw new Error('Drivers must implement `executeCommand` or `execute` method');
172
194
  }
173
195
 
174
196
  return function addRoutes(app, {
@@ -209,8 +231,7 @@ function buildHandler(app, method, path, spec, driver, isSessCmd) {
209
231
  return;
210
232
  }
211
233
 
212
- _sessionsCache.default.getLogger(req.params.sessionId, currentProtocol).debug(`Would have proxied ` + `command directly, but a plugin exists which might require its value, so will let ` + `its value be collected internally and made part of plugin chain`);
213
-
234
+ getLogger(driver, req.params.sessionId).debug(`Would have proxied ` + `command directly, but a plugin exists which might require its value, so will let ` + `its value be collected internally and made part of plugin chain`);
214
235
  didPluginOverrideProxy = true;
215
236
  }
216
237
 
@@ -227,7 +248,7 @@ function buildHandler(app, method, path, spec, driver, isSessCmd) {
227
248
  }
228
249
 
229
250
  if (spec.command === CREATE_SESSION_COMMAND) {
230
- currentProtocol = determineProtocol(...makeArgs(req.params, jsonObj, spec.payloadParams || {}));
251
+ currentProtocol = determineProtocol(makeArgs(req.params, jsonObj, spec.payloadParams || {}));
231
252
  }
232
253
 
233
254
  checkParams(spec.payloadParams, jsonObj, currentProtocol);
@@ -238,7 +259,7 @@ function buildHandler(app, method, path, spec, driver, isSessCmd) {
238
259
  _validators.validators[spec.command](...args);
239
260
  }
240
261
 
241
- _sessionsCache.default.getLogger(req.params.sessionId, currentProtocol).debug(`Calling ` + `${driver.constructor.name}.${spec.command}() with args: ` + _lodash.default.truncate(JSON.stringify(args), {
262
+ getLogger(driver, req.params.sessionId).debug(`Calling ` + `${driver.constructor.name}.${spec.command}() with args: ` + _lodash.default.truncate(JSON.stringify(args), {
242
263
  length: _constants.MAX_LOG_BODY_LENGTH
243
264
  }));
244
265
 
@@ -263,10 +284,7 @@ function buildHandler(app, method, path, spec, driver, isSessCmd) {
263
284
 
264
285
  if (spec.command === CREATE_SESSION_COMMAND) {
265
286
  newSessionId = driverRes[0];
266
-
267
- _sessionsCache.default.putSession(newSessionId, currentProtocol);
268
-
269
- _sessionsCache.default.getLogger(newSessionId, currentProtocol).debug(`Cached the protocol value '${currentProtocol}' for the new session ${newSessionId}`);
287
+ getLogger(driver, newSessionId).debug(`Cached the protocol value '${currentProtocol}' for the new session ${newSessionId}`);
270
288
 
271
289
  if (currentProtocol === _constants.PROTOCOLS.MJSONWP) {
272
290
  driverRes = driverRes[1];
@@ -280,12 +298,10 @@ function buildHandler(app, method, path, spec, driver, isSessCmd) {
280
298
  driverRes = (0, _helpers.formatResponseValue)(driverRes);
281
299
 
282
300
  if (spec.command === DELETE_SESSION_COMMAND) {
283
- _sessionsCache.default.getLogger(req.params.sessionId, currentProtocol).debug(`Received response: ${_lodash.default.truncate(JSON.stringify(driverRes), {
301
+ getLogger(driver, req.params.sessionId).debug(`Received response: ${_lodash.default.truncate(JSON.stringify(driverRes), {
284
302
  length: _constants.MAX_LOG_BODY_LENGTH
285
303
  })}`);
286
-
287
- _sessionsCache.default.getLogger(req.params.sessionId, currentProtocol).debug('But deleting session, so not returning');
288
-
304
+ getLogger(driver, req.params.sessionId).debug('But deleting session, so not returning');
289
305
  driverRes = null;
290
306
  }
291
307
 
@@ -298,14 +314,9 @@ function buildHandler(app, method, path, spec, driver, isSessCmd) {
298
314
  }
299
315
 
300
316
  httpResBody.value = driverRes;
301
-
302
- _sessionsCache.default.getLogger(req.params.sessionId || newSessionId, currentProtocol).debug(`Responding ` + `to client with driver.${spec.command}() result: ${_lodash.default.truncate(JSON.stringify(driverRes), {
317
+ getLogger(driver, req.params.sessionId || newSessionId).debug(`Responding ` + `to client with driver.${spec.command}() result: ${_lodash.default.truncate(JSON.stringify(driverRes), {
303
318
  length: _constants.MAX_LOG_BODY_LENGTH
304
319
  })}`);
305
-
306
- if (spec.command === DELETE_SESSION_COMMAND) {
307
- _sessionsCache.default.resetLogger(req.params.sessionId);
308
- }
309
320
  } catch (err) {
310
321
  let actualErr = err;
311
322
  currentProtocol = currentProtocol || extractProtocol(driver, req.params.sessionId || newSessionId);
@@ -318,21 +329,10 @@ function buildHandler(app, method, path, spec, driver, isSessCmd) {
318
329
  if ((0, _errors.isErrorType)(err, _errors.errors.ProxyRequestError)) {
319
330
  actualErr = err.getActualError();
320
331
  } else {
321
- _sessionsCache.default.getLogger(req.params.sessionId || newSessionId, currentProtocol).debug(`Encountered internal error running command: ${errMsg}`);
332
+ getLogger(driver, req.params.sessionId || newSessionId).debug(`Encountered internal error running command: ${errMsg}`);
322
333
  }
323
334
 
324
- if (currentProtocol === _constants.PROTOCOLS.W3C) {
325
- [httpStatus, httpResBody] = (0, _errors.getResponseForW3CError)(actualErr);
326
- } else if (currentProtocol === _constants.PROTOCOLS.MJSONWP) {
327
- [httpStatus, httpResBody] = (0, _errors.getResponseForJsonwpError)(actualErr);
328
- } else {
329
- let jsonwpRes = (0, _errors.getResponseForJsonwpError)(actualErr);
330
- let w3cRes = (0, _errors.getResponseForW3CError)(actualErr);
331
- httpResBody = { ...jsonwpRes[1],
332
- ...w3cRes[1]
333
- };
334
- httpStatus = jsonwpRes[0];
335
- }
335
+ [httpStatus, httpResBody] = (0, _errors.getResponseForW3CError)(actualErr);
336
336
  }
337
337
 
338
338
  if (_lodash.default.isString(httpResBody)) {
@@ -352,7 +352,7 @@ function buildHandler(app, method, path, spec, driver, isSessCmd) {
352
352
  delete httpResBody.sessionId;
353
353
  }
354
354
 
355
- httpResBody = (0, _helpers.formatStatus)(httpResBody, httpStatus, currentProtocol);
355
+ httpResBody = (0, _helpers.formatStatus)(httpResBody);
356
356
  res.status(httpStatus).json(httpResBody);
357
357
  }
358
358
  };
@@ -367,7 +367,7 @@ function driverShouldDoJwpProxy(driver, req, command) {
367
367
  return false;
368
368
  }
369
369
 
370
- if (command === 'deleteSession') {
370
+ if (command === DELETE_SESSION_COMMAND) {
371
371
  return false;
372
372
  }
373
373
 
@@ -379,15 +379,14 @@ function driverShouldDoJwpProxy(driver, req, command) {
379
379
  }
380
380
 
381
381
  async function doJwpProxy(driver, req, res) {
382
- _sessionsCache.default.getLogger(req.params.sessionId, extractProtocol(driver, req.params.sessionId)).info('Driver proxy active, passing request on via HTTP proxy');
382
+ getLogger(driver, req.params.sessionId).info('Driver proxy active, passing request on via HTTP proxy');
383
383
 
384
384
  if (!driver.canProxy(req.params.sessionId)) {
385
- throw new Error('Trying to proxy to a JSONWP server but driver is unable to proxy');
385
+ throw new Error('Trying to proxy to a server but the driver is unable to proxy');
386
386
  }
387
387
 
388
388
  try {
389
- const proxiedRes = await driver.executeCommand('proxyReqRes', req, res, req.params.sessionId);
390
- if (proxiedRes && proxiedRes.error) throw proxiedRes.error;
389
+ await driver.executeCommand('proxyReqRes', req, res, req.params.sessionId);
391
390
  } catch (err) {
392
391
  if ((0, _errors.isErrorType)(err, _errors.errors.ProxyRequestError)) {
393
392
  throw err;
@@ -398,4 +397,4 @@ async function doJwpProxy(driver, req, res) {
398
397
  }require('source-map-support').install();
399
398
 
400
399
 
401
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/protocol/protocol.js"],"names":["CREATE_SESSION_COMMAND","DELETE_SESSION_COMMAND","Protocol","determineProtocol","desiredCapabilities","requiredCapabilities","capabilities","_","isPlainObject","PROTOCOLS","W3C","MJSONWP","extractProtocol","driver","sessionId","dstDriver","isFunction","driverForSession","protocol","SESSIONS_CACHE","getProtocol","isSessionCommand","command","includes","NO_SESSION_ID_COMMANDS","wrapParams","paramSets","jsonObj","res","isArray","isObject","wrap","unwrapParams","unwrap","checkParams","requiredParams","optionalParams","receivedParams","keys","required","first","optional","validate","message","errors","BadParametersError","length","indexOf","push","params","difference","makeArgs","requestParams","payloadParams","urlParams","reverse","without","args","flatten","map","p","concat","u","routeConfiguringFunction","sessionExists","Error","executeCommand","execute","addRoutes","app","basePath","DEFAULT_BASE_PATH","extraMethodMap","allMethods","METHOD_MAP","path","methods","toPairs","method","spec","buildHandler","isSessCmd","asyncHandler","req","body","httpResBody","httpStatus","newSessionId","currentProtocol","NoSuchDriverError","didPluginOverrideProxy","neverProxy","driverShouldDoJwpProxy","pluginsToHandleCmd","doJwpProxy","getLogger","debug","NotImplementedError","driverRes","validators","constructor","name","truncate","JSON","stringify","MAX_LOG_BODY_LENGTH","reqForProxy","has","error","value","putSession","util","hasValue","status","isNaN","parseInt","stacktrace","resetLogger","err","actualErr","errMsg","stack","ProxyRequestError","getActualError","jsonwpRes","w3cRes","isString","send","json","toLowerCase","B","resolve","done","proxyActive","proxyRouteIsAvoided","originalUrl","info","canProxy","proxiedRes"],"mappings":";;;;;;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AAIA;;AACA;;AACA;;AAGA;;AACA;;AAGA,MAAMA,sBAAsB,GAAG,eAA/B;;AACA,MAAMC,sBAAsB,GAAG,eAA/B;;;AAEA,MAAMC,QAAN,CAAe;;;;AAEf,SAASC,iBAAT,CAA4BC,mBAA5B,EAAiDC,oBAAjD,EAAuEC,YAAvE,EAAqF;AACnF,SAAOC,gBAAEC,aAAF,CAAgBF,YAAhB,IACLG,qBAAUC,GADL,GAELD,qBAAUE,OAFZ;AAGD;;AAED,SAASC,eAAT,CAA0BC,MAA1B,EAAkCC,SAAS,GAAG,IAA9C,EAAoD;AAClD,QAAMC,SAAS,GAAGR,gBAAES,UAAF,CAAaH,MAAM,CAACI,gBAApB,IACdJ,MAAM,CAACI,gBAAP,CAAwBH,SAAxB,CADc,GAEdD,MAFJ;;AAGA,MAAIE,SAAS,KAAKF,MAAlB,EAA0B;AAIxB,WAAOA,MAAM,CAACK,QAAd;AACD;;AAGD,SAAOH,SAAS,GAAGA,SAAS,CAACG,QAAb,GAAwBC,uBAAeC,WAAf,CAA2BN,SAA3B,CAAxC;AACD;;AAED,SAASO,gBAAT,CAA2BC,OAA3B,EAAoC;AAClC,SAAO,CAACf,gBAAEgB,QAAF,CAAWC,8BAAX,EAAmCF,OAAnC,CAAR;AACD;;AAED,SAASG,UAAT,CAAqBC,SAArB,EAAgCC,OAAhC,EAAyC;AAOvC,MAAIC,GAAG,GAAGD,OAAV;;AACA,MAAIpB,gBAAEsB,OAAF,CAAUF,OAAV,KAAsB,CAACpB,gBAAEuB,QAAF,CAAWH,OAAX,CAA3B,EAAgD;AAC9CC,IAAAA,GAAG,GAAG,EAAN;AACAA,IAAAA,GAAG,CAACF,SAAS,CAACK,IAAX,CAAH,GAAsBJ,OAAtB;AACD;;AACD,SAAOC,GAAP;AACD;;AAED,SAASI,YAAT,CAAuBN,SAAvB,EAAkCC,OAAlC,EAA2C;AAIzC,MAAIC,GAAG,GAAGD,OAAV;;AACA,MAAIpB,gBAAEuB,QAAF,CAAWH,OAAX,CAAJ,EAAyB;AAEvB,QAAIA,OAAO,CAACD,SAAS,CAACO,MAAX,CAAX,EAA+B;AAC7BL,MAAAA,GAAG,GAAGD,OAAO,CAACD,SAAS,CAACO,MAAX,CAAb;AACD;AACF;;AACD,SAAOL,GAAP;AACD;;AAED,SAASM,WAAT,CAAsBR,SAAtB,EAAiCC,OAAjC,EAA0CT,QAA1C,EAAoD;AAClD,MAAIiB,cAAc,GAAG,EAArB;AACA,MAAIC,cAAc,GAAG,EAArB;;AACA,MAAIC,cAAc,GAAG9B,gBAAE+B,IAAF,CAAOX,OAAP,CAArB;;AAEA,MAAID,SAAJ,EAAe;AACb,QAAIA,SAAS,CAACa,QAAd,EAAwB;AAGtB,UAAI,CAAChC,gBAAEsB,OAAF,CAAUtB,gBAAEiC,KAAF,CAAQd,SAAS,CAACa,QAAlB,CAAV,CAAL,EAA6C;AAC3CJ,QAAAA,cAAc,GAAG,CAACT,SAAS,CAACa,QAAX,CAAjB;AACD,OAFD,MAEO;AACLJ,QAAAA,cAAc,GAAGT,SAAS,CAACa,QAA3B;AACD;AACF;;AAED,QAAIb,SAAS,CAACe,QAAd,EAAwB;AACtBL,MAAAA,cAAc,GAAGV,SAAS,CAACe,QAA3B;AACD;;AAMD,QAAIf,SAAS,CAACgB,QAAd,EAAwB;AACtB,UAAIC,OAAO,GAAGjB,SAAS,CAACgB,QAAV,CAAmBf,OAAnB,EAA4BT,QAA5B,CAAd;;AACA,UAAIyB,OAAJ,EAAa;AACX,cAAM,IAAIC,eAAOC,kBAAX,CAA8BF,OAA9B,EAAuChB,OAAvC,CAAN;AACD;AACF;AACF;;AAGD,MAAIQ,cAAc,CAACW,MAAf,KAA0B,CAA9B,EAAiC;AAC/B;AACD;;AAGD,MAAIV,cAAc,CAACW,OAAf,CAAuB,WAAvB,MAAwC,CAAC,CAA7C,EAAgD;AAC9CX,IAAAA,cAAc,CAACY,IAAf,CAAoB,WAApB;AACD;;AAGD,MAAIZ,cAAc,CAACW,OAAf,CAAuB,IAAvB,MAAiC,CAAC,CAAtC,EAAyC;AACvCX,IAAAA,cAAc,CAACY,IAAf,CAAoB,IAApB;AACD;;AAGD,OAAK,IAAIC,MAAT,IAAmBd,cAAnB,EAAmC;AACjC,QAAI5B,gBAAE2C,UAAF,CAAab,cAAb,EAA6BY,MAA7B,EAAqCb,cAArC,EAAqDU,MAArD,KAAgE,CAAhE,IACAvC,gBAAE2C,UAAF,CAAaD,MAAb,EAAqBZ,cAArB,EAAqCS,MAArC,KAAgD,CADpD,EACuD;AAGrD;AACD;AACF;;AACD,QAAM,IAAIF,eAAOC,kBAAX,CAA8BnB,SAA9B,EAAyCW,cAAzC,CAAN;AACD;;AASD,SAASc,QAAT,CAAmBC,aAAnB,EAAkCzB,OAAlC,EAA2C0B,aAA3C,EAA0DnC,QAA1D,EAAoE;AAKlE,MAAIoC,SAAS,GAAG/C,gBAAE+B,IAAF,CAAOc,aAAP,EAAsBG,OAAtB,EAAhB;;AAMA,MAAIpB,cAAc,GAAGkB,aAAa,CAACd,QAAnC;;AACA,MAAIhC,gBAAEsB,OAAF,CAAUtB,gBAAEiC,KAAF,CAAQa,aAAa,CAACd,QAAtB,CAAV,CAAJ,EAAgD;AAK9C,QAAID,IAAI,GAAG/B,gBAAE+B,IAAF,CAAOX,OAAP,CAAX;;AACA,SAAK,IAAIsB,MAAT,IAAmBI,aAAa,CAACd,QAAjC,EAA2C;AACzC,UAAIhC,gBAAEiD,OAAF,CAAUP,MAAV,EAAkB,GAAGX,IAArB,EAA2BQ,MAA3B,KAAsC,CAA1C,EAA6C;AAC3CX,QAAAA,cAAc,GAAGc,MAAjB;AACA;AACD;AACF;AACF;;AAGD,MAAIQ,IAAJ;;AACA,MAAIlD,gBAAES,UAAF,CAAaqC,aAAa,CAACF,QAA3B,CAAJ,EAA0C;AAOxCM,IAAAA,IAAI,GAAGJ,aAAa,CAACF,QAAd,CAAuBxB,OAAvB,EAAgCT,QAAhC,CAAP;AACD,GARD,MAQO;AAGLuC,IAAAA,IAAI,GAAGlD,gBAAEmD,OAAF,CAAUvB,cAAV,EAA0BwB,GAA1B,CAA+BC,CAAD,IAAOjC,OAAO,CAACiC,CAAD,CAA5C,CAAP;;AACA,QAAIP,aAAa,CAACZ,QAAlB,EAA4B;AAC1BgB,MAAAA,IAAI,GAAGA,IAAI,CAACI,MAAL,CAAYtD,gBAAEmD,OAAF,CAAUL,aAAa,CAACZ,QAAxB,EAAkCkB,GAAlC,CAAuCC,CAAD,IAAOjC,OAAO,CAACiC,CAAD,CAApD,CAAZ,CAAP;AACD;AACF;;AAGDH,EAAAA,IAAI,GAAGA,IAAI,CAACI,MAAL,CAAYP,SAAS,CAACK,GAAV,CAAeG,CAAD,IAAOV,aAAa,CAACU,CAAD,CAAlC,CAAZ,CAAP;AACA,SAAOL,IAAP;AACD;;AAED,SAASM,wBAAT,CAAmClD,MAAnC,EAA2C;AACzC,MAAI,CAACA,MAAM,CAACmD,aAAZ,EAA2B;AACzB,UAAM,IAAIC,KAAJ,CAAU,0DAAV,CAAN;AACD;;AAED,MAAI,EAAEpD,MAAM,CAACqD,cAAP,IAAyBrD,MAAM,CAACsD,OAAlC,CAAJ,EAAgD;AAC9C,UAAM,IAAIF,KAAJ,CAAU,wEAAV,CAAN;AACD;;AAID,SAAO,SAASG,SAAT,CAAoBC,GAApB,EAAyB;AAACC,IAAAA,QAAQ,GAAGC,4BAAZ;AAA+BC,IAAAA,cAAc,GAAG;AAAhD,GAAzB,EAA8E;AAGnF3D,IAAAA,MAAM,CAACyD,QAAP,GAAkBA,QAAlB;AAEA,UAAMG,UAAU,GAAG,EAAC,GAAGC,kBAAJ;AAAgB,SAAGF;AAAnB,KAAnB;;AAEA,SAAK,MAAM,CAACG,IAAD,EAAOC,OAAP,CAAX,IAA8BrE,gBAAEsE,OAAF,CAAUJ,UAAV,CAA9B,EAAqD;AACnD,WAAK,MAAM,CAACK,MAAD,EAASC,IAAT,CAAX,IAA6BxE,gBAAEsE,OAAF,CAAUD,OAAV,CAA7B,EAAiD;AAE/CI,QAAAA,YAAY,CAACX,GAAD,EAAMS,MAAN,EAAe,GAAER,QAAS,GAAEK,IAAK,EAAjC,EAAoCI,IAApC,EAA0ClE,MAA1C,EAAkDQ,gBAAgB,CAAC0D,IAAI,CAACzD,OAAN,CAAlE,CAAZ;AACD;AACF;AACF,GAbD;AAcD;;AAED,SAAS0D,YAAT,CAAuBX,GAAvB,EAA4BS,MAA5B,EAAoCH,IAApC,EAA0CI,IAA1C,EAAgDlE,MAAhD,EAAwDoE,SAAxD,EAAmE;AACjE,MAAIC,YAAY,GAAG,OAAOC,GAAP,EAAYvD,GAAZ,KAAoB;AACrC,QAAID,OAAO,GAAGwD,GAAG,CAACC,IAAlB;AACA,QAAIC,WAAW,GAAG,EAAlB;AACA,QAAIC,UAAU,GAAG,GAAjB;AACA,QAAIC,YAAJ;AACA,QAAIC,eAAe,GAAG5E,eAAe,CAACC,MAAD,EAASsE,GAAG,CAAClC,MAAJ,CAAWnC,SAApB,CAArC;;AAEA,QAAI;AAGF,UAAImE,SAAS,IAAI,CAACpE,MAAM,CAACmD,aAAP,CAAqBmB,GAAG,CAAClC,MAAJ,CAAWnC,SAAhC,CAAlB,EAA8D;AAC5D,cAAM,IAAI8B,eAAO6C,iBAAX,EAAN;AACD;;AAUD,UAAIC,sBAAsB,GAAG,KAA7B;;AACA,UAAIT,SAAS,IAAI,CAACF,IAAI,CAACY,UAAnB,IAAiCC,sBAAsB,CAAC/E,MAAD,EAASsE,GAAT,EAAcJ,IAAI,CAACzD,OAAnB,CAA3D,EAAwF;AACtF,YAAI,CAACT,MAAM,CAACgF,kBAAR,IACAhF,MAAM,CAACgF,kBAAP,CAA0Bd,IAAI,CAACzD,OAA/B,EAAwC6D,GAAG,CAAClC,MAAJ,CAAWnC,SAAnD,EAA8DgC,MAA9D,KAAyE,CAD7E,EACgF;AAC9E,gBAAMgD,UAAU,CAACjF,MAAD,EAASsE,GAAT,EAAcvD,GAAd,CAAhB;AACA;AACD;;AACDT,+BAAe4E,SAAf,CAAyBZ,GAAG,CAAClC,MAAJ,CAAWnC,SAApC,EAA+C0E,eAA/C,EAAgEQ,KAAhE,CAAuE,qBAAD,GACnE,mFADmE,GAEnE,iEAFH;;AAGAN,QAAAA,sBAAsB,GAAG,IAAzB;AACD;;AAID,UAAI,CAACX,IAAI,CAACzD,OAAV,EAAmB;AACjB,cAAM,IAAIsB,eAAOqD,mBAAX,EAAN;AACD;;AAGD,UAAIlB,IAAI,CAAC1B,aAAL,IAAsB0B,IAAI,CAAC1B,aAAL,CAAmBtB,IAA7C,EAAmD;AACjDJ,QAAAA,OAAO,GAAGF,UAAU,CAACsD,IAAI,CAAC1B,aAAN,EAAqB1B,OAArB,CAApB;AACD;;AAGD,UAAIoD,IAAI,CAAC1B,aAAL,IAAsB0B,IAAI,CAAC1B,aAAL,CAAmBpB,MAA7C,EAAqD;AACnDN,QAAAA,OAAO,GAAGK,YAAY,CAAC+C,IAAI,CAAC1B,aAAN,EAAqB1B,OAArB,CAAtB;AACD;;AAED,UAAIoD,IAAI,CAACzD,OAAL,KAAiBtB,sBAArB,EAA6C;AAG3CwF,QAAAA,eAAe,GAAGrF,iBAAiB,CAAC,GAAGgD,QAAQ,CAACgC,GAAG,CAAClC,MAAL,EAAatB,OAAb,EAAsBoD,IAAI,CAAC1B,aAAL,IAAsB,EAA5C,CAAZ,CAAnC;AACD;;AAGDnB,MAAAA,WAAW,CAAC6C,IAAI,CAAC1B,aAAN,EAAqB1B,OAArB,EAA8B6D,eAA9B,CAAX;AAIA,UAAI/B,IAAI,GAAGN,QAAQ,CAACgC,GAAG,CAAClC,MAAL,EAAatB,OAAb,EAAsBoD,IAAI,CAAC1B,aAAL,IAAsB,EAA5C,EAAgDmC,eAAhD,CAAnB;AACA,UAAIU,SAAJ;;AAEA,UAAIC,uBAAWpB,IAAI,CAACzD,OAAhB,CAAJ,EAA8B;AAC5B6E,+BAAWpB,IAAI,CAACzD,OAAhB,EAAyB,GAAGmC,IAA5B;AACD;;AAGDtC,6BAAe4E,SAAf,CAAyBZ,GAAG,CAAClC,MAAJ,CAAWnC,SAApC,EAA+C0E,eAA/C,EAAgEQ,KAAhE,CAAuE,UAAD,GACnE,GAAEnF,MAAM,CAACuF,WAAP,CAAmBC,IAAK,IAAGtB,IAAI,CAACzD,OAAQ,gBADyB,GAEpEf,gBAAE+F,QAAF,CAAWC,IAAI,CAACC,SAAL,CAAe/C,IAAf,CAAX,EAAiC;AAACX,QAAAA,MAAM,EAAE2D;AAAT,OAAjC,CAFF;;AAIA,UAAIf,sBAAJ,EAA4B;AAI1BjC,QAAAA,IAAI,CAACT,IAAL,CAAU;AAAC0D,UAAAA,WAAW,EAAEvB;AAAd,SAAV;AACD;;AAEDe,MAAAA,SAAS,GAAG,MAAMrF,MAAM,CAACqD,cAAP,CAAsBa,IAAI,CAACzD,OAA3B,EAAoC,GAAGmC,IAAvC,CAAlB;AAGA+B,MAAAA,eAAe,GAAG5E,eAAe,CAACC,MAAD,EAASsE,GAAG,CAAClC,MAAJ,CAAWnC,SAApB,CAAf,IAAiD0E,eAAnE;;AAIA,UAAIjF,gBAAEC,aAAF,CAAgB0F,SAAhB,KAA8B3F,gBAAEoG,GAAF,CAAMT,SAAN,EAAiB,UAAjB,CAAlC,EAAgE;AAC9DV,QAAAA,eAAe,GAAGU,SAAS,CAAChF,QAAV,IAAsBsE,eAAxC;;AACA,YAAIU,SAAS,CAACU,KAAd,EAAqB;AACnB,gBAAMV,SAAS,CAACU,KAAhB;AACD;;AACDV,QAAAA,SAAS,GAAGA,SAAS,CAACW,KAAtB;AACD;;AAGD,UAAI9B,IAAI,CAACzD,OAAL,KAAiBtB,sBAArB,EAA6C;AAC3CuF,QAAAA,YAAY,GAAGW,SAAS,CAAC,CAAD,CAAxB;;AACA/E,+BAAe2F,UAAf,CAA0BvB,YAA1B,EAAwCC,eAAxC;;AACArE,+BAAe4E,SAAf,CAAyBR,YAAzB,EAAuCC,eAAvC,EACGQ,KADH,CACU,8BAA6BR,eAAgB,yBAAwBD,YAAa,EAD5F;;AAEA,YAAIC,eAAe,KAAK/E,qBAAUE,OAAlC,EAA2C;AACzCuF,UAAAA,SAAS,GAAGA,SAAS,CAAC,CAAD,CAArB;AACD,SAFD,MAEO,IAAIV,eAAe,KAAK/E,qBAAUC,GAAlC,EAAuC;AAC5CwF,UAAAA,SAAS,GAAG;AACV5F,YAAAA,YAAY,EAAE4F,SAAS,CAAC,CAAD;AADb,WAAZ;AAGD;AACF;;AAEDA,MAAAA,SAAS,GAAG,kCAAoBA,SAApB,CAAZ;;AAGA,UAAInB,IAAI,CAACzD,OAAL,KAAiBrB,sBAArB,EAA6C;AAC3CkB,+BAAe4E,SAAf,CAAyBZ,GAAG,CAAClC,MAAJ,CAAWnC,SAApC,EAA+C0E,eAA/C,EACGQ,KADH,CACU,sBAAqBzF,gBAAE+F,QAAF,CAAWC,IAAI,CAACC,SAAL,CAAeN,SAAf,CAAX,EAAsC;AAACpD,UAAAA,MAAM,EAAE2D;AAAT,SAAtC,CAAqE,EADpG;;AAEAtF,+BAAe4E,SAAf,CAAyBZ,GAAG,CAAClC,MAAJ,CAAWnC,SAApC,EAA+C0E,eAA/C,EAAgEQ,KAAhE,CAAsE,wCAAtE;;AACAE,QAAAA,SAAS,GAAG,IAAZ;AACD;;AAGD,UAAIa,cAAKC,QAAL,CAAcd,SAAd,CAAJ,EAA8B;AAC5B,YAAIa,cAAKC,QAAL,CAAcd,SAAS,CAACe,MAAxB,KAAmC,CAACC,KAAK,CAAChB,SAAS,CAACe,MAAX,CAAzC,IAA+DE,QAAQ,CAACjB,SAAS,CAACe,MAAX,EAAmB,EAAnB,CAAR,KAAmC,CAAtG,EAAyG;AACvG,gBAAM,wCAA2Bf,SAAS,CAACe,MAArC,EAA6Cf,SAAS,CAACW,KAAvD,CAAN;AACD,SAFD,MAEO,IAAItG,gBAAEC,aAAF,CAAgB0F,SAAS,CAACW,KAA1B,KAAoCX,SAAS,CAACW,KAAV,CAAgBD,KAAxD,EAA+D;AACpE,gBAAM,kCAAqBV,SAAS,CAACW,KAAV,CAAgBD,KAArC,EAA4CV,SAAS,CAACW,KAAV,CAAgBlE,OAA5D,EAAqEuD,SAAS,CAACW,KAAV,CAAgBO,UAArF,CAAN;AACD;AACF;;AAED/B,MAAAA,WAAW,CAACwB,KAAZ,GAAoBX,SAApB;;AACA/E,6BAAe4E,SAAf,CAAyBZ,GAAG,CAAClC,MAAJ,CAAWnC,SAAX,IAAwByE,YAAjD,EAA+DC,eAA/D,EAAgFQ,KAAhF,CAAuF,aAAD,GACnF,yBAAwBjB,IAAI,CAACzD,OAAQ,cAAaf,gBAAE+F,QAAF,CAAWC,IAAI,CAACC,SAAL,CAAeN,SAAf,CAAX,EAAsC;AAACpD,QAAAA,MAAM,EAAE2D;AAAT,OAAtC,CAAqE,EAD1H;;AAGA,UAAI1B,IAAI,CAACzD,OAAL,KAAiBrB,sBAArB,EAA6C;AAI3CkB,+BAAekG,WAAf,CAA2BlC,GAAG,CAAClC,MAAJ,CAAWnC,SAAtC;AACD;AACF,KArID,CAqIE,OAAOwG,GAAP,EAAY;AAGZ,UAAIC,SAAS,GAAGD,GAAhB;AAEA9B,MAAAA,eAAe,GAAGA,eAAe,IAAI5E,eAAe,CAACC,MAAD,EAASsE,GAAG,CAAClC,MAAJ,CAAWnC,SAAX,IAAwByE,YAAjC,CAApD;AAEA,UAAIiC,MAAM,GAAGF,GAAG,CAACF,UAAJ,IAAkBE,GAAG,CAACG,KAAnC;;AACA,UAAI,CAAClH,gBAAEgB,QAAF,CAAWiG,MAAX,EAAmBF,GAAG,CAAC3E,OAAvB,CAAL,EAAsC;AAGpC6E,QAAAA,MAAM,GAAI,GAAEF,GAAG,CAAC3E,OAAQ,GAAE6E,MAAM,GAAI,OAAOA,MAAX,GAAqB,EAAG,EAAxD;AACD;;AACD,UAAI,yBAAYF,GAAZ,EAAiB1E,eAAO8E,iBAAxB,CAAJ,EAAgD;AAC9CH,QAAAA,SAAS,GAAGD,GAAG,CAACK,cAAJ,EAAZ;AACD,OAFD,MAEO;AACLxG,+BAAe4E,SAAf,CAAyBZ,GAAG,CAAClC,MAAJ,CAAWnC,SAAX,IAAwByE,YAAjD,EAA+DC,eAA/D,EACGQ,KADH,CACU,+CAA8CwB,MAAO,EAD/D;AAED;;AAED,UAAIhC,eAAe,KAAK/E,qBAAUC,GAAlC,EAAuC;AACrC,SAAC4E,UAAD,EAAaD,WAAb,IAA4B,oCAAuBkC,SAAvB,CAA5B;AACD,OAFD,MAEO,IAAI/B,eAAe,KAAK/E,qBAAUE,OAAlC,EAA2C;AAChD,SAAC2E,UAAD,EAAaD,WAAb,IAA4B,uCAA0BkC,SAA1B,CAA5B;AACD,OAFM,MAEA;AAGL,YAAIK,SAAS,GAAG,uCAA0BL,SAA1B,CAAhB;AACA,YAAIM,MAAM,GAAG,oCAAuBN,SAAvB,CAAb;AAEAlC,QAAAA,WAAW,GAAG,EACZ,GAAGuC,SAAS,CAAC,CAAD,CADA;AAEZ,aAAGC,MAAM,CAAC,CAAD;AAFG,SAAd;AAMAvC,QAAAA,UAAU,GAAGsC,SAAS,CAAC,CAAD,CAAtB;AACD;AACF;;AAGD,QAAIrH,gBAAEuH,QAAF,CAAWzC,WAAX,CAAJ,EAA6B;AAC3BzD,MAAAA,GAAG,CAACqF,MAAJ,CAAW3B,UAAX,EAAuByC,IAAvB,CAA4B1C,WAA5B;AACD,KAFD,MAEO;AACL,UAAIE,YAAJ,EAAkB;AAChB,YAAIC,eAAe,KAAK/E,qBAAUC,GAAlC,EAAuC;AACrC2E,UAAAA,WAAW,CAACwB,KAAZ,CAAkB/F,SAAlB,GAA8ByE,YAA9B;AACD,SAFD,MAEO;AACLF,UAAAA,WAAW,CAACvE,SAAZ,GAAwByE,YAAxB;AACD;AACF,OAND,MAMO;AACLF,QAAAA,WAAW,CAACvE,SAAZ,GAAwBqE,GAAG,CAAClC,MAAJ,CAAWnC,SAAX,IAAwB,IAAhD;AACD;;AAED,UAAI0E,eAAe,KAAK/E,qBAAUC,GAAlC,EAAuC;AACrC,eAAO2E,WAAW,CAACvE,SAAnB;AACD;;AAEDuE,MAAAA,WAAW,GAAG,2BAAaA,WAAb,EAA0BC,UAA1B,EAAsCE,eAAtC,CAAd;AACA5D,MAAAA,GAAG,CAACqF,MAAJ,CAAW3B,UAAX,EAAuB0C,IAAvB,CAA4B3C,WAA5B;AACD;AACF,GAzMD;;AA2MAhB,EAAAA,GAAG,CAACS,MAAM,CAACmD,WAAP,EAAD,CAAH,CAA0BtD,IAA1B,EAAgC,CAACQ,GAAD,EAAMvD,GAAN,KAAc;AAC5CsG,sBAAEC,OAAF,CAAUjD,YAAY,CAACC,GAAD,EAAMvD,GAAN,CAAtB,EAAkCwG,IAAlC;AACD,GAFD;AAGD;;AAED,SAASxC,sBAAT,CAAiC/E,MAAjC,EAAyCsE,GAAzC,EAA8C7D,OAA9C,EAAuD;AAErD,MAAI,CAACT,MAAM,CAACwH,WAAP,CAAmBlD,GAAG,CAAClC,MAAJ,CAAWnC,SAA9B,CAAL,EAA+C;AAC7C,WAAO,KAAP;AACD;;AAID,MAAIQ,OAAO,KAAK,eAAhB,EAAiC;AAC/B,WAAO,KAAP;AACD;;AAID,MAAIT,MAAM,CAACyH,mBAAP,CAA2BnD,GAAG,CAAClC,MAAJ,CAAWnC,SAAtC,EAAiDqE,GAAG,CAACL,MAArD,EAA6DK,GAAG,CAACoD,WAAjE,EAA8EpD,GAAG,CAACC,IAAlF,CAAJ,EAA6F;AAC3F,WAAO,KAAP;AACD;;AAED,SAAO,IAAP;AACD;;AAED,eAAeU,UAAf,CAA2BjF,MAA3B,EAAmCsE,GAAnC,EAAwCvD,GAAxC,EAA6C;AAC3CT,yBAAe4E,SAAf,CAAyBZ,GAAG,CAAClC,MAAJ,CAAWnC,SAApC,EAA+CF,eAAe,CAACC,MAAD,EAASsE,GAAG,CAAClC,MAAJ,CAAWnC,SAApB,CAA9D,EACG0H,IADH,CACQ,wDADR;;AAIA,MAAI,CAAC3H,MAAM,CAAC4H,QAAP,CAAgBtD,GAAG,CAAClC,MAAJ,CAAWnC,SAA3B,CAAL,EAA4C;AAC1C,UAAM,IAAImD,KAAJ,CAAU,kEAAV,CAAN;AACD;;AACD,MAAI;AACF,UAAMyE,UAAU,GAAG,MAAM7H,MAAM,CAACqD,cAAP,CAAsB,aAAtB,EAAqCiB,GAArC,EAA0CvD,GAA1C,EAA+CuD,GAAG,CAAClC,MAAJ,CAAWnC,SAA1D,CAAzB;AACA,QAAI4H,UAAU,IAAIA,UAAU,CAAC9B,KAA7B,EAAoC,MAAM8B,UAAU,CAAC9B,KAAjB;AACrC,GAHD,CAGE,OAAOU,GAAP,EAAY;AACZ,QAAI,yBAAYA,GAAZ,EAAiB1E,eAAO8E,iBAAxB,CAAJ,EAAgD;AAC9C,YAAMJ,GAAN;AACD,KAFD,MAEO;AACL,YAAM,IAAIrD,KAAJ,CAAW,iCAAgCqD,GAAG,CAAC3E,OAAQ,EAAvD,CAAN;AACD;AACF;AACF","sourcesContent":["import _ from 'lodash';\nimport { util } from '@appium/support';\nimport { validators } from './validators';\nimport {\n  errors, isErrorType, getResponseForW3CError, getResponseForJsonwpError,\n  errorFromMJSONWPStatusCode, errorFromW3CJsonCode,\n} from './errors';\nimport { METHOD_MAP, NO_SESSION_ID_COMMANDS } from './routes';\nimport B from 'bluebird';\nimport {\n  formatResponseValue, formatStatus,\n} from './helpers';\nimport { MAX_LOG_BODY_LENGTH, PROTOCOLS, DEFAULT_BASE_PATH } from '../constants';\nimport SESSIONS_CACHE from './sessions-cache';\n\n\nconst CREATE_SESSION_COMMAND = 'createSession';\nconst DELETE_SESSION_COMMAND = 'deleteSession';\n\nclass Protocol {}\n\nfunction determineProtocol (desiredCapabilities, requiredCapabilities, capabilities) {\n  return _.isPlainObject(capabilities) ?\n    PROTOCOLS.W3C :\n    PROTOCOLS.MJSONWP;\n}\n\nfunction extractProtocol (driver, sessionId = null) {\n  const dstDriver = _.isFunction(driver.driverForSession)\n    ? driver.driverForSession(sessionId)\n    : driver;\n  if (dstDriver === driver) {\n    // Shortcircuit if the driver instance is not an umbrella driver\n    // or it is Fake driver instance, where `driver.driverForSession`\n    // always returns self instance\n    return driver.protocol;\n  }\n\n  // Extract the protocol for the current session if the given driver is the umbrella one\n  return dstDriver ? dstDriver.protocol : SESSIONS_CACHE.getProtocol(sessionId);\n}\n\nfunction isSessionCommand (command) {\n  return !_.includes(NO_SESSION_ID_COMMANDS, command);\n}\n\nfunction wrapParams (paramSets, jsonObj) {\n  /* There are commands like performTouch which take a single parameter (primitive type or array).\n   * Some drivers choose to pass this parameter as a value (eg. [action1, action2...]) while others to\n   * wrap it within an object(eg' {gesture:  [action1, action2...]}), which makes it hard to validate.\n   * The wrap option in the spec enforce wrapping before validation, so that all params are wrapped at\n   * the time they are validated and later passed to the commands.\n   */\n  let res = jsonObj;\n  if (_.isArray(jsonObj) || !_.isObject(jsonObj)) {\n    res = {};\n    res[paramSets.wrap] = jsonObj;\n  }\n  return res;\n}\n\nfunction unwrapParams (paramSets, jsonObj) {\n  /* There are commands like setNetworkConnection which send parameters wrapped inside a key such as\n   * \"parameters\". This function unwraps them (eg. {\"parameters\": {\"type\": 1}} becomes {\"type\": 1}).\n   */\n  let res = jsonObj;\n  if (_.isObject(jsonObj)) {\n    // some clients, like ruby, don't wrap\n    if (jsonObj[paramSets.unwrap]) {\n      res = jsonObj[paramSets.unwrap];\n    }\n  }\n  return res;\n}\n\nfunction checkParams (paramSets, jsonObj, protocol) {\n  let requiredParams = [];\n  let optionalParams = [];\n  let receivedParams = _.keys(jsonObj);\n\n  if (paramSets) {\n    if (paramSets.required) {\n      // we might have an array of parameters,\n      // or an array of arrays of parameters, so standardize\n      if (!_.isArray(_.first(paramSets.required))) {\n        requiredParams = [paramSets.required];\n      } else {\n        requiredParams = paramSets.required;\n      }\n    }\n    // optional parameters are just an array\n    if (paramSets.optional) {\n      optionalParams = paramSets.optional;\n    }\n\n    // If a function was provided as the 'validate' key, it will here be called with\n    // jsonObj as the param. If it returns something falsy, verification will be\n    // considered to have passed. If it returns something else, that will be the\n    // argument to an error which is thrown to the user\n    if (paramSets.validate) {\n      let message = paramSets.validate(jsonObj, protocol);\n      if (message) {\n        throw new errors.BadParametersError(message, jsonObj);\n      }\n    }\n  }\n\n  // if we have no required parameters, all is well\n  if (requiredParams.length === 0) {\n    return;\n  }\n\n  // some clients pass in the session id in the params\n  if (optionalParams.indexOf('sessionId') === -1) {\n    optionalParams.push('sessionId');\n  }\n\n  // some clients pass in an element id in the params\n  if (optionalParams.indexOf('id') === -1) {\n    optionalParams.push('id');\n  }\n\n  // go through the required parameters and check against our arguments\n  for (let params of requiredParams) {\n    if (_.difference(receivedParams, params, optionalParams).length === 0 &&\n        _.difference(params, receivedParams).length === 0) {\n      // we have a set of parameters that is correct\n      // so short-circuit\n      return;\n    }\n  }\n  throw new errors.BadParametersError(paramSets, receivedParams);\n}\n\n/*\n * This method takes 3 pieces of data: request parameters ('requestParams'),\n * a request JSON body ('jsonObj'), and 'payloadParams', which is the section\n * from the route definition for a particular endpoint which has instructions\n * on handling parameters. This method returns an array of arguments which will\n * be applied to a command.\n */\nfunction makeArgs (requestParams, jsonObj, payloadParams, protocol) {\n  // We want to pass the \"url\" parameters to the commands in reverse order\n  // since the command will sometimes want to ignore, say, the sessionId.\n  // This has the effect of putting sessionId last, which means in JS we can\n  // omit it from the function signature if we're not going to use it.\n  let urlParams = _.keys(requestParams).reverse();\n\n  // In the simple case, the required parameters are a basic array in\n  // payloadParams.required, so start there. It's possible that there are\n  // multiple optional sets of required params, though, so handle that case\n  // too.\n  let requiredParams = payloadParams.required;\n  if (_.isArray(_.first(payloadParams.required))) {\n    // If there are optional sets of required params, then we will have an\n    // array of arrays in payloadParams.required, so loop through each set and\n    // pick the one that matches which JSON params were actually sent. We've\n    // already been through validation so we're guaranteed to find a match.\n    let keys = _.keys(jsonObj);\n    for (let params of payloadParams.required) {\n      if (_.without(params, ...keys).length === 0) {\n        requiredParams = params;\n        break;\n      }\n    }\n  }\n\n  // Now we construct our list of arguments which will be passed to the command\n  let args;\n  if (_.isFunction(payloadParams.makeArgs)) {\n    // In the route spec, a particular route might define a 'makeArgs' function\n    // if it wants full control over how to turn JSON parameters into command\n    // arguments. So we pass it the JSON parameters and it returns an array\n    // which will be applied to the handling command. For example if it returns\n    // [1, 2, 3], we will call `command(1, 2, 3, ...)` (url params are separate\n    // from JSON params and get concatenated below).\n    args = payloadParams.makeArgs(jsonObj, protocol);\n  } else {\n    // Otherwise, collect all the required and optional params and flatten them\n    // into an argument array\n    args = _.flatten(requiredParams).map((p) => jsonObj[p]);\n    if (payloadParams.optional) {\n      args = args.concat(_.flatten(payloadParams.optional).map((p) => jsonObj[p]));\n    }\n  }\n  // Finally, get our url params (session id, element id, etc...) on the end of\n  // the list\n  args = args.concat(urlParams.map((u) => requestParams[u]));\n  return args;\n}\n\nfunction routeConfiguringFunction (driver) {\n  if (!driver.sessionExists) {\n    throw new Error('Drivers used with MJSONWP must implement `sessionExists`');\n  }\n\n  if (!(driver.executeCommand || driver.execute)) {\n    throw new Error('Drivers used with MJSONWP must implement `executeCommand` or `execute`');\n  }\n\n  // return a function which will add all the routes to the driver. Here extraMethods might be\n  // passed in as defined by Appium plugins, so we need to add those to the default list\n  return function addRoutes (app, {basePath = DEFAULT_BASE_PATH, extraMethodMap = {}}) {\n    // store basePath on the driver instance so it can use it if necessary\n    // for example in determining proxy avoidance\n    driver.basePath = basePath;\n\n    const allMethods = {...METHOD_MAP, ...extraMethodMap};\n\n    for (const [path, methods] of _.toPairs(allMethods)) {\n      for (const [method, spec] of _.toPairs(methods)) {\n        // set up the express route handler\n        buildHandler(app, method, `${basePath}${path}`, spec, driver, isSessionCommand(spec.command));\n      }\n    }\n  };\n}\n\nfunction buildHandler (app, method, path, spec, driver, isSessCmd) {\n  let asyncHandler = async (req, res) => {\n    let jsonObj = req.body;\n    let httpResBody = {};\n    let httpStatus = 200;\n    let newSessionId;\n    let currentProtocol = extractProtocol(driver, req.params.sessionId);\n\n    try {\n      // if this is a session command but we don't have a session,\n      // error out early (especially before proxying)\n      if (isSessCmd && !driver.sessionExists(req.params.sessionId)) {\n        throw new errors.NoSuchDriverError();\n      }\n\n      // if the driver is currently proxying commands to another JSONWP server, bypass all our\n      // checks and assume the upstream server knows what it's doing. But keep this in the\n      // try/catch block so if proxying itself fails, we give a message to the client. Of course we\n      // only want to do these when we have a session command; the Appium driver must be\n      // responsible for start/stop session, etc... We also allow the command spec to declare that\n      // this command should never be proxied (which is useful for plugin developers who add\n      // commands and generally would not want that command to be proxied instead of handled by the\n      // plugin)\n      let didPluginOverrideProxy = false;\n      if (isSessCmd && !spec.neverProxy && driverShouldDoJwpProxy(driver, req, spec.command)) {\n        if (!driver.pluginsToHandleCmd ||\n            driver.pluginsToHandleCmd(spec.command, req.params.sessionId).length === 0) {\n          await doJwpProxy(driver, req, res);\n          return;\n        }\n        SESSIONS_CACHE.getLogger(req.params.sessionId, currentProtocol).debug(`Would have proxied ` +\n          `command directly, but a plugin exists which might require its value, so will let ` +\n          `its value be collected internally and made part of plugin chain`);\n        didPluginOverrideProxy = true;\n      }\n\n      // if a command is not in our method map, it's because we\n      // have no plans to ever implement it\n      if (!spec.command) {\n        throw new errors.NotImplementedError();\n      }\n\n      // wrap params if necessary\n      if (spec.payloadParams && spec.payloadParams.wrap) {\n        jsonObj = wrapParams(spec.payloadParams, jsonObj);\n      }\n\n      // unwrap params if necessary\n      if (spec.payloadParams && spec.payloadParams.unwrap) {\n        jsonObj = unwrapParams(spec.payloadParams, jsonObj);\n      }\n\n      if (spec.command === CREATE_SESSION_COMMAND) {\n        // try to determine protocol by session creation args, so we can throw a\n        // properly formatted error if arguments validation fails\n        currentProtocol = determineProtocol(...makeArgs(req.params, jsonObj, spec.payloadParams || {}));\n      }\n\n      // ensure that the json payload conforms to the spec\n      checkParams(spec.payloadParams, jsonObj, currentProtocol);\n\n      // turn the command and json payload into an argument list for\n      // the driver methods\n      let args = makeArgs(req.params, jsonObj, spec.payloadParams || {}, currentProtocol);\n      let driverRes;\n      // validate command args according to MJSONWP\n      if (validators[spec.command]) {\n        validators[spec.command](...args);\n      }\n\n      // run the driver command wrapped inside the argument validators\n      SESSIONS_CACHE.getLogger(req.params.sessionId, currentProtocol).debug(`Calling ` +\n        `${driver.constructor.name}.${spec.command}() with args: ` +\n        _.truncate(JSON.stringify(args), {length: MAX_LOG_BODY_LENGTH}));\n\n      if (didPluginOverrideProxy) {\n        // TODO for now we add this information on the args list, but that's mixing purposes here.\n        // We really should add another 'options' parameter to 'executeCommand', but this would be\n        // a breaking change for all drivers so would need to be handled carefully.\n        args.push({reqForProxy: req});\n      }\n\n      driverRes = await driver.executeCommand(spec.command, ...args);\n\n      // Get the protocol after executeCommand\n      currentProtocol = extractProtocol(driver, req.params.sessionId) || currentProtocol;\n\n      // If `executeCommand` was overridden and the method returns an object\n      // with a protocol and value/error property, re-assign the protocol\n      if (_.isPlainObject(driverRes) && _.has(driverRes, 'protocol')) {\n        currentProtocol = driverRes.protocol || currentProtocol;\n        if (driverRes.error) {\n          throw driverRes.error;\n        }\n        driverRes = driverRes.value;\n      }\n\n      // unpack createSession response\n      if (spec.command === CREATE_SESSION_COMMAND) {\n        newSessionId = driverRes[0];\n        SESSIONS_CACHE.putSession(newSessionId, currentProtocol);\n        SESSIONS_CACHE.getLogger(newSessionId, currentProtocol)\n          .debug(`Cached the protocol value '${currentProtocol}' for the new session ${newSessionId}`);\n        if (currentProtocol === PROTOCOLS.MJSONWP) {\n          driverRes = driverRes[1];\n        } else if (currentProtocol === PROTOCOLS.W3C) {\n          driverRes = {\n            capabilities: driverRes[1],\n          };\n        }\n      }\n\n      driverRes = formatResponseValue(driverRes);\n\n      // delete should not return anything even if successful\n      if (spec.command === DELETE_SESSION_COMMAND) {\n        SESSIONS_CACHE.getLogger(req.params.sessionId, currentProtocol)\n          .debug(`Received response: ${_.truncate(JSON.stringify(driverRes), {length: MAX_LOG_BODY_LENGTH})}`);\n        SESSIONS_CACHE.getLogger(req.params.sessionId, currentProtocol).debug('But deleting session, so not returning');\n        driverRes = null;\n      }\n\n      // if the status is not 0,  throw the appropriate error for status code.\n      if (util.hasValue(driverRes)) {\n        if (util.hasValue(driverRes.status) && !isNaN(driverRes.status) && parseInt(driverRes.status, 10) !== 0) {\n          throw errorFromMJSONWPStatusCode(driverRes.status, driverRes.value);\n        } else if (_.isPlainObject(driverRes.value) && driverRes.value.error) {\n          throw errorFromW3CJsonCode(driverRes.value.error, driverRes.value.message, driverRes.value.stacktrace);\n        }\n      }\n\n      httpResBody.value = driverRes;\n      SESSIONS_CACHE.getLogger(req.params.sessionId || newSessionId, currentProtocol).debug(`Responding ` +\n        `to client with driver.${spec.command}() result: ${_.truncate(JSON.stringify(driverRes), {length: MAX_LOG_BODY_LENGTH})}`);\n\n      if (spec.command === DELETE_SESSION_COMMAND) {\n        // We don't want to keep the logger instance in the cache\n        // after the session is deleted, because it contains the logging history\n        // and consumes the memory\n        SESSIONS_CACHE.resetLogger(req.params.sessionId);\n      }\n    } catch (err) {\n      // if anything goes wrong, figure out what our response should be\n      // based on the type of error that we encountered\n      let actualErr = err;\n\n      currentProtocol = currentProtocol || extractProtocol(driver, req.params.sessionId || newSessionId);\n\n      let errMsg = err.stacktrace || err.stack;\n      if (!_.includes(errMsg, err.message)) {\n        // if the message has more information, add it. but often the message\n        // is the first part of the stack trace\n        errMsg = `${err.message}${errMsg ? ('\\n' + errMsg) : ''}`;\n      }\n      if (isErrorType(err, errors.ProxyRequestError)) {\n        actualErr = err.getActualError();\n      } else {\n        SESSIONS_CACHE.getLogger(req.params.sessionId || newSessionId, currentProtocol)\n          .debug(`Encountered internal error running command: ${errMsg}`);\n      }\n\n      if (currentProtocol === PROTOCOLS.W3C) {\n        [httpStatus, httpResBody] = getResponseForW3CError(actualErr);\n      } else if (currentProtocol === PROTOCOLS.MJSONWP) {\n        [httpStatus, httpResBody] = getResponseForJsonwpError(actualErr);\n      } else {\n        // If it's unknown what the protocol is (like if it's `getStatus` prior to `createSession`), merge the responses\n        // together to be protocol-agnostic\n        let jsonwpRes = getResponseForJsonwpError(actualErr);\n        let w3cRes = getResponseForW3CError(actualErr);\n\n        httpResBody = {\n          ...jsonwpRes[1],\n          ...w3cRes[1],\n        };\n\n        // Use the JSONWP status code (which is usually 500)\n        httpStatus = jsonwpRes[0];\n      }\n    }\n\n    // decode the response, which is either a string or json\n    if (_.isString(httpResBody)) {\n      res.status(httpStatus).send(httpResBody);\n    } else {\n      if (newSessionId) {\n        if (currentProtocol === PROTOCOLS.W3C) {\n          httpResBody.value.sessionId = newSessionId;\n        } else {\n          httpResBody.sessionId = newSessionId;\n        }\n      } else {\n        httpResBody.sessionId = req.params.sessionId || null;\n      }\n      // Don't include sessionId in W3C responses\n      if (currentProtocol === PROTOCOLS.W3C) {\n        delete httpResBody.sessionId;\n      }\n\n      httpResBody = formatStatus(httpResBody, httpStatus, currentProtocol);\n      res.status(httpStatus).json(httpResBody);\n    }\n  };\n  // add the method to the app\n  app[method.toLowerCase()](path, (req, res) => {\n    B.resolve(asyncHandler(req, res)).done();\n  });\n}\n\nfunction driverShouldDoJwpProxy (driver, req, command) {\n  // drivers need to explicitly say when the proxy is active\n  if (!driver.proxyActive(req.params.sessionId)) {\n    return false;\n  }\n\n  // we should never proxy deleteSession because we need to give the containing\n  // driver an opportunity to clean itself up\n  if (command === 'deleteSession') {\n    return false;\n  }\n\n  // validate avoidance schema, and say we shouldn't proxy if anything in the\n  // avoid list matches our req\n  if (driver.proxyRouteIsAvoided(req.params.sessionId, req.method, req.originalUrl, req.body)) {\n    return false;\n  }\n\n  return true;\n}\n\nasync function doJwpProxy (driver, req, res) {\n  SESSIONS_CACHE.getLogger(req.params.sessionId, extractProtocol(driver, req.params.sessionId))\n    .info('Driver proxy active, passing request on via HTTP proxy');\n\n  // check that the inner driver has a proxy function\n  if (!driver.canProxy(req.params.sessionId)) {\n    throw new Error('Trying to proxy to a JSONWP server but driver is unable to proxy');\n  }\n  try {\n    const proxiedRes = await driver.executeCommand('proxyReqRes', req, res, req.params.sessionId);\n    if (proxiedRes && proxiedRes.error) throw proxiedRes.error; // eslint-disable-line curly\n  } catch (err) {\n    if (isErrorType(err, errors.ProxyRequestError)) {\n      throw err;\n    } else {\n      throw new Error(`Could not proxy. Proxy error: ${err.message}`);\n    }\n  }\n}\n\n\nexport {\n  Protocol, routeConfiguringFunction, isSessionCommand,\n  driverShouldDoJwpProxy, determineProtocol, CREATE_SESSION_COMMAND, DELETE_SESSION_COMMAND,\n};\n"],"file":"lib/protocol/protocol.js","sourceRoot":"../../.."}
400
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/protocol/protocol.js"],"names":["CREATE_SESSION_COMMAND","DELETE_SESSION_COMMAND","GET_STATUS_COMMAND","Protocol","determineProtocol","createSessionArgs","_","some","isW3cCaps","PROTOCOLS","W3C","MJSONWP","extractProtocol","driver","sessionId","dstDriver","isFunction","driverForSession","protocol","isSessionCommand","command","includes","NO_SESSION_ID_COMMANDS","getLogger","log","info","logPrefix","constructor","name","node","getObjectId","substring","logger","wrapParams","paramSets","jsonObj","res","isArray","isObject","wrap","unwrapParams","unwrap","checkParams","requiredParams","optionalParams","receivedParams","keys","required","first","optional","validate","message","errors","BadParametersError","length","indexOf","push","params","difference","makeArgs","requestParams","payloadParams","urlParams","reverse","without","args","flatten","map","p","concat","u","routeConfiguringFunction","sessionExists","Error","executeCommand","execute","addRoutes","app","basePath","DEFAULT_BASE_PATH","extraMethodMap","allMethods","METHOD_MAP","path","methods","toPairs","method","spec","buildHandler","isSessCmd","asyncHandler","req","body","httpResBody","httpStatus","newSessionId","currentProtocol","NoSuchDriverError","didPluginOverrideProxy","neverProxy","driverShouldDoJwpProxy","pluginsToHandleCmd","doJwpProxy","debug","NotImplementedError","driverRes","validators","truncate","JSON","stringify","MAX_LOG_BODY_LENGTH","reqForProxy","isPlainObject","has","error","value","capabilities","util","hasValue","status","isNaN","parseInt","stacktrace","err","actualErr","errMsg","stack","ProxyRequestError","getActualError","isString","send","json","toLowerCase","B","resolve","done","proxyActive","proxyRouteIsAvoided","originalUrl","canProxy"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport { util, logger, node } from '@appium/support';\nimport { validators } from './validators';\nimport {\n  errors, isErrorType, getResponseForW3CError,\n  errorFromMJSONWPStatusCode, errorFromW3CJsonCode,\n} from './errors';\nimport { METHOD_MAP, NO_SESSION_ID_COMMANDS } from './routes';\nimport B from 'bluebird';\nimport { formatResponseValue, formatStatus } from './helpers';\nimport { MAX_LOG_BODY_LENGTH, PROTOCOLS, DEFAULT_BASE_PATH } from '../constants';\nimport { isW3cCaps } from '../helpers/capabilities';\n\n\nconst CREATE_SESSION_COMMAND = 'createSession';\nconst DELETE_SESSION_COMMAND = 'deleteSession';\nconst GET_STATUS_COMMAND = 'getStatus';\n\nclass Protocol {}\n\nfunction determineProtocol (createSessionArgs) {\n  return _.some(createSessionArgs, isW3cCaps) ? PROTOCOLS.W3C : PROTOCOLS.MJSONWP;\n}\n\nfunction extractProtocol (driver, sessionId = null) {\n  const dstDriver = _.isFunction(driver.driverForSession)\n    ? driver.driverForSession(sessionId)\n    : driver;\n  if (dstDriver === driver) {\n    // Shortcircuit if the driver instance is not an umbrella driver\n    // or it is Fake driver instance, where `driver.driverForSession`\n    // always returns self instance\n    return driver.protocol;\n  }\n\n  // Extract the protocol for the current session if the given driver is the umbrella one\n  return dstDriver?.protocol ?? PROTOCOLS.W3C;\n}\n\nfunction isSessionCommand (command) {\n  return !_.includes(NO_SESSION_ID_COMMANDS, command);\n}\n\nfunction getLogger (driver, sessionId = null) {\n  const dstDriver = sessionId && _.isFunction(driver.driverForSession)\n    ? (driver.driverForSession(sessionId) ?? driver)\n    : driver;\n  if (_.isFunction(dstDriver.log?.info)) {\n    return dstDriver.log;\n  }\n\n  let logPrefix = dstDriver.constructor\n    ? `${dstDriver.constructor.name}@${node.getObjectId(dstDriver).substring(0, 8)}`\n    : 'AppiumDriver';\n  if (sessionId) {\n    logPrefix += ` (${sessionId.substring(0, 8)})`;\n  }\n  return logger.getLogger(logPrefix);\n}\n\nfunction wrapParams (paramSets, jsonObj) {\n  /* There are commands like performTouch which take a single parameter (primitive type or array).\n   * Some drivers choose to pass this parameter as a value (eg. [action1, action2...]) while others to\n   * wrap it within an object(eg' {gesture:  [action1, action2...]}), which makes it hard to validate.\n   * The wrap option in the spec enforce wrapping before validation, so that all params are wrapped at\n   * the time they are validated and later passed to the commands.\n   */\n  let res = jsonObj;\n  if (_.isArray(jsonObj) || !_.isObject(jsonObj)) {\n    res = {};\n    res[paramSets.wrap] = jsonObj;\n  }\n  return res;\n}\n\nfunction unwrapParams (paramSets, jsonObj) {\n  /* There are commands like setNetworkConnection which send parameters wrapped inside a key such as\n   * \"parameters\". This function unwraps them (eg. {\"parameters\": {\"type\": 1}} becomes {\"type\": 1}).\n   */\n  let res = jsonObj;\n  if (_.isObject(jsonObj)) {\n    // some clients, like ruby, don't wrap\n    if (jsonObj[paramSets.unwrap]) {\n      res = jsonObj[paramSets.unwrap];\n    }\n  }\n  return res;\n}\n\nfunction checkParams (paramSets, jsonObj, protocol) {\n  let requiredParams = [];\n  let optionalParams = [];\n  let receivedParams = _.keys(jsonObj);\n\n  if (paramSets) {\n    if (paramSets.required) {\n      // we might have an array of parameters,\n      // or an array of arrays of parameters, so standardize\n      if (!_.isArray(_.first(paramSets.required))) {\n        requiredParams = [paramSets.required];\n      } else {\n        requiredParams = paramSets.required;\n      }\n    }\n    // optional parameters are just an array\n    if (paramSets.optional) {\n      optionalParams = paramSets.optional;\n    }\n\n    // If a function was provided as the 'validate' key, it will here be called with\n    // jsonObj as the param. If it returns something falsy, verification will be\n    // considered to have passed. If it returns something else, that will be the\n    // argument to an error which is thrown to the user\n    if (paramSets.validate) {\n      let message = paramSets.validate(jsonObj, protocol);\n      if (message) {\n        throw new errors.BadParametersError(message, jsonObj);\n      }\n    }\n  }\n\n  // if we have no required parameters, all is well\n  if (requiredParams.length === 0) {\n    return;\n  }\n\n  // some clients pass in the session id in the params\n  if (optionalParams.indexOf('sessionId') === -1) {\n    optionalParams.push('sessionId');\n  }\n\n  // some clients pass in an element id in the params\n  if (optionalParams.indexOf('id') === -1) {\n    optionalParams.push('id');\n  }\n\n  // go through the required parameters and check against our arguments\n  for (let params of requiredParams) {\n    if (_.difference(receivedParams, params, optionalParams).length === 0 &&\n        _.difference(params, receivedParams).length === 0) {\n      // we have a set of parameters that is correct\n      // so short-circuit\n      return;\n    }\n  }\n  throw new errors.BadParametersError(paramSets, receivedParams);\n}\n\n/*\n * This method takes 3 pieces of data: request parameters ('requestParams'),\n * a request JSON body ('jsonObj'), and 'payloadParams', which is the section\n * from the route definition for a particular endpoint which has instructions\n * on handling parameters. This method returns an array of arguments which will\n * be applied to a command.\n */\nfunction makeArgs (requestParams, jsonObj, payloadParams, protocol) {\n  // We want to pass the \"url\" parameters to the commands in reverse order\n  // since the command will sometimes want to ignore, say, the sessionId.\n  // This has the effect of putting sessionId last, which means in JS we can\n  // omit it from the function signature if we're not going to use it.\n  let urlParams = _.keys(requestParams).reverse();\n\n  // In the simple case, the required parameters are a basic array in\n  // payloadParams.required, so start there. It's possible that there are\n  // multiple optional sets of required params, though, so handle that case\n  // too.\n  let requiredParams = payloadParams.required;\n  if (_.isArray(_.first(payloadParams.required))) {\n    // If there are optional sets of required params, then we will have an\n    // array of arrays in payloadParams.required, so loop through each set and\n    // pick the one that matches which JSON params were actually sent. We've\n    // already been through validation so we're guaranteed to find a match.\n    let keys = _.keys(jsonObj);\n    for (let params of payloadParams.required) {\n      if (_.without(params, ...keys).length === 0) {\n        requiredParams = params;\n        break;\n      }\n    }\n  }\n\n  // Now we construct our list of arguments which will be passed to the command\n  let args;\n  if (_.isFunction(payloadParams.makeArgs)) {\n    // In the route spec, a particular route might define a 'makeArgs' function\n    // if it wants full control over how to turn JSON parameters into command\n    // arguments. So we pass it the JSON parameters and it returns an array\n    // which will be applied to the handling command. For example if it returns\n    // [1, 2, 3], we will call `command(1, 2, 3, ...)` (url params are separate\n    // from JSON params and get concatenated below).\n    args = payloadParams.makeArgs(jsonObj, protocol);\n  } else {\n    // Otherwise, collect all the required and optional params and flatten them\n    // into an argument array\n    args = _.flatten(requiredParams).map((p) => jsonObj[p]);\n    if (payloadParams.optional) {\n      args = args.concat(_.flatten(payloadParams.optional).map((p) => jsonObj[p]));\n    }\n  }\n  // Finally, get our url params (session id, element id, etc...) on the end of\n  // the list\n  args = args.concat(urlParams.map((u) => requestParams[u]));\n  return args;\n}\n\nfunction routeConfiguringFunction (driver) {\n  if (!driver.sessionExists) {\n    throw new Error('Drivers must implement `sessionExists` property');\n  }\n\n  if (!(driver.executeCommand || driver.execute)) {\n    throw new Error('Drivers must implement `executeCommand` or `execute` method');\n  }\n\n  // return a function which will add all the routes to the driver. Here extraMethods might be\n  // passed in as defined by Appium plugins, so we need to add those to the default list\n  return function addRoutes (app, {basePath = DEFAULT_BASE_PATH, extraMethodMap = {}}) {\n    // store basePath on the driver instance so it can use it if necessary\n    // for example in determining proxy avoidance\n    driver.basePath = basePath;\n\n    const allMethods = {...METHOD_MAP, ...extraMethodMap};\n\n    for (const [path, methods] of _.toPairs(allMethods)) {\n      for (const [method, spec] of _.toPairs(methods)) {\n        // set up the express route handler\n        buildHandler(app, method, `${basePath}${path}`, spec, driver, isSessionCommand(spec.command));\n      }\n    }\n  };\n}\n\nfunction buildHandler (app, method, path, spec, driver, isSessCmd) {\n  let asyncHandler = async (req, res) => {\n    let jsonObj = req.body;\n    let httpResBody = {};\n    let httpStatus = 200;\n    let newSessionId;\n    let currentProtocol = extractProtocol(driver, req.params.sessionId);\n\n    try {\n      // if this is a session command but we don't have a session,\n      // error out early (especially before proxying)\n      if (isSessCmd && !driver.sessionExists(req.params.sessionId)) {\n        throw new errors.NoSuchDriverError();\n      }\n\n      // if the driver is currently proxying commands to another JSONWP server, bypass all our\n      // checks and assume the upstream server knows what it's doing. But keep this in the\n      // try/catch block so if proxying itself fails, we give a message to the client. Of course we\n      // only want to do these when we have a session command; the Appium driver must be\n      // responsible for start/stop session, etc... We also allow the command spec to declare that\n      // this command should never be proxied (which is useful for plugin developers who add\n      // commands and generally would not want that command to be proxied instead of handled by the\n      // plugin)\n      let didPluginOverrideProxy = false;\n      if (isSessCmd && !spec.neverProxy && driverShouldDoJwpProxy(driver, req, spec.command)) {\n        if (!driver.pluginsToHandleCmd ||\n            driver.pluginsToHandleCmd(spec.command, req.params.sessionId).length === 0) {\n          await doJwpProxy(driver, req, res);\n          return;\n        }\n        getLogger(driver, req.params.sessionId).debug(`Would have proxied ` +\n          `command directly, but a plugin exists which might require its value, so will let ` +\n          `its value be collected internally and made part of plugin chain`);\n        didPluginOverrideProxy = true;\n      }\n\n      // if a command is not in our method map, it's because we\n      // have no plans to ever implement it\n      if (!spec.command) {\n        throw new errors.NotImplementedError();\n      }\n\n      // wrap params if necessary\n      if (spec.payloadParams && spec.payloadParams.wrap) {\n        jsonObj = wrapParams(spec.payloadParams, jsonObj);\n      }\n\n      // unwrap params if necessary\n      if (spec.payloadParams && spec.payloadParams.unwrap) {\n        jsonObj = unwrapParams(spec.payloadParams, jsonObj);\n      }\n\n      if (spec.command === CREATE_SESSION_COMMAND) {\n        // try to determine protocol by session creation args, so we can throw a\n        // properly formatted error if arguments validation fails\n        currentProtocol = determineProtocol(makeArgs(req.params, jsonObj, spec.payloadParams || {}));\n      }\n\n      // ensure that the json payload conforms to the spec\n      checkParams(spec.payloadParams, jsonObj, currentProtocol);\n\n      // turn the command and json payload into an argument list for\n      // the driver methods\n      let args = makeArgs(req.params, jsonObj, spec.payloadParams || {}, currentProtocol);\n      let driverRes;\n      // validate command args according to MJSONWP\n      if (validators[spec.command]) {\n        validators[spec.command](...args);\n      }\n\n      // run the driver command wrapped inside the argument validators\n      getLogger(driver, req.params.sessionId).debug(`Calling ` +\n        `${driver.constructor.name}.${spec.command}() with args: ` +\n        _.truncate(JSON.stringify(args), {length: MAX_LOG_BODY_LENGTH}));\n\n      if (didPluginOverrideProxy) {\n        // TODO for now we add this information on the args list, but that's mixing purposes here.\n        // We really should add another 'options' parameter to 'executeCommand', but this would be\n        // a breaking change for all drivers so would need to be handled carefully.\n        args.push({reqForProxy: req});\n      }\n\n      driverRes = await driver.executeCommand(spec.command, ...args);\n\n      // Get the protocol after executeCommand\n      currentProtocol = extractProtocol(driver, req.params.sessionId) || currentProtocol;\n\n      // If `executeCommand` was overridden and the method returns an object\n      // with a protocol and value/error property, re-assign the protocol\n      if (_.isPlainObject(driverRes) && _.has(driverRes, 'protocol')) {\n        currentProtocol = driverRes.protocol || currentProtocol;\n        if (driverRes.error) {\n          throw driverRes.error;\n        }\n        driverRes = driverRes.value;\n      }\n\n      // unpack createSession response\n      if (spec.command === CREATE_SESSION_COMMAND) {\n        newSessionId = driverRes[0];\n        getLogger(driver, newSessionId)\n          .debug(`Cached the protocol value '${currentProtocol}' for the new session ${newSessionId}`);\n        if (currentProtocol === PROTOCOLS.MJSONWP) {\n          driverRes = driverRes[1];\n        } else if (currentProtocol === PROTOCOLS.W3C) {\n          driverRes = {\n            capabilities: driverRes[1],\n          };\n        }\n      }\n\n      driverRes = formatResponseValue(driverRes);\n\n      // delete should not return anything even if successful\n      if (spec.command === DELETE_SESSION_COMMAND) {\n        getLogger(driver, req.params.sessionId)\n          .debug(`Received response: ${_.truncate(JSON.stringify(driverRes), {length: MAX_LOG_BODY_LENGTH})}`);\n        getLogger(driver, req.params.sessionId).debug('But deleting session, so not returning');\n        driverRes = null;\n      }\n\n      // if the status is not 0,  throw the appropriate error for status code.\n      if (util.hasValue(driverRes)) {\n        if (util.hasValue(driverRes.status) && !isNaN(driverRes.status) && parseInt(driverRes.status, 10) !== 0) {\n          throw errorFromMJSONWPStatusCode(driverRes.status, driverRes.value);\n        } else if (_.isPlainObject(driverRes.value) && driverRes.value.error) {\n          throw errorFromW3CJsonCode(driverRes.value.error, driverRes.value.message, driverRes.value.stacktrace);\n        }\n      }\n\n      httpResBody.value = driverRes;\n      getLogger(driver, req.params.sessionId || newSessionId).debug(`Responding ` +\n        `to client with driver.${spec.command}() result: ${_.truncate(JSON.stringify(driverRes), {length: MAX_LOG_BODY_LENGTH})}`);\n    } catch (err) {\n      // if anything goes wrong, figure out what our response should be\n      // based on the type of error that we encountered\n      let actualErr = err;\n\n      currentProtocol = currentProtocol || extractProtocol(driver, req.params.sessionId || newSessionId);\n\n      let errMsg = err.stacktrace || err.stack;\n      if (!_.includes(errMsg, err.message)) {\n        // if the message has more information, add it. but often the message\n        // is the first part of the stack trace\n        errMsg = `${err.message}${errMsg ? ('\\n' + errMsg) : ''}`;\n      }\n      if (isErrorType(err, errors.ProxyRequestError)) {\n        actualErr = err.getActualError();\n      } else {\n        getLogger(driver, req.params.sessionId || newSessionId)\n          .debug(`Encountered internal error running command: ${errMsg}`);\n      }\n\n      [httpStatus, httpResBody] = getResponseForW3CError(actualErr);\n    }\n\n    // decode the response, which is either a string or json\n    if (_.isString(httpResBody)) {\n      res.status(httpStatus).send(httpResBody);\n    } else {\n      if (newSessionId) {\n        if (currentProtocol === PROTOCOLS.W3C) {\n          httpResBody.value.sessionId = newSessionId;\n        } else {\n          httpResBody.sessionId = newSessionId;\n        }\n      } else {\n        httpResBody.sessionId = req.params.sessionId || null;\n      }\n      // Don't include sessionId in W3C responses\n      if (currentProtocol === PROTOCOLS.W3C) {\n        delete httpResBody.sessionId;\n      }\n\n      httpResBody = formatStatus(httpResBody);\n      res.status(httpStatus).json(httpResBody);\n    }\n  };\n  // add the method to the app\n  app[method.toLowerCase()](path, (req, res) => {\n    B.resolve(asyncHandler(req, res)).done();\n  });\n}\n\nfunction driverShouldDoJwpProxy (driver, req, command) {\n  // drivers need to explicitly say when the proxy is active\n  if (!driver.proxyActive(req.params.sessionId)) {\n    return false;\n  }\n\n  // we should never proxy deleteSession because we need to give the containing\n  // driver an opportunity to clean itself up\n  if (command === DELETE_SESSION_COMMAND) {\n    return false;\n  }\n\n  // validate avoidance schema, and say we shouldn't proxy if anything in the\n  // avoid list matches our req\n  if (driver.proxyRouteIsAvoided(req.params.sessionId, req.method, req.originalUrl, req.body)) {\n    return false;\n  }\n\n  return true;\n}\n\nasync function doJwpProxy (driver, req, res) {\n  getLogger(driver, req.params.sessionId)\n    .info('Driver proxy active, passing request on via HTTP proxy');\n\n  // check that the inner driver has a proxy function\n  if (!driver.canProxy(req.params.sessionId)) {\n    throw new Error('Trying to proxy to a server but the driver is unable to proxy');\n  }\n  try {\n    await driver.executeCommand('proxyReqRes', req, res, req.params.sessionId);\n  } catch (err) {\n    if (isErrorType(err, errors.ProxyRequestError)) {\n      throw err;\n    } else {\n      throw new Error(`Could not proxy. Proxy error: ${err.message}`);\n    }\n  }\n}\n\n\nexport {\n  Protocol, routeConfiguringFunction, isSessionCommand,\n  driverShouldDoJwpProxy, determineProtocol, CREATE_SESSION_COMMAND,\n  DELETE_SESSION_COMMAND, GET_STATUS_COMMAND,\n};\n"],"file":"lib/protocol/protocol.js","sourceRoot":"../../.."}