@appium/base-driver 8.7.3 → 9.1.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 (121) hide show
  1. package/build/lib/basedriver/capabilities.d.ts +11 -163
  2. package/build/lib/basedriver/capabilities.d.ts.map +1 -1
  3. package/build/lib/basedriver/capabilities.js +355 -236
  4. package/build/lib/basedriver/capabilities.js.map +1 -1
  5. package/build/lib/basedriver/commands/event.d.ts +7 -6
  6. package/build/lib/basedriver/commands/event.d.ts.map +1 -1
  7. package/build/lib/basedriver/commands/event.js +55 -35
  8. package/build/lib/basedriver/commands/event.js.map +1 -1
  9. package/build/lib/basedriver/commands/execute.d.ts +7 -6
  10. package/build/lib/basedriver/commands/execute.d.ts.map +1 -1
  11. package/build/lib/basedriver/commands/execute.js +66 -58
  12. package/build/lib/basedriver/commands/execute.js.map +1 -1
  13. package/build/lib/basedriver/commands/find.d.ts +9 -7
  14. package/build/lib/basedriver/commands/find.d.ts.map +1 -1
  15. package/build/lib/basedriver/commands/find.js +102 -54
  16. package/build/lib/basedriver/commands/find.js.map +1 -1
  17. package/build/lib/basedriver/commands/index.d.ts +3 -7
  18. package/build/lib/basedriver/commands/index.d.ts.map +1 -1
  19. package/build/lib/basedriver/commands/index.js +30 -33
  20. package/build/lib/basedriver/commands/index.js.map +1 -1
  21. package/build/lib/basedriver/commands/log.d.ts +8 -9
  22. package/build/lib/basedriver/commands/log.d.ts.map +1 -1
  23. package/build/lib/basedriver/commands/log.js +54 -38
  24. package/build/lib/basedriver/commands/log.js.map +1 -1
  25. package/build/lib/basedriver/commands/session.d.ts +7 -6
  26. package/build/lib/basedriver/commands/session.d.ts.map +1 -1
  27. package/build/lib/basedriver/commands/session.js +46 -39
  28. package/build/lib/basedriver/commands/session.js.map +1 -1
  29. package/build/lib/basedriver/commands/settings.d.ts +7 -7
  30. package/build/lib/basedriver/commands/settings.d.ts.map +1 -1
  31. package/build/lib/basedriver/commands/settings.js +35 -28
  32. package/build/lib/basedriver/commands/settings.js.map +1 -1
  33. package/build/lib/basedriver/commands/timeout.d.ts +7 -5
  34. package/build/lib/basedriver/commands/timeout.d.ts.map +1 -1
  35. package/build/lib/basedriver/commands/timeout.js +144 -162
  36. package/build/lib/basedriver/commands/timeout.js.map +1 -1
  37. package/build/lib/basedriver/core.d.ts +6 -157
  38. package/build/lib/basedriver/core.d.ts.map +1 -1
  39. package/build/lib/basedriver/core.js +361 -230
  40. package/build/lib/basedriver/core.js.map +1 -1
  41. package/build/lib/basedriver/desired-caps.js +80 -110
  42. package/build/lib/basedriver/desired-caps.js.map +1 -1
  43. package/build/lib/basedriver/device-settings.js +57 -62
  44. package/build/lib/basedriver/device-settings.js.map +1 -1
  45. package/build/lib/basedriver/driver.d.ts +11 -262
  46. package/build/lib/basedriver/driver.d.ts.map +1 -1
  47. package/build/lib/basedriver/driver.js +362 -262
  48. package/build/lib/basedriver/driver.js.map +1 -1
  49. package/build/lib/basedriver/helpers.js +500 -495
  50. package/build/lib/basedriver/helpers.js.map +1 -1
  51. package/build/lib/basedriver/logger.d.ts +1 -1
  52. package/build/lib/basedriver/logger.d.ts.map +1 -1
  53. package/build/lib/basedriver/logger.js +5 -15
  54. package/build/lib/basedriver/logger.js.map +1 -1
  55. package/build/lib/constants.js +14 -14
  56. package/build/lib/constants.js.map +1 -1
  57. package/build/lib/express/crash.js +8 -15
  58. package/build/lib/express/crash.js.map +1 -1
  59. package/build/lib/express/express-logging.js +49 -59
  60. package/build/lib/express/express-logging.js.map +1 -1
  61. package/build/lib/express/idempotency.js +125 -177
  62. package/build/lib/express/idempotency.js.map +1 -1
  63. package/build/lib/express/logger.d.ts +1 -1
  64. package/build/lib/express/logger.d.ts.map +1 -1
  65. package/build/lib/express/logger.js +5 -15
  66. package/build/lib/express/logger.js.map +1 -1
  67. package/build/lib/express/middleware.js +82 -107
  68. package/build/lib/express/middleware.js.map +1 -1
  69. package/build/lib/express/server.d.ts +17 -5
  70. package/build/lib/express/server.d.ts.map +1 -1
  71. package/build/lib/express/server.js +259 -224
  72. package/build/lib/express/server.js.map +1 -1
  73. package/build/lib/express/static.js +64 -81
  74. package/build/lib/express/static.js.map +1 -1
  75. package/build/lib/express/websocket.js +115 -87
  76. package/build/lib/express/websocket.js.map +1 -1
  77. package/build/lib/helpers/capabilities.d.ts +1 -59
  78. package/build/lib/helpers/capabilities.d.ts.map +1 -1
  79. package/build/lib/helpers/capabilities.js +72 -69
  80. package/build/lib/helpers/capabilities.js.map +1 -1
  81. package/build/lib/index.js +64 -180
  82. package/build/lib/index.js.map +1 -1
  83. package/build/lib/jsonwp-proxy/protocol-converter.js +215 -227
  84. package/build/lib/jsonwp-proxy/protocol-converter.js.map +1 -1
  85. package/build/lib/jsonwp-proxy/proxy.d.ts.map +1 -1
  86. package/build/lib/jsonwp-proxy/proxy.js +355 -393
  87. package/build/lib/jsonwp-proxy/proxy.js.map +1 -1
  88. package/build/lib/jsonwp-status/status.js +119 -130
  89. package/build/lib/jsonwp-status/status.js.map +1 -1
  90. package/build/lib/protocol/errors.d.ts +135 -32
  91. package/build/lib/protocol/errors.d.ts.map +1 -1
  92. package/build/lib/protocol/errors.js +871 -919
  93. package/build/lib/protocol/errors.js.map +1 -1
  94. package/build/lib/protocol/helpers.js +37 -37
  95. package/build/lib/protocol/helpers.js.map +1 -1
  96. package/build/lib/protocol/index.js +22 -109
  97. package/build/lib/protocol/index.js.map +1 -1
  98. package/build/lib/protocol/protocol.js +394 -350
  99. package/build/lib/protocol/protocol.js.map +1 -1
  100. package/build/lib/protocol/routes.d.ts +1248 -4
  101. package/build/lib/protocol/routes.d.ts.map +1 -1
  102. package/build/lib/protocol/routes.js +972 -1327
  103. package/build/lib/protocol/routes.js.map +1 -1
  104. package/build/lib/protocol/validators.js +32 -39
  105. package/build/lib/protocol/validators.js.map +1 -1
  106. package/build/tsconfig.tsbuildinfo +1 -1
  107. package/lib/basedriver/capabilities.js +83 -39
  108. package/lib/basedriver/commands/event.js +10 -5
  109. package/lib/basedriver/commands/execute.js +14 -9
  110. package/lib/basedriver/commands/find.js +18 -12
  111. package/lib/basedriver/commands/index.js +21 -16
  112. package/lib/basedriver/commands/log.js +24 -18
  113. package/lib/basedriver/commands/session.js +10 -5
  114. package/lib/basedriver/commands/settings.js +9 -6
  115. package/lib/basedriver/commands/timeout.js +10 -4
  116. package/lib/basedriver/core.js +2 -3
  117. package/lib/basedriver/driver.js +12 -16
  118. package/lib/express/server.js +6 -3
  119. package/lib/protocol/errors.js +155 -44
  120. package/lib/protocol/routes.js +19 -7
  121. package/package.json +16 -18
@@ -1,253 +1,372 @@
1
1
  "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
6
- exports.STANDARD_CAPS = exports.PREFIXED_APPIUM_OPTS_CAP = exports.APPIUM_VENDOR_PREFIX = exports.APPIUM_OPTS_CAP = void 0;
7
- exports.findNonPrefixedCaps = findNonPrefixedCaps;
8
- exports.isStandardCap = isStandardCap;
9
- exports.mergeCaps = mergeCaps;
10
- exports.parseCaps = parseCaps;
11
- exports.processCapabilities = processCapabilities;
12
- exports.promoteAppiumOptions = promoteAppiumOptions;
13
- exports.stripAppiumPrefixes = stripAppiumPrefixes;
14
- exports.validateCaps = validateCaps;
15
-
16
- require("source-map-support/register");
17
-
18
- var _lodash = _interopRequireDefault(require("lodash"));
19
-
20
- var _desiredCaps = require("./desired-caps");
21
-
22
- var _support = require("@appium/support");
23
-
24
- var _logger = _interopRequireDefault(require("./logger"));
25
-
26
- var _errors = require("../protocol/errors");
27
-
28
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
29
-
2
+ // @ts-check
3
+ var __importDefault = (this && this.__importDefault) || function (mod) {
4
+ return (mod && mod.__esModule) ? mod : { "default": mod };
5
+ };
6
+ Object.defineProperty(exports, "__esModule", { value: true });
7
+ exports.PREFIXED_APPIUM_OPTS_CAP = exports.promoteAppiumOptions = exports.stripAppiumPrefixes = exports.isStandardCap = exports.findNonPrefixedCaps = exports.APPIUM_VENDOR_PREFIX = exports.mergeCaps = exports.validateCaps = exports.processCapabilities = exports.parseCaps = exports.STANDARD_CAPS = void 0;
8
+ const lodash_1 = __importDefault(require("lodash"));
9
+ const desired_caps_1 = require("./desired-caps");
10
+ const support_1 = require("@appium/support");
11
+ const logger_1 = __importDefault(require("./logger"));
12
+ const errors_1 = require("../protocol/errors");
30
13
  const APPIUM_VENDOR_PREFIX = 'appium:';
31
14
  exports.APPIUM_VENDOR_PREFIX = APPIUM_VENDOR_PREFIX;
32
- const APPIUM_OPTS_CAP = 'options';
33
- exports.APPIUM_OPTS_CAP = APPIUM_OPTS_CAP;
34
- const PREFIXED_APPIUM_OPTS_CAP = `${APPIUM_VENDOR_PREFIX}${APPIUM_OPTS_CAP}`;
15
+ const PREFIXED_APPIUM_OPTS_CAP = `${APPIUM_VENDOR_PREFIX}options`;
35
16
  exports.PREFIXED_APPIUM_OPTS_CAP = PREFIXED_APPIUM_OPTS_CAP;
36
-
17
+ /**
18
+ * Takes primary caps object and merges it into a secondary caps object.
19
+ * @template {Constraints} [T={}]
20
+ * @template {Constraints} [U={}]
21
+ * @param {Capabilities<T>} [primary]
22
+ * @param {Capabilities<U>} [secondary]
23
+ * @returns {Merge<Capabilities<T>, Capabilities<U>>}
24
+ * @see https://www.w3.org/TR/webdriver/#dfn-merging-capabilities)
25
+ */
37
26
  function mergeCaps(primary = {}, secondary = {}) {
38
- let result = { ...primary
39
- };
40
-
41
- for (let [name, value] of _lodash.default.toPairs(secondary)) {
42
- if (!_lodash.default.isUndefined(primary[name])) {
43
- throw new _errors.errors.InvalidArgumentError(`property '${name}' should not exist on both primary (${JSON.stringify(primary)}) and secondary (${JSON.stringify(secondary)}) object`);
44
- }
45
-
46
- result[name] = value;
47
- }
48
-
49
- return result;
27
+ let result = /** @type {Merge<Capabilities<T>, Capabilities<U>>} */ ({
28
+ ...primary,
29
+ });
30
+ for (let [name, value] of /** @type {[keyof typeof secondary, any]} */ (lodash_1.default.toPairs(secondary))) {
31
+ // Overwriting is not allowed. Primary and secondary must have different properties (w3c rule 4.4)
32
+ if (!lodash_1.default.isUndefined(primary[name])) {
33
+ throw new errors_1.errors.InvalidArgumentError(`property '${name}' should not exist on both primary (${JSON.stringify(primary)}) and secondary (${JSON.stringify(secondary)}) object`);
34
+ }
35
+ result[ /** @type {keyof typeof result} */(name)] = value;
36
+ }
37
+ return result;
50
38
  }
51
-
52
- function validateCaps(caps, constraints = {}, opts = {}) {
53
- let {
54
- skipPresenceConstraint
55
- } = opts;
56
-
57
- if (!_lodash.default.isPlainObject(caps)) {
58
- throw new _errors.errors.InvalidArgumentError(`must be a JSON object`);
59
- }
60
-
61
- constraints = _lodash.default.mapValues(constraints, skipPresenceConstraint ? constraint => _lodash.default.omit(constraint, 'presence') : constraint => {
62
- if (constraint.presence === true) {
63
- return { ..._lodash.default.omit(constraint, 'presence'),
64
- presence: {
65
- allowEmpty: false
39
+ exports.mergeCaps = mergeCaps;
40
+ // Validates caps against a set of constraints
41
+ /**
42
+ * @template {Constraints} [C={}]
43
+ * @param {Capabilities<C>} caps
44
+ * @param {C} [constraints]
45
+ * @param {ValidateCapsOpts} [opts]
46
+ * @returns {Capabilities<C>}
47
+ */
48
+ function validateCaps(caps, constraints = /** @type {C} */ ({}), opts = {}) {
49
+ let { skipPresenceConstraint } = opts;
50
+ if (!lodash_1.default.isPlainObject(caps)) {
51
+ throw new errors_1.errors.InvalidArgumentError(`must be a JSON object`);
52
+ }
53
+ // Remove the 'presence' constraint if we're not checking for it
54
+ constraints = /** @type {C} */ (lodash_1.default.mapValues(constraints, skipPresenceConstraint
55
+ ? /** @param {Constraint} constraint */
56
+ (constraint) => lodash_1.default.omit(constraint, 'presence')
57
+ : /** @param {Constraint} constraint */
58
+ (constraint) => {
59
+ if (constraint.presence === true) {
60
+ return { ...lodash_1.default.omit(constraint, 'presence'), presence: { allowEmpty: false } };
61
+ }
62
+ return constraint;
63
+ }));
64
+ const validationErrors = desired_caps_1.validator.validate(lodash_1.default.pickBy(caps, support_1.util.hasValue), constraints, {
65
+ fullMessages: false,
66
+ });
67
+ if (validationErrors) {
68
+ let message = [];
69
+ for (let [attribute, reasons] of lodash_1.default.toPairs(validationErrors)) {
70
+ for (let reason of reasons) {
71
+ message.push(`'${attribute}' ${reason}`);
72
+ }
66
73
  }
67
- };
68
- }
69
-
70
- return constraint;
71
- });
72
-
73
- const validationErrors = _desiredCaps.validator.validate(_lodash.default.pickBy(caps, _support.util.hasValue), constraints, {
74
- fullMessages: false
75
- });
76
-
77
- if (validationErrors) {
78
- let message = [];
79
-
80
- for (let [attribute, reasons] of _lodash.default.toPairs(validationErrors)) {
81
- for (let reason of reasons) {
82
- message.push(`'${attribute}' ${reason}`);
83
- }
84
- }
85
-
86
- throw new _errors.errors.InvalidArgumentError(message.join('; '));
87
- }
88
-
89
- return caps;
74
+ throw new errors_1.errors.InvalidArgumentError(message.join('; '));
75
+ }
76
+ // Return caps
77
+ return caps;
90
78
  }
91
-
92
- const STANDARD_CAPS = Object.freeze(new Set(['browserName', 'browserVersion', 'platformName', 'acceptInsecureCerts', 'pageLoadStrategy', 'proxy', 'setWindowRect', 'timeouts', 'unhandledPromptBehavior']));
93
- exports.STANDARD_CAPS = STANDARD_CAPS;
94
- const STANDARD_CAPS_LOWER = new Set([...STANDARD_CAPS].map(cap => cap.toLowerCase()));
95
-
79
+ exports.validateCaps = validateCaps;
80
+ /**
81
+ * Standard, non-prefixed capabilities
82
+ * @see https://www.w3.org/TR/webdriver/#dfn-table-of-standard-capabilities)
83
+ */
84
+ exports.STANDARD_CAPS = Object.freeze(new Set(
85
+ /** @type {StringKeyOf<import('@appium/types').StandardCapabilities>[]} */ ([
86
+ 'browserName',
87
+ 'browserVersion',
88
+ 'platformName',
89
+ 'acceptInsecureCerts',
90
+ 'pageLoadStrategy',
91
+ 'proxy',
92
+ 'setWindowRect',
93
+ 'timeouts',
94
+ 'unhandledPromptBehavior',
95
+ 'webSocketUrl',
96
+ ])));
97
+ const STANDARD_CAPS_LOWER = new Set([...exports.STANDARD_CAPS].map((cap) => cap.toLowerCase()));
98
+ /**
99
+ * @param {string} cap
100
+ * @returns {boolean}
101
+ */
96
102
  function isStandardCap(cap) {
97
- return STANDARD_CAPS_LOWER.has(cap.toLowerCase());
103
+ return STANDARD_CAPS_LOWER.has(cap.toLowerCase());
98
104
  }
99
-
105
+ exports.isStandardCap = isStandardCap;
106
+ /**
107
+ * If the 'appium:' prefix was provided and it's a valid capability, strip out the prefix
108
+ * @template {Constraints} [C={}]
109
+ * @param {import('@appium/types').NSCapabilities<C>} caps
110
+ * @see https://www.w3.org/TR/webdriver/#dfn-extension-capabilities
111
+ * @internal
112
+ * @returns {import('@appium/types').Capabilities<C>}
113
+ */
100
114
  function stripAppiumPrefixes(caps) {
101
- const [prefixedCaps, nonPrefixedCaps] = _lodash.default.partition(_lodash.default.keys(caps), cap => String(cap).startsWith(APPIUM_VENDOR_PREFIX));
102
-
103
- let strippedCaps = _lodash.default.pick(caps, nonPrefixedCaps);
104
-
105
- const badPrefixedCaps = [];
106
-
107
- for (let prefixedCap of prefixedCaps) {
108
- const strippedCapName = prefixedCap.substring(APPIUM_VENDOR_PREFIX.length);
109
-
110
- if (isStandardCap(strippedCapName)) {
111
- badPrefixedCaps.push(strippedCapName);
112
-
113
- if (_lodash.default.isNil(strippedCaps[strippedCapName])) {
114
- strippedCaps[strippedCapName] = caps[prefixedCap];
115
- } else {
116
- _logger.default.warn(`Ignoring capability '${prefixedCap}=${caps[prefixedCap]}' and ` + `using capability '${strippedCapName}=${strippedCaps[strippedCapName]}'`);
117
- }
118
- } else {
119
- strippedCaps[strippedCapName] = caps[prefixedCap];
120
- }
121
- }
122
-
123
- if (badPrefixedCaps.length > 0) {
124
- _logger.default.warn(`The capabilities ${JSON.stringify(badPrefixedCaps)} are standard capabilities and do not require "appium:" prefix`);
125
- }
126
-
127
- return strippedCaps;
115
+ // split into prefixed and non-prefixed.
116
+ // non-prefixed should be standard caps at this point
117
+ const [prefixedCaps, nonPrefixedCaps] = lodash_1.default.partition(lodash_1.default.keys(caps), (cap) => String(cap).startsWith(APPIUM_VENDOR_PREFIX));
118
+ // initialize this with the k/v pairs of the non-prefixed caps
119
+ let strippedCaps = /** @type {import('@appium/types').Capabilities<C>} */ (lodash_1.default.pick(caps, nonPrefixedCaps));
120
+ const badPrefixedCaps = [];
121
+ // Strip out the 'appium:' prefix
122
+ for (let prefixedCap of prefixedCaps) {
123
+ const strippedCapName = /** @type {StringKeyOf<import('@appium/types').Capabilities<C>>} */ (prefixedCap.substring(APPIUM_VENDOR_PREFIX.length));
124
+ // If it's standard capability that was prefixed, add it to an array of incorrectly prefixed capabilities
125
+ if (isStandardCap(strippedCapName)) {
126
+ badPrefixedCaps.push(strippedCapName);
127
+ if (lodash_1.default.isNil(strippedCaps[strippedCapName])) {
128
+ strippedCaps[strippedCapName] = caps[prefixedCap];
129
+ }
130
+ else {
131
+ logger_1.default.warn(`Ignoring capability '${prefixedCap}=${caps[prefixedCap]}' and ` +
132
+ `using capability '${strippedCapName}=${strippedCaps[strippedCapName]}'`);
133
+ }
134
+ }
135
+ else {
136
+ strippedCaps[strippedCapName] = caps[prefixedCap];
137
+ }
138
+ }
139
+ // If we found standard caps that were incorrectly prefixed, throw an exception (e.g.: don't accept 'appium:platformName', only accept just 'platformName')
140
+ if (badPrefixedCaps.length > 0) {
141
+ logger_1.default.warn(`The capabilities ${JSON.stringify(badPrefixedCaps)} are standard capabilities and do not require "appium:" prefix`);
142
+ }
143
+ return strippedCaps;
128
144
  }
129
-
130
- function findNonPrefixedCaps({
131
- alwaysMatch = {},
132
- firstMatch = []
133
- }) {
134
- return _lodash.default.chain([alwaysMatch, ...firstMatch]).reduce((unprefixedCaps, caps) => [...unprefixedCaps, ...Object.keys(caps).filter(cap => !cap.includes(':') && !isStandardCap(cap))], []).uniq().value();
145
+ exports.stripAppiumPrefixes = stripAppiumPrefixes;
146
+ /**
147
+ * Get an array of all the unprefixed caps that are being used in 'alwaysMatch' and all of the 'firstMatch' object
148
+ * @template {Constraints} [C={}]
149
+ * @param {import('@appium/types').W3CCapabilities<C>} caps A capabilities object
150
+ */
151
+ function findNonPrefixedCaps({ alwaysMatch = {}, firstMatch = [] }) {
152
+ return lodash_1.default.chain([alwaysMatch, ...firstMatch])
153
+ .reduce((unprefixedCaps, caps) => [
154
+ ...unprefixedCaps,
155
+ ...Object.keys(caps).filter((cap) => !cap.includes(':') && !isStandardCap(cap)),
156
+ ], [])
157
+ .uniq()
158
+ .value();
135
159
  }
136
-
137
- function parseCaps(caps, constraints = {}, shouldValidateCaps = true) {
138
- if (!_lodash.default.isPlainObject(caps)) {
139
- throw new _errors.errors.InvalidArgumentError('The capabilities argument was not valid for the following reason(s): "capabilities" must be a JSON object.');
140
- }
141
-
142
- let {
143
- alwaysMatch: requiredCaps = {},
144
- firstMatch: allFirstMatchCaps = [{}]
145
- } = caps;
146
-
147
- if (!_lodash.default.isArray(allFirstMatchCaps)) {
148
- throw new _errors.errors.InvalidArgumentError('The capabilities.firstMatch argument was not valid for the following reason(s): "capabilities.firstMatch" must be a JSON array or undefined');
149
- }
150
-
151
- if (allFirstMatchCaps.length === 0) {
152
- _logger.default.warn(`The firstMatch array in the given capabilities has no entries. Adding an empty entry fo rnow, ` + `but it will require one or more entries as W3C spec.`);
153
-
154
- allFirstMatchCaps.push({});
155
- }
156
-
157
- let nonPrefixedCaps = findNonPrefixedCaps(caps);
158
-
159
- if (!_lodash.default.isEmpty(nonPrefixedCaps)) {
160
- throw new _errors.errors.InvalidArgumentError(`All non-standard capabilities should have a vendor prefix. The following capabilities did not have one: ${nonPrefixedCaps}`);
161
- }
162
-
163
- let strippedRequiredCaps = stripAppiumPrefixes(requiredCaps);
164
- let strippedAllFirstMatchCaps = allFirstMatchCaps.map(stripAppiumPrefixes);
165
-
166
- if (shouldValidateCaps) {
167
- strippedRequiredCaps = validateCaps(strippedRequiredCaps, constraints, {
168
- skipPresenceConstraint: true
169
- });
170
- }
171
-
172
- const filteredConstraints = _lodash.default.omitBy(constraints, (_, key) => key in strippedRequiredCaps);
173
-
174
- let validationErrors = [];
175
-
176
- let validatedFirstMatchCaps = _lodash.default.compact(strippedAllFirstMatchCaps.map(firstMatchCaps => {
177
- try {
178
- return shouldValidateCaps ? validateCaps(firstMatchCaps, filteredConstraints) : firstMatchCaps;
179
- } catch (e) {
180
- validationErrors.push(e.message);
181
- }
182
- }));
183
-
184
- let matchedCaps = null;
185
-
186
- for (let firstMatchCaps of validatedFirstMatchCaps) {
187
- try {
188
- matchedCaps = mergeCaps(strippedRequiredCaps, firstMatchCaps);
189
-
190
- if (matchedCaps) {
191
- break;
192
- }
193
- } catch (err) {
194
- _logger.default.warn(err.message);
195
-
196
- validationErrors.push(err.message);
197
- }
198
- }
199
-
200
- return {
201
- requiredCaps,
202
- allFirstMatchCaps,
203
- validatedFirstMatchCaps,
204
- matchedCaps,
205
- validationErrors
206
- };
160
+ exports.findNonPrefixedCaps = findNonPrefixedCaps;
161
+ /**
162
+ * Parse capabilities
163
+ * @template {Constraints} [C={}]
164
+ * @param {import('@appium/types').W3CCapabilities<C>} caps
165
+ * @param {C} [constraints]
166
+ * @param {boolean} [shouldValidateCaps]
167
+ * @see https://www.w3.org/TR/webdriver/#processing-capabilities
168
+ */
169
+ function parseCaps(caps, constraints = /** @type {C} */ ({}), shouldValidateCaps = true) {
170
+ // If capabilities request is not an object, return error (#1.1)
171
+ if (!lodash_1.default.isPlainObject(caps)) {
172
+ throw new errors_1.errors.InvalidArgumentError('The capabilities argument was not valid for the following reason(s): "capabilities" must be a JSON object.');
173
+ }
174
+ // Let 'requiredCaps' be property named 'alwaysMatch' from capabilities request (#2)
175
+ // and 'allFirstMatchCaps' be property named 'firstMatch' from capabilities request (#3)
176
+ let { alwaysMatch: requiredCaps = {}, // If 'requiredCaps' is undefined, set it to an empty JSON object (#2.1)
177
+ firstMatch: allFirstMatchCaps = [{}], // If 'firstMatch' is undefined set it to a singleton list with one empty object (#3.1)
178
+ } = caps;
179
+ // Reject 'firstMatch' argument if it's not an array (#3.2)
180
+ if (!lodash_1.default.isArray(allFirstMatchCaps)) {
181
+ throw new errors_1.errors.InvalidArgumentError('The capabilities.firstMatch argument was not valid for the following reason(s): "capabilities.firstMatch" must be a JSON array or undefined');
182
+ }
183
+ // If an empty array as provided, we'll be forgiving and make it an array of one empty object
184
+ // In the future, reject 'firstMatch' argument if its array did not have one or more entries (#3.2)
185
+ if (allFirstMatchCaps.length === 0) {
186
+ logger_1.default.warn(`The firstMatch array in the given capabilities has no entries. Adding an empty entry fo rnow, ` +
187
+ `but it will require one or more entries as W3C spec.`);
188
+ allFirstMatchCaps.push({});
189
+ }
190
+ // Check for non-prefixed, non-standard capabilities and log warnings if they are found
191
+ let nonPrefixedCaps = findNonPrefixedCaps(caps);
192
+ if (!lodash_1.default.isEmpty(nonPrefixedCaps)) {
193
+ throw new errors_1.errors.InvalidArgumentError(`All non-standard capabilities should have a vendor prefix. The following capabilities did not have one: ${nonPrefixedCaps}`);
194
+ }
195
+ // Strip out the 'appium:' prefix from all
196
+ let strippedRequiredCaps = stripAppiumPrefixes(requiredCaps);
197
+ let strippedAllFirstMatchCaps = allFirstMatchCaps.map(stripAppiumPrefixes);
198
+ // Validate the requiredCaps. But don't validate 'presence' because if that constraint fails on 'alwaysMatch' it could still pass on one of the 'firstMatch' keys
199
+ if (shouldValidateCaps) {
200
+ strippedRequiredCaps = validateCaps(strippedRequiredCaps, constraints, {
201
+ skipPresenceConstraint: true,
202
+ });
203
+ }
204
+ // Remove the 'presence' constraint for any keys that are already present in 'requiredCaps'
205
+ // since we know that this constraint has already passed
206
+ const filteredConstraints = /** @type {C} */ (lodash_1.default.omitBy(constraints, (_, key) => key in strippedRequiredCaps));
207
+ // Validate all of the first match capabilities and return an array with only the valid caps (see spec #5)
208
+ /** @type {string[]} */
209
+ let validationErrors = [];
210
+ let validatedFirstMatchCaps = lodash_1.default.compact(strippedAllFirstMatchCaps.map(
211
+ /**
212
+ * @param {import('@appium/types').Capabilities<C>} firstMatchCaps
213
+ */
214
+ (firstMatchCaps) => {
215
+ try {
216
+ // Validate firstMatch caps
217
+ return shouldValidateCaps
218
+ ? validateCaps(firstMatchCaps, filteredConstraints)
219
+ : firstMatchCaps;
220
+ }
221
+ catch (e) {
222
+ validationErrors.push(e.message);
223
+ }
224
+ }));
225
+ // Try to merge requiredCaps with first match capabilities, break once it finds its first match (see spec #6)
226
+ let matchedCaps = null;
227
+ for (let firstMatchCaps of validatedFirstMatchCaps) {
228
+ try {
229
+ matchedCaps = mergeCaps(strippedRequiredCaps, firstMatchCaps);
230
+ if (matchedCaps) {
231
+ break;
232
+ }
233
+ }
234
+ catch (err) {
235
+ logger_1.default.warn(err.message);
236
+ validationErrors.push(err.message);
237
+ }
238
+ }
239
+ // Returns variables for testing purposes
240
+ return {
241
+ requiredCaps,
242
+ allFirstMatchCaps,
243
+ validatedFirstMatchCaps,
244
+ matchedCaps,
245
+ validationErrors,
246
+ };
207
247
  }
208
-
209
- function processCapabilities(w3cCaps, constraints = {}, shouldValidateCaps = true) {
210
- const {
211
- matchedCaps,
212
- validationErrors
213
- } = parseCaps(w3cCaps, constraints, shouldValidateCaps);
214
-
215
- if (!_support.util.hasValue(matchedCaps)) {
216
- if (_lodash.default.isArray(w3cCaps.firstMatch) && w3cCaps.firstMatch.length > 1) {
217
- throw new _errors.errors.InvalidArgumentError(`Could not find matching capabilities from ${JSON.stringify(w3cCaps)}:\n ${validationErrors.join('\n')}`);
218
- } else {
219
- throw new _errors.errors.InvalidArgumentError(validationErrors[0]);
220
- }
221
- }
222
-
223
- return matchedCaps ?? {};
248
+ exports.parseCaps = parseCaps;
249
+ // Calls parseCaps and just returns the matchedCaps variable
250
+ /**
251
+ * @template {Constraints} C
252
+ * @param {import('@appium/types').W3CCapabilities<C>} w3cCaps
253
+ * @param {C} [constraints]
254
+ * @param {boolean} [shouldValidateCaps]
255
+ * @returns {import('@appium/types').Capabilities<C>}
256
+ */
257
+ function processCapabilities(w3cCaps, constraints = /** @type {C} */ ({}), shouldValidateCaps = true) {
258
+ const { matchedCaps, validationErrors } = parseCaps(w3cCaps, constraints, shouldValidateCaps);
259
+ // If we found an error throw an exception
260
+ if (!support_1.util.hasValue(matchedCaps)) {
261
+ if (lodash_1.default.isArray(w3cCaps.firstMatch) && w3cCaps.firstMatch.length > 1) {
262
+ // If there was more than one 'firstMatch' cap, indicate that we couldn't find a matching capabilities set and show all the errors
263
+ throw new errors_1.errors.InvalidArgumentError(`Could not find matching capabilities from ${JSON.stringify(w3cCaps)}:\n ${validationErrors.join('\n')}`);
264
+ }
265
+ else {
266
+ // Otherwise, just show the singular error message
267
+ throw new errors_1.errors.InvalidArgumentError(validationErrors[0]);
268
+ }
269
+ }
270
+ return /** @type {Capabilities<C>} */ (matchedCaps ?? {});
224
271
  }
225
-
272
+ exports.processCapabilities = processCapabilities;
273
+ /**
274
+ * Return a copy of a capabilities object which has taken everything within the 'options'
275
+ * capability and promoted it to the top level.
276
+ *
277
+ * @template {Constraints} C
278
+ * @param {import('@appium/types').W3CCapabilities<C>} originalCaps
279
+ * @return {import('@appium/types').W3CCapabilities<C>} the capabilities with 'options' promoted if necessary
280
+ */
226
281
  function promoteAppiumOptions(originalCaps) {
227
- const appiumOptions = originalCaps[APPIUM_OPTS_CAP];
228
-
229
- if (!appiumOptions) {
230
- return originalCaps;
231
- }
232
-
233
- let caps = _lodash.default.cloneDeep(originalCaps);
234
-
235
- if (!_lodash.default.isPlainObject(appiumOptions)) {
236
- throw new _errors.errors.SessionNotCreatedError(`The ${APPIUM_OPTS_CAP} capability must be an object`);
237
- }
238
-
239
- const strippedAppiumOptions = stripAppiumPrefixes(appiumOptions);
240
-
241
- const overwrittenKeys = _lodash.default.intersection(Object.keys(caps), Object.keys(strippedAppiumOptions));
242
-
243
- if (overwrittenKeys.length > 0) {
244
- _logger.default.warn(`Found capabilities inside ${PREFIXED_APPIUM_OPTS_CAP} that will overwrite ` + `capabilities at the top level: ${JSON.stringify(overwrittenKeys)}`);
245
- }
246
-
247
- caps = { ...caps,
248
- ...strippedAppiumOptions
249
- };
250
- delete caps[APPIUM_OPTS_CAP];
251
- return caps;
282
+ const promoteForObject = (obj) => {
283
+ const appiumOptions = obj[PREFIXED_APPIUM_OPTS_CAP];
284
+ if (!appiumOptions) {
285
+ return obj;
286
+ }
287
+ if (!lodash_1.default.isPlainObject(appiumOptions)) {
288
+ throw new errors_1.errors.SessionNotCreatedError(`The ${PREFIXED_APPIUM_OPTS_CAP} capability must be an object`);
289
+ }
290
+ if (lodash_1.default.isEmpty(appiumOptions)) {
291
+ return obj;
292
+ }
293
+ logger_1.default.debug(`Found ${PREFIXED_APPIUM_OPTS_CAP} capability present; will promote items inside to caps`);
294
+ /**
295
+ * @param {string} capName
296
+ */
297
+ const shouldAddVendorPrefix = (capName) => !capName.startsWith(APPIUM_VENDOR_PREFIX);
298
+ const verifyIfAcceptable = (capName) => {
299
+ if (!lodash_1.default.isString(capName)) {
300
+ throw new errors_1.errors.SessionNotCreatedError(`Capability names in ${PREFIXED_APPIUM_OPTS_CAP} must be strings. '${capName}' is unexpected`);
301
+ }
302
+ if (isStandardCap(capName)) {
303
+ throw new errors_1.errors.SessionNotCreatedError(`${PREFIXED_APPIUM_OPTS_CAP} must only contain vendor-specific capabilties. '${capName}' is unexpected`);
304
+ }
305
+ return capName;
306
+ };
307
+ const preprocessedOptions = (0, lodash_1.default)(appiumOptions)
308
+ .mapKeys((value, key) => verifyIfAcceptable(key))
309
+ .mapKeys((value, key) => (shouldAddVendorPrefix(key) ? `${APPIUM_VENDOR_PREFIX}${key}` : key))
310
+ .value();
311
+ // warn if we are going to overwrite any keys on the base caps object
312
+ const overwrittenKeys = lodash_1.default.intersection(Object.keys(obj), Object.keys(preprocessedOptions));
313
+ if (overwrittenKeys.length > 0) {
314
+ logger_1.default.warn(`Found capabilities inside ${PREFIXED_APPIUM_OPTS_CAP} that will overwrite ` +
315
+ `capabilities at the top level: ${JSON.stringify(overwrittenKeys)}`);
316
+ }
317
+ return lodash_1.default.cloneDeep({
318
+ ...lodash_1.default.omit(obj, PREFIXED_APPIUM_OPTS_CAP),
319
+ ...preprocessedOptions,
320
+ });
321
+ };
322
+ const { alwaysMatch, firstMatch } = originalCaps;
323
+ const result = {};
324
+ if (lodash_1.default.isPlainObject(alwaysMatch)) {
325
+ result.alwaysMatch = promoteForObject(alwaysMatch);
326
+ }
327
+ else if ('alwaysMatch' in originalCaps) {
328
+ result.alwaysMatch = alwaysMatch;
329
+ }
330
+ if (lodash_1.default.isArray(firstMatch)) {
331
+ result.firstMatch = firstMatch.map(promoteForObject);
332
+ }
333
+ else if ('firstMatch' in originalCaps) {
334
+ result.firstMatch = firstMatch;
335
+ }
336
+ return result;
252
337
  }
253
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["APPIUM_VENDOR_PREFIX","APPIUM_OPTS_CAP","PREFIXED_APPIUM_OPTS_CAP","mergeCaps","primary","secondary","result","name","value","_","toPairs","isUndefined","errors","InvalidArgumentError","JSON","stringify","validateCaps","caps","constraints","opts","skipPresenceConstraint","isPlainObject","mapValues","constraint","omit","presence","allowEmpty","validationErrors","validator","validate","pickBy","util","hasValue","fullMessages","message","attribute","reasons","reason","push","join","STANDARD_CAPS","Object","freeze","Set","STANDARD_CAPS_LOWER","map","cap","toLowerCase","isStandardCap","has","stripAppiumPrefixes","prefixedCaps","nonPrefixedCaps","partition","keys","String","startsWith","strippedCaps","pick","badPrefixedCaps","prefixedCap","strippedCapName","substring","length","isNil","log","warn","findNonPrefixedCaps","alwaysMatch","firstMatch","chain","reduce","unprefixedCaps","filter","includes","uniq","parseCaps","shouldValidateCaps","requiredCaps","allFirstMatchCaps","isArray","isEmpty","strippedRequiredCaps","strippedAllFirstMatchCaps","filteredConstraints","omitBy","key","validatedFirstMatchCaps","compact","firstMatchCaps","e","matchedCaps","err","processCapabilities","w3cCaps","promoteAppiumOptions","originalCaps","appiumOptions","cloneDeep","SessionNotCreatedError","strippedAppiumOptions","overwrittenKeys","intersection"],"sources":["../../../lib/basedriver/capabilities.js"],"sourcesContent":["// @ts-check\n\nimport _ from 'lodash';\nimport {validator} from './desired-caps';\nimport {util} from '@appium/support';\nimport log from './logger';\nimport {errors} from '../protocol/errors';\n\nconst APPIUM_VENDOR_PREFIX = 'appium:';\nconst APPIUM_OPTS_CAP = 'options';\nconst PREFIXED_APPIUM_OPTS_CAP = `${APPIUM_VENDOR_PREFIX}${APPIUM_OPTS_CAP}`;\n\n/**\n * Takes primary caps object and merges it into a secondary caps object.\n * @template {Constraints} [T={}]\n * @template {Constraints} [U={}]\n * @param {Capabilities<T>} [primary]\n * @param {Capabilities<U>} [secondary]\n * @returns {import('type-fest').Merge<Capabilities<T>, Capabilities<U>>}\n * @see https://www.w3.org/TR/webdriver/#dfn-merging-capabilities)\n */\nfunction mergeCaps(primary = {}, secondary = {}) {\n  let result = /** @type {import('type-fest').Merge<Capabilities<T>, Capabilities<U>>} */ ({\n    ...primary,\n  });\n\n  for (let [name, value] of /** @type {[keyof typeof secondary, any]} */ (_.toPairs(secondary))) {\n    // Overwriting is not allowed. Primary and secondary must have different properties (w3c rule 4.4)\n    if (!_.isUndefined(primary[name])) {\n      throw new errors.InvalidArgumentError(\n        `property '${name}' should not exist on both primary (${JSON.stringify(\n          primary\n        )}) and secondary (${JSON.stringify(secondary)}) object`\n      );\n    }\n    result[/** @type {keyof typeof result} */ (name)] = value;\n  }\n\n  return result;\n}\n\n// Validates caps against a set of constraints\n/**\n * @template {Constraints} [C={}]\n * @param {Capabilities<C>} caps\n * @param {C} [constraints]\n * @param {ValidateCapsOpts} [opts]\n * @returns {Capabilities<C>}\n */\nfunction validateCaps(caps, constraints = /** @type {C} */ ({}), opts = {}) {\n  let {skipPresenceConstraint} = opts;\n\n  if (!_.isPlainObject(caps)) {\n    throw new errors.InvalidArgumentError(`must be a JSON object`);\n  }\n\n  // Remove the 'presence' constraint if we're not checking for it\n  constraints = /** @type {C} */ (\n    _.mapValues(\n      constraints,\n      skipPresenceConstraint\n        ? /** @param {Constraint} constraint */\n          (constraint) => _.omit(constraint, 'presence')\n        : /** @param {Constraint} constraint */\n          (constraint) => {\n            if (constraint.presence === true) {\n              return {..._.omit(constraint, 'presence'), presence: {allowEmpty: false}};\n            }\n            return constraint;\n          }\n    )\n  );\n\n  const validationErrors = validator.validate(_.pickBy(caps, util.hasValue), constraints, {\n    fullMessages: false,\n  });\n\n  if (validationErrors) {\n    let message = [];\n    for (let [attribute, reasons] of _.toPairs(validationErrors)) {\n      for (let reason of reasons) {\n        message.push(`'${attribute}' ${reason}`);\n      }\n    }\n    throw new errors.InvalidArgumentError(message.join('; '));\n  }\n\n  // Return caps\n  return caps;\n}\n\n/**\n * Standard, non-prefixed capabilities\n * @see https://www.w3.org/TR/webdriver/#dfn-table-of-standard-capabilities)\n */\nexport const STANDARD_CAPS = Object.freeze(\n  new Set(\n    /** @type {import('type-fest').StringKeyOf<import('@appium/types').StandardCapabilities>[]} */ ([\n      'browserName',\n      'browserVersion',\n      'platformName',\n      'acceptInsecureCerts',\n      'pageLoadStrategy',\n      'proxy',\n      'setWindowRect',\n      'timeouts',\n      'unhandledPromptBehavior',\n    ])\n  )\n);\n\nconst STANDARD_CAPS_LOWER = new Set([...STANDARD_CAPS].map((cap) => cap.toLowerCase()));\n\n/**\n * @param {string} cap\n * @returns {boolean}\n */\nfunction isStandardCap(cap) {\n  return STANDARD_CAPS_LOWER.has(cap.toLowerCase());\n}\n\n/**\n * If the 'appium:' prefix was provided and it's a valid capability, strip out the prefix\n * @template {Constraints} [C={}]\n * @param {import('@appium/types').NSCapabilities<C>} caps\n * @see https://www.w3.org/TR/webdriver/#dfn-extension-capabilities\n * @internal\n * @returns {import('@appium/types').Capabilities<C>}\n */\nfunction stripAppiumPrefixes(caps) {\n  // split into prefixed and non-prefixed.\n  // non-prefixed should be standard caps at this point\n  const [prefixedCaps, nonPrefixedCaps] = _.partition(_.keys(caps), (cap) =>\n    String(cap).startsWith(APPIUM_VENDOR_PREFIX)\n  );\n\n  // initialize this with the k/v pairs of the non-prefixed caps\n  let strippedCaps = /** @type {import('@appium/types').Capabilities<C>} */ (\n    _.pick(caps, nonPrefixedCaps)\n  );\n  const badPrefixedCaps = [];\n\n  // Strip out the 'appium:' prefix\n  for (let prefixedCap of prefixedCaps) {\n    const strippedCapName =\n      /** @type {import('type-fest').StringKeyOf<import('@appium/types').Capabilities<C>>} */ (\n        prefixedCap.substring(APPIUM_VENDOR_PREFIX.length)\n      );\n\n    // If it's standard capability that was prefixed, add it to an array of incorrectly prefixed capabilities\n    if (isStandardCap(strippedCapName)) {\n      badPrefixedCaps.push(strippedCapName);\n      if (_.isNil(strippedCaps[strippedCapName])) {\n        strippedCaps[strippedCapName] = caps[prefixedCap];\n      } else {\n        log.warn(\n          `Ignoring capability '${prefixedCap}=${caps[prefixedCap]}' and ` +\n            `using capability '${strippedCapName}=${strippedCaps[strippedCapName]}'`\n        );\n      }\n    } else {\n      strippedCaps[strippedCapName] = caps[prefixedCap];\n    }\n  }\n\n  // If we found standard caps that were incorrectly prefixed, throw an exception (e.g.: don't accept 'appium:platformName', only accept just 'platformName')\n  if (badPrefixedCaps.length > 0) {\n    log.warn(\n      `The capabilities ${JSON.stringify(\n        badPrefixedCaps\n      )} are standard capabilities and do not require \"appium:\" prefix`\n    );\n  }\n  return strippedCaps;\n}\n\n/**\n * Get an array of all the unprefixed caps that are being used in 'alwaysMatch' and all of the 'firstMatch' object\n * @template {Constraints} [C={}]\n * @param {import('@appium/types').W3CCapabilities<C>} caps A capabilities object\n */\nfunction findNonPrefixedCaps({alwaysMatch = {}, firstMatch = []}) {\n  return _.chain([alwaysMatch, ...firstMatch])\n    .reduce(\n      (unprefixedCaps, caps) => [\n        ...unprefixedCaps,\n        ...Object.keys(caps).filter((cap) => !cap.includes(':') && !isStandardCap(cap)),\n      ],\n      []\n    )\n    .uniq()\n    .value();\n}\n\n/**\n * Parse capabilities\n * @template {Constraints} [C={}]\n * @param {import('@appium/types').W3CCapabilities<C>} caps\n * @param {C} [constraints]\n * @param {boolean} [shouldValidateCaps]\n * @see https://www.w3.org/TR/webdriver/#processing-capabilities\n */\nfunction parseCaps(caps, constraints = /** @type {C} */ ({}), shouldValidateCaps = true) {\n  // If capabilities request is not an object, return error (#1.1)\n  if (!_.isPlainObject(caps)) {\n    throw new errors.InvalidArgumentError(\n      'The capabilities argument was not valid for the following reason(s): \"capabilities\" must be a JSON object.'\n    );\n  }\n\n  // Let 'requiredCaps' be property named 'alwaysMatch' from capabilities request (#2)\n  // and 'allFirstMatchCaps' be property named 'firstMatch' from capabilities request (#3)\n  let {\n    alwaysMatch: requiredCaps = {}, // If 'requiredCaps' is undefined, set it to an empty JSON object (#2.1)\n    firstMatch: allFirstMatchCaps = [{}], // If 'firstMatch' is undefined set it to a singleton list with one empty object (#3.1)\n  } = caps;\n\n  // Reject 'firstMatch' argument if it's not an array (#3.2)\n  if (!_.isArray(allFirstMatchCaps)) {\n    throw new errors.InvalidArgumentError(\n      'The capabilities.firstMatch argument was not valid for the following reason(s): \"capabilities.firstMatch\" must be a JSON array or undefined'\n    );\n  }\n\n  // If an empty array as provided, we'll be forgiving and make it an array of one empty object\n  // In the future, reject 'firstMatch' argument if its array did not have one or more entries (#3.2)\n  if (allFirstMatchCaps.length === 0) {\n    log.warn(\n      `The firstMatch array in the given capabilities has no entries. Adding an empty entry fo rnow, ` +\n        `but it will require one or more entries as W3C spec.`\n    );\n    allFirstMatchCaps.push({});\n  }\n\n  // Check for non-prefixed, non-standard capabilities and log warnings if they are found\n  let nonPrefixedCaps = findNonPrefixedCaps(caps);\n  if (!_.isEmpty(nonPrefixedCaps)) {\n    throw new errors.InvalidArgumentError(\n      `All non-standard capabilities should have a vendor prefix. The following capabilities did not have one: ${nonPrefixedCaps}`\n    );\n  }\n\n  // Strip out the 'appium:' prefix from all\n  let strippedRequiredCaps = stripAppiumPrefixes(requiredCaps);\n  let strippedAllFirstMatchCaps = allFirstMatchCaps.map(stripAppiumPrefixes);\n\n  // Validate the requiredCaps. But don't validate 'presence' because if that constraint fails on 'alwaysMatch' it could still pass on one of the 'firstMatch' keys\n  if (shouldValidateCaps) {\n    strippedRequiredCaps = validateCaps(strippedRequiredCaps, constraints, {\n      skipPresenceConstraint: true,\n    });\n  }\n  // Remove the 'presence' constraint for any keys that are already present in 'requiredCaps'\n  // since we know that this constraint has already passed\n  const filteredConstraints = /** @type {C} */ (\n    _.omitBy(constraints, (_, key) => key in strippedRequiredCaps)\n  );\n\n  // Validate all of the first match capabilities and return an array with only the valid caps (see spec #5)\n  /** @type {string[]} */\n  let validationErrors = [];\n  let validatedFirstMatchCaps = _.compact(\n    strippedAllFirstMatchCaps.map(\n      /**\n       * @param {import('@appium/types').Capabilities<C>} firstMatchCaps\n       */\n      (firstMatchCaps) => {\n        try {\n          // Validate firstMatch caps\n          return shouldValidateCaps\n            ? validateCaps(firstMatchCaps, filteredConstraints)\n            : firstMatchCaps;\n        } catch (e) {\n          validationErrors.push(e.message);\n        }\n      }\n    )\n  );\n\n  // Try to merge requiredCaps with first match capabilities, break once it finds its first match (see spec #6)\n  let matchedCaps = null;\n  for (let firstMatchCaps of validatedFirstMatchCaps) {\n    try {\n      matchedCaps = mergeCaps(strippedRequiredCaps, firstMatchCaps);\n      if (matchedCaps) {\n        break;\n      }\n    } catch (err) {\n      log.warn(err.message);\n      validationErrors.push(err.message);\n    }\n  }\n\n  // Returns variables for testing purposes\n  return {\n    requiredCaps,\n    allFirstMatchCaps,\n    validatedFirstMatchCaps,\n    matchedCaps,\n    validationErrors,\n  };\n}\n\n// Calls parseCaps and just returns the matchedCaps variable\n/**\n * @template {Constraints} C\n * @param {import('@appium/types').W3CCapabilities<C>} w3cCaps\n * @param {C} [constraints]\n * @param {boolean} [shouldValidateCaps]\n * @returns {import('@appium/types').Capabilities<C>}\n */\nfunction processCapabilities(\n  w3cCaps,\n  constraints = /** @type {C} */ ({}),\n  shouldValidateCaps = true\n) {\n  const {matchedCaps, validationErrors} = parseCaps(w3cCaps, constraints, shouldValidateCaps);\n\n  // If we found an error throw an exception\n  if (!util.hasValue(matchedCaps)) {\n    if (_.isArray(w3cCaps.firstMatch) && w3cCaps.firstMatch.length > 1) {\n      // If there was more than one 'firstMatch' cap, indicate that we couldn't find a matching capabilities set and show all the errors\n      throw new errors.InvalidArgumentError(\n        `Could not find matching capabilities from ${JSON.stringify(\n          w3cCaps\n        )}:\\n ${validationErrors.join('\\n')}`\n      );\n    } else {\n      // Otherwise, just show the singular error message\n      throw new errors.InvalidArgumentError(validationErrors[0]);\n    }\n  }\n\n  return /** @type {Capabilities<C>} */ (matchedCaps ?? {});\n}\n\n/**\n * Return a copy of a capabilities object which has taken everything within the 'options'\n * capability and promoted it to the top level. Note that this function is assumed to be run after\n * all vendor prefixes have already been stripped from the top level. So we are dealing with e.g.\n * 'options' and not 'appium:options' at this point. Any prefixes _inside_ the 'options' capability\n * will themselves be stripped. This is designed as an internal function, not one to operate on\n * user-constructed capabilities.\n *\n * @param {object} originalCaps - the capabilities to analyze and promote from 'options'\n * @return {object!} - the capabilities with 'options' promoted if necessary\n */\nfunction promoteAppiumOptions(originalCaps) {\n  const appiumOptions = originalCaps[APPIUM_OPTS_CAP];\n  if (!appiumOptions) {\n    return originalCaps;\n  }\n\n  let caps = _.cloneDeep(originalCaps);\n  if (!_.isPlainObject(appiumOptions)) {\n    throw new errors.SessionNotCreatedError(`The ${APPIUM_OPTS_CAP} capability must be an object`);\n  }\n\n  // first get rid of any prefixes inside appium:options\n  const strippedAppiumOptions = stripAppiumPrefixes(appiumOptions);\n  // warn if we are going to overwrite any keys on the base caps object\n  const overwrittenKeys = _.intersection(Object.keys(caps), Object.keys(strippedAppiumOptions));\n  if (overwrittenKeys.length > 0) {\n    log.warn(\n      `Found capabilities inside ${PREFIXED_APPIUM_OPTS_CAP} that will overwrite ` +\n        `capabilities at the top level: ${JSON.stringify(overwrittenKeys)}`\n    );\n  }\n\n  // now just apply them to the main caps object\n  caps = {...caps, ...strippedAppiumOptions};\n\n  // and remove all traces of the options cap\n  delete caps[APPIUM_OPTS_CAP];\n  return caps;\n}\n\nexport {\n  parseCaps,\n  processCapabilities,\n  validateCaps,\n  mergeCaps,\n  APPIUM_VENDOR_PREFIX,\n  APPIUM_OPTS_CAP,\n  findNonPrefixedCaps,\n  isStandardCap,\n  stripAppiumPrefixes,\n  promoteAppiumOptions,\n  PREFIXED_APPIUM_OPTS_CAP,\n};\n\n/**\n * @typedef {import('@appium/types').Constraints} Constraints\n * @typedef {import('@appium/types').Constraint} Constraint\n * @typedef {import('@appium/types').StringRecord} StringRecord\n * @typedef {import('@appium/types').BaseDriverCapConstraints} BaseDriverCapConstraints\n */\n\n/**\n * @typedef ValidateCapsOpts\n * @property {boolean} [skipPresenceConstraint] - if true, skip the presence constraint\n */\n\n/**\n * @template {Constraints} [C=BaseDriverCapConstraints]\n * @template {StringRecord|void} [Extra=void]\n * @typedef {import('@appium/types').NSCapabilities<C, Extra>} NSCapabilities\n */\n\n/**\n * @template {Constraints} [C=BaseDriverCapConstraints]\n * @template {StringRecord|void} [Extra=void]\n * @typedef {import('@appium/types').Capabilities<C, Extra>} Capabilities\n */\n\n/**\n * @template {Constraints} [C=BaseDriverCapConstraints]\n * @template {StringRecord|void} [Extra=void]\n * @typedef {import('@appium/types').W3CCapabilities<C, Extra>} W3CCapabilities\n */\n"],"mappings":"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"}
338
+ exports.promoteAppiumOptions = promoteAppiumOptions;
339
+ /**
340
+ * @typedef {import('@appium/types').Constraints} Constraints
341
+ * @typedef {import('@appium/types').Constraint} Constraint
342
+ * @typedef {import('@appium/types').StringRecord} StringRecord
343
+ * @typedef {import('@appium/types').BaseDriverCapConstraints} BaseDriverCapConstraints
344
+ */
345
+ /**
346
+ * @typedef ValidateCapsOpts
347
+ * @property {boolean} [skipPresenceConstraint] - if true, skip the presence constraint
348
+ */
349
+ /**
350
+ * @template {Constraints} [C=BaseDriverCapConstraints]
351
+ * @template {StringRecord|void} [Extra=void]
352
+ * @typedef {import('@appium/types').NSCapabilities<C, Extra>} NSCapabilities
353
+ */
354
+ /**
355
+ * @template {Constraints} [C=BaseDriverCapConstraints]
356
+ * @template {StringRecord|void} [Extra=void]
357
+ * @typedef {import('@appium/types').Capabilities<C, Extra>} Capabilities
358
+ */
359
+ /**
360
+ * @template {Constraints} [C=BaseDriverCapConstraints]
361
+ * @template {StringRecord|void} [Extra=void]
362
+ * @typedef {import('@appium/types').W3CCapabilities<C, Extra>} W3CCapabilities
363
+ */
364
+ /**
365
+ * @template T,U
366
+ * @typedef {import('type-fest').Merge<T,U>} Merge
367
+ */
368
+ /**
369
+ * @template T
370
+ * @typedef {import('type-fest').StringKeyOf<T>} StringKeyOf
371
+ */
372
+ //# sourceMappingURL=capabilities.js.map