appium 2.0.0-beta.25 → 2.0.0-beta.28

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 (191) hide show
  1. package/build/lib/appium.d.ts +215 -0
  2. package/build/lib/appium.d.ts.map +1 -0
  3. package/build/lib/appium.js +94 -101
  4. package/build/lib/cli/args.d.ts +20 -0
  5. package/build/lib/cli/args.d.ts.map +1 -0
  6. package/build/lib/cli/args.js +19 -39
  7. package/build/lib/cli/driver-command.d.ts +36 -0
  8. package/build/lib/cli/driver-command.d.ts.map +1 -0
  9. package/build/lib/cli/driver-command.js +10 -13
  10. package/build/lib/cli/extension-command.d.ts +345 -0
  11. package/build/lib/cli/extension-command.d.ts.map +1 -0
  12. package/build/lib/cli/extension-command.js +117 -94
  13. package/build/lib/cli/extension.d.ts +14 -0
  14. package/build/lib/cli/extension.d.ts.map +1 -0
  15. package/build/lib/cli/extension.js +14 -22
  16. package/build/lib/cli/parser.d.ts +79 -0
  17. package/build/lib/cli/parser.d.ts.map +1 -0
  18. package/build/lib/cli/parser.js +9 -19
  19. package/build/lib/cli/plugin-command.d.ts +39 -0
  20. package/build/lib/cli/plugin-command.d.ts.map +1 -0
  21. package/build/lib/cli/plugin-command.js +9 -14
  22. package/build/lib/cli/utils.d.ts +29 -0
  23. package/build/lib/cli/utils.d.ts.map +1 -0
  24. package/build/lib/cli/utils.js +2 -4
  25. package/build/lib/config-file.d.ts +100 -0
  26. package/build/lib/config-file.d.ts.map +1 -0
  27. package/build/lib/config-file.js +2 -4
  28. package/build/lib/config.d.ts +40 -0
  29. package/build/lib/config.d.ts.map +1 -0
  30. package/build/lib/config.js +8 -7
  31. package/build/lib/constants.d.ts +48 -0
  32. package/build/lib/constants.d.ts.map +1 -0
  33. package/build/lib/constants.js +60 -0
  34. package/build/lib/extension/driver-config.d.ts +84 -0
  35. package/build/lib/extension/driver-config.d.ts.map +1 -0
  36. package/build/lib/extension/driver-config.js +190 -0
  37. package/build/lib/extension/extension-config.d.ts +170 -0
  38. package/build/lib/extension/extension-config.d.ts.map +1 -0
  39. package/build/lib/extension/extension-config.js +297 -0
  40. package/build/lib/extension/index.d.ts +39 -0
  41. package/build/lib/extension/index.d.ts.map +1 -0
  42. package/build/lib/extension/index.js +77 -0
  43. package/build/lib/extension/manifest.d.ts +174 -0
  44. package/build/lib/extension/manifest.d.ts.map +1 -0
  45. package/build/lib/extension/manifest.js +246 -0
  46. package/build/lib/extension/package-changed.d.ts +11 -0
  47. package/build/lib/extension/package-changed.d.ts.map +1 -0
  48. package/build/lib/extension/package-changed.js +68 -0
  49. package/build/lib/extension/plugin-config.d.ts +62 -0
  50. package/build/lib/extension/plugin-config.d.ts.map +1 -0
  51. package/build/lib/extension/plugin-config.js +87 -0
  52. package/build/lib/grid-register.d.ts +10 -0
  53. package/build/lib/grid-register.d.ts.map +1 -0
  54. package/build/lib/grid-register.js +2 -4
  55. package/build/lib/logger.d.ts +3 -0
  56. package/build/lib/logger.d.ts.map +1 -0
  57. package/build/lib/logger.js +2 -4
  58. package/build/lib/logsink.d.ts +4 -0
  59. package/build/lib/logsink.d.ts.map +1 -0
  60. package/build/lib/logsink.js +2 -4
  61. package/build/lib/main.d.ts +51 -0
  62. package/build/lib/main.d.ts.map +1 -0
  63. package/build/lib/main.js +40 -68
  64. package/build/lib/schema/arg-spec.d.ts +143 -0
  65. package/build/lib/schema/arg-spec.d.ts.map +1 -0
  66. package/build/lib/schema/arg-spec.js +11 -14
  67. package/build/lib/schema/cli-args.d.ts +19 -0
  68. package/build/lib/schema/cli-args.d.ts.map +1 -0
  69. package/build/lib/schema/cli-args.js +2 -4
  70. package/build/lib/schema/cli-transformers.d.ts +5 -0
  71. package/build/lib/schema/cli-transformers.d.ts.map +1 -0
  72. package/build/lib/schema/cli-transformers.js +2 -4
  73. package/build/lib/schema/index.d.ts +3 -0
  74. package/build/lib/schema/index.d.ts.map +1 -0
  75. package/build/lib/schema/index.js +2 -4
  76. package/build/lib/schema/keywords.d.ts +24 -0
  77. package/build/lib/schema/keywords.d.ts.map +1 -0
  78. package/build/lib/schema/keywords.js +2 -4
  79. package/build/lib/schema/schema.d.ts +259 -0
  80. package/build/lib/schema/schema.d.ts.map +1 -0
  81. package/build/lib/schema/schema.js +57 -39
  82. package/build/lib/utils.d.ts +66 -0
  83. package/build/lib/utils.d.ts.map +1 -0
  84. package/build/lib/utils.js +6 -35
  85. package/build/tsconfig.tsbuildinfo +1 -0
  86. package/lib/appium.js +188 -117
  87. package/lib/cli/args.js +19 -24
  88. package/lib/cli/driver-command.js +19 -8
  89. package/lib/cli/extension-command.js +314 -184
  90. package/lib/cli/extension.js +18 -16
  91. package/lib/cli/parser.js +7 -16
  92. package/lib/cli/plugin-command.js +16 -7
  93. package/lib/cli/utils.js +1 -1
  94. package/lib/config-file.js +6 -7
  95. package/lib/config.js +17 -12
  96. package/lib/constants.js +78 -0
  97. package/lib/extension/driver-config.js +249 -0
  98. package/lib/extension/extension-config.js +458 -0
  99. package/lib/extension/index.js +102 -0
  100. package/lib/extension/manifest.js +486 -0
  101. package/lib/extension/package-changed.js +63 -0
  102. package/lib/extension/plugin-config.js +113 -0
  103. package/lib/grid-register.js +4 -4
  104. package/lib/logsink.js +4 -0
  105. package/lib/main.js +54 -92
  106. package/lib/schema/arg-spec.js +11 -7
  107. package/lib/schema/cli-args.js +1 -1
  108. package/lib/schema/cli-transformers.js +0 -1
  109. package/lib/schema/keywords.js +1 -2
  110. package/lib/schema/schema.js +62 -31
  111. package/lib/utils.js +48 -45
  112. package/package.json +30 -24
  113. package/{postinstall.js → scripts/postinstall.js} +1 -1
  114. package/types/appium-manifest.d.ts +61 -0
  115. package/types/cli.d.ts +134 -0
  116. package/types/extension.d.ts +56 -0
  117. package/types/external-manifest.d.ts +58 -0
  118. package/types/index.d.ts +7 -0
  119. package/bin/ios-webkit-debug-proxy-launcher.js +0 -71
  120. package/build/check-npm-pack-files.js +0 -23
  121. package/build/commands-yml/parse.js +0 -319
  122. package/build/commands-yml/validator.js +0 -130
  123. package/build/index.js +0 -19
  124. package/build/lib/appium-config.schema.json +0 -0
  125. package/build/lib/cli/npm.js +0 -220
  126. package/build/lib/driver-config.js +0 -100
  127. package/build/lib/drivers.js +0 -100
  128. package/build/lib/ext-config-io.js +0 -165
  129. package/build/lib/extension-config.js +0 -320
  130. package/build/lib/plugin-config.js +0 -69
  131. package/build/lib/plugins.js +0 -18
  132. package/build/lib/schema/appium-config-schema.js +0 -253
  133. package/build/postinstall.js +0 -90
  134. package/build/test/cli/cli-e2e-specs.js +0 -221
  135. package/build/test/cli/cli-helpers.js +0 -86
  136. package/build/test/cli/cli-specs.js +0 -71
  137. package/build/test/cli/fixtures/test-driver/package.json +0 -27
  138. package/build/test/cli/schema-args-specs.js +0 -48
  139. package/build/test/cli/schema-e2e-specs.js +0 -47
  140. package/build/test/config-e2e-specs.js +0 -112
  141. package/build/test/config-file-e2e-specs.js +0 -191
  142. package/build/test/config-file-specs.js +0 -281
  143. package/build/test/config-specs.js +0 -258
  144. package/build/test/driver-e2e-specs.js +0 -435
  145. package/build/test/driver-specs.js +0 -386
  146. package/build/test/ext-config-io-specs.js +0 -181
  147. package/build/test/extension-config-specs.js +0 -365
  148. package/build/test/fixtures/allow-feat.txt +0 -5
  149. package/build/test/fixtures/caps.json +0 -3
  150. package/build/test/fixtures/config/allow-insecure.txt +0 -3
  151. package/build/test/fixtures/config/appium.config.bad-nodeconfig.json +0 -5
  152. package/build/test/fixtures/config/appium.config.bad.json +0 -32
  153. package/build/test/fixtures/config/appium.config.ext-good.json +0 -9
  154. package/build/test/fixtures/config/appium.config.ext-unknown-props.json +0 -10
  155. package/build/test/fixtures/config/appium.config.good.js +0 -40
  156. package/build/test/fixtures/config/appium.config.good.json +0 -33
  157. package/build/test/fixtures/config/appium.config.good.yaml +0 -30
  158. package/build/test/fixtures/config/appium.config.invalid.json +0 -31
  159. package/build/test/fixtures/config/appium.config.security-array.json +0 -5
  160. package/build/test/fixtures/config/appium.config.security-delimited.json +0 -5
  161. package/build/test/fixtures/config/appium.config.security-path.json +0 -5
  162. package/build/test/fixtures/config/driver-fake.config.json +0 -8
  163. package/build/test/fixtures/config/nodeconfig.json +0 -3
  164. package/build/test/fixtures/config/plugin-fake.config.json +0 -0
  165. package/build/test/fixtures/default-args.js +0 -35
  166. package/build/test/fixtures/deny-feat.txt +0 -5
  167. package/build/test/fixtures/driver.schema.js +0 -20
  168. package/build/test/fixtures/extensions.yaml +0 -27
  169. package/build/test/fixtures/flattened-schema.js +0 -532
  170. package/build/test/fixtures/plugin.schema.js +0 -20
  171. package/build/test/fixtures/schema-with-extensions.js +0 -28
  172. package/build/test/grid-register-specs.js +0 -74
  173. package/build/test/helpers.js +0 -75
  174. package/build/test/logger-specs.js +0 -76
  175. package/build/test/npm-specs.js +0 -20
  176. package/build/test/parser-specs.js +0 -319
  177. package/build/test/plugin-e2e-specs.js +0 -316
  178. package/build/test/schema/arg-spec-specs.js +0 -70
  179. package/build/test/schema/cli-args-specs.js +0 -408
  180. package/build/test/schema/schema-specs.js +0 -407
  181. package/build/test/utils-specs.js +0 -288
  182. package/lib/cli/npm.js +0 -251
  183. package/lib/driver-config.js +0 -101
  184. package/lib/drivers.js +0 -84
  185. package/lib/ext-config-io.js +0 -287
  186. package/lib/extension-config.js +0 -366
  187. package/lib/plugin-config.js +0 -63
  188. package/lib/plugins.js +0 -13
  189. package/lib/schema/appium-config-schema.js +0 -287
  190. package/types/appium-config.d.ts +0 -197
  191. package/types/types.d.ts +0 -206
@@ -7,18 +7,12 @@ Object.defineProperty(exports, "__esModule", {
7
7
  });
8
8
  exports.NoDriverProxyCommandError = exports.AppiumDriver = void 0;
9
9
 
10
- var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
11
-
12
10
  require("source-map-support/register");
13
11
 
14
12
  var _lodash = _interopRequireDefault(require("lodash"));
15
13
 
16
- var _logger = _interopRequireDefault(require("./logger"));
17
-
18
14
  var _config = require("./config");
19
15
 
20
- var _drivers = require("./drivers");
21
-
22
16
  var _baseDriver = require("@appium/base-driver");
23
17
 
24
18
  var _asyncLock = _interopRequireDefault(require("async-lock"));
@@ -29,7 +23,7 @@ var _support = require("@appium/support");
29
23
 
30
24
  var _schema = require("./schema");
31
25
 
32
- const desiredCapabilityConstraints = {
26
+ const desiredCapabilityConstraints = Object.freeze({
33
27
  automationName: {
34
28
  presence: true,
35
29
  isString: true
@@ -38,29 +32,41 @@ const desiredCapabilityConstraints = {
38
32
  presence: true,
39
33
  isString: true
40
34
  }
41
- };
35
+ });
42
36
  const sessionsListGuard = new _asyncLock.default();
43
37
  const pendingDriversGuard = new _asyncLock.default();
44
38
 
45
- class AppiumDriver extends _baseDriver.BaseDriver {
46
- constructor(args) {
47
- if (args.tmpDir) {
48
- process.env.APPIUM_TMP_DIR = args.tmpDir;
39
+ class AppiumDriver extends _baseDriver.DriverCore {
40
+ sessions = {};
41
+ pendingDrivers = {};
42
+ newCommandTimeoutMs = 0;
43
+ pluginClasses = [];
44
+ sessionPlugins = {};
45
+ sessionlessPlugins = [];
46
+ driverConfig;
47
+ server;
48
+
49
+ constructor(opts) {
50
+ if (opts.tmpDir) {
51
+ process.env.APPIUM_TMP_DIR = opts.tmpDir;
49
52
  }
50
53
 
51
- super(args);
52
- (0, _defineProperty2.default)(this, "driverConfig", void 0);
53
- (0, _defineProperty2.default)(this, "server", void 0);
54
+ super(opts);
54
55
  this.desiredCapConstraints = desiredCapabilityConstraints;
55
- this.newCommandTimeoutMs = 0;
56
- this.args = { ...args
56
+ this.args = { ...opts
57
57
  };
58
- this.sessions = {};
59
- this.pendingDrivers = {};
60
- this.pluginClasses = [];
61
- this.sessionPlugins = {};
62
- this.sessionlessPlugins = [];
63
- (0, _config.updateBuildInfo)();
58
+ (0, _config.updateBuildInfo)().catch(err => {
59
+ this.log.debug(err);
60
+ });
61
+ }
62
+
63
+ get log() {
64
+ if (!this._log) {
65
+ const instanceName = `${this.constructor.name}@${_support.node.getObjectId(this).substring(0, 4)}`;
66
+ this._log = _support.logger.getLogger(instanceName);
67
+ }
68
+
69
+ return this._log;
64
70
  }
65
71
 
66
72
  get isCommandsQueueEnabled() {
@@ -91,17 +97,10 @@ class AppiumDriver extends _baseDriver.BaseDriver {
91
97
  }
92
98
 
93
99
  printNewSessionAnnouncement(driverName, driverVersion, driverBaseVersion) {
94
- _logger.default.info(driverVersion ? `Appium v${_config.APPIUM_VER} creating new ${driverName} (v${driverVersion}) session` : `Appium v${_config.APPIUM_VER} creating new ${driverName} session`);
95
-
96
- _logger.default.info(`Checking BaseDriver versions for Appium and ${driverName}`);
97
-
98
- _logger.default.info(AppiumDriver.baseVersion ? `Appium's BaseDriver version is ${AppiumDriver.baseVersion}` : `Could not determine Appium's BaseDriver version`);
99
-
100
- _logger.default.info(driverBaseVersion ? `${driverName}'s BaseDriver version is ${driverBaseVersion}` : `Could not determine ${driverName}'s BaseDriver version`);
101
- }
102
-
103
- _findMatchingDriver(...args) {
104
- return (0, _drivers.findMatchingDriver)(...args);
100
+ this.log.info(driverVersion ? `Appium v${_config.APPIUM_VER} creating new ${driverName} (v${driverVersion}) session` : `Appium v${_config.APPIUM_VER} creating new ${driverName} session`);
101
+ this.log.info(`Checking BaseDriver versions for Appium and ${driverName}`);
102
+ this.log.info(AppiumDriver.baseVersion ? `Appium's BaseDriver version is ${AppiumDriver.baseVersion}` : `Could not determine Appium's BaseDriver version`);
103
+ this.log.info(driverBaseVersion ? `${driverName}'s BaseDriver version is ${driverBaseVersion}` : `Could not determine ${driverName}'s BaseDriver version`);
105
104
  }
106
105
 
107
106
  assignCliArgsToExtension(extType, extName, extInstance) {
@@ -119,17 +118,24 @@ class AppiumDriver extends _baseDriver.BaseDriver {
119
118
  }
120
119
  }
121
120
 
122
- async createSession(jsonwpCaps, reqCaps, w3cCapabilities) {
121
+ async createSession(jsonwpCaps, reqCaps, w3cCapabilities, driverData) {
122
+ var _alwaysMatch, _w3cCapabilities;
123
+
123
124
  const defaultCapabilities = _lodash.default.cloneDeep(this.args.defaultCapabilities);
124
125
 
125
126
  const defaultSettings = (0, _utils.pullSettings)(defaultCapabilities);
126
127
  jsonwpCaps = _lodash.default.cloneDeep(jsonwpCaps);
127
- const jwpSettings = Object.assign({}, defaultSettings, (0, _utils.pullSettings)(jsonwpCaps));
128
+ const jwpSettings = { ...defaultSettings,
129
+ ...(0, _utils.pullSettings)(jsonwpCaps)
130
+ };
128
131
  w3cCapabilities = _lodash.default.cloneDeep(w3cCapabilities);
129
- const w3cSettings = Object.assign({}, jwpSettings);
130
- Object.assign(w3cSettings, (0, _utils.pullSettings)((w3cCapabilities || {}).alwaysMatch || {}));
132
+ const w3cSettings = { ...jwpSettings,
133
+ ...(0, _utils.pullSettings)((_alwaysMatch = ((_w3cCapabilities = w3cCapabilities) !== null && _w3cCapabilities !== void 0 ? _w3cCapabilities : {}).alwaysMatch) !== null && _alwaysMatch !== void 0 ? _alwaysMatch : {})
134
+ };
135
+
136
+ for (const firstMatchEntry of (_firstMatch = ((_w3cCapabilities2 = w3cCapabilities) !== null && _w3cCapabilities2 !== void 0 ? _w3cCapabilities2 : {}).firstMatch) !== null && _firstMatch !== void 0 ? _firstMatch : []) {
137
+ var _firstMatch, _w3cCapabilities2;
131
138
 
132
- for (const firstMatchEntry of (w3cCapabilities || {}).firstMatch || []) {
133
139
  Object.assign(w3cSettings, (0, _utils.pullSettings)(firstMatchEntry));
134
140
  }
135
141
 
@@ -141,10 +147,10 @@ class AppiumDriver extends _baseDriver.BaseDriver {
141
147
  const {
142
148
  desiredCaps,
143
149
  processedJsonwpCapabilities,
144
- processedW3CCapabilities,
145
- error
150
+ processedW3CCapabilities
146
151
  } = parsedCaps;
147
152
  protocol = parsedCaps.protocol;
153
+ const error = parsedCaps.error;
148
154
 
149
155
  if (error) {
150
156
  throw error;
@@ -154,52 +160,48 @@ class AppiumDriver extends _baseDriver.BaseDriver {
154
160
  driver: InnerDriver,
155
161
  version: driverVersion,
156
162
  driverName
157
- } = this._findMatchingDriver(this.driverConfig, desiredCaps);
158
-
163
+ } = this.driverConfig.findMatchingDriver(desiredCaps);
159
164
  this.printNewSessionAnnouncement(InnerDriver.name, driverVersion, InnerDriver.baseVersion);
160
165
 
161
166
  if (this.args.sessionOverride) {
162
167
  await this.deleteAllSessions();
163
168
  }
164
169
 
165
- let runningDriversData, otherPendingDriversData;
170
+ let runningDriversData = [];
171
+ let otherPendingDriversData = [];
166
172
  const driverInstance = new InnerDriver(this.args, true);
167
173
 
168
174
  if (this.args.relaxedSecurityEnabled) {
169
- _logger.default.info(`Applying relaxed security to '${InnerDriver.name}' as per ` + `server command line argument. All insecure features will be ` + `enabled unless explicitly disabled by --deny-insecure`);
170
-
175
+ this.log.info(`Applying relaxed security to '${InnerDriver.name}' as per ` + `server command line argument. All insecure features will be ` + `enabled unless explicitly disabled by --deny-insecure`);
171
176
  driverInstance.relaxedSecurityEnabled = true;
172
177
  }
173
178
 
174
179
  if (!_lodash.default.isEmpty(this.args.denyInsecure)) {
175
- _logger.default.info('Explicitly preventing use of insecure features:');
176
-
177
- this.args.denyInsecure.map(a => _logger.default.info(` ${a}`));
180
+ this.log.info('Explicitly preventing use of insecure features:');
181
+ this.args.denyInsecure.map(a => this.log.info(` ${a}`));
178
182
  driverInstance.denyInsecure = this.args.denyInsecure;
179
183
  }
180
184
 
181
185
  if (!_lodash.default.isEmpty(this.args.allowInsecure)) {
182
- _logger.default.info('Explicitly enabling use of insecure features:');
183
-
184
- this.args.allowInsecure.map(a => _logger.default.info(` ${a}`));
186
+ this.log.info('Explicitly enabling use of insecure features:');
187
+ this.args.allowInsecure.map(a => this.log.info(` ${a}`));
185
188
  driverInstance.allowInsecure = this.args.allowInsecure;
186
189
  }
187
190
 
188
191
  this.assignCliArgsToExtension('driver', driverName, driverInstance);
189
- driverInstance.server = this.server;
190
- driverInstance.serverHost = this.args.address;
191
- driverInstance.serverPort = this.args.port;
192
- driverInstance.serverPath = this.args.basePath;
192
+ driverInstance.assignServer(this.server, this.args.address, this.args.port, this.args.basePath);
193
193
 
194
194
  try {
195
- runningDriversData = await this.curSessionDataForDriver(InnerDriver);
195
+ var _await$this$curSessio;
196
+
197
+ runningDriversData = (_await$this$curSessio = await this.curSessionDataForDriver(InnerDriver)) !== null && _await$this$curSessio !== void 0 ? _await$this$curSessio : [];
196
198
  } catch (e) {
197
199
  throw new _baseDriver.errors.SessionNotCreatedError(e.message);
198
200
  }
199
201
 
200
202
  await pendingDriversGuard.acquire(AppiumDriver.name, () => {
201
203
  this.pendingDrivers[InnerDriver.name] = this.pendingDrivers[InnerDriver.name] || [];
202
- otherPendingDriversData = this.pendingDrivers[InnerDriver.name].map(drv => drv.driverData);
204
+ otherPendingDriversData = _lodash.default.compact(this.pendingDrivers[InnerDriver.name].map(drv => drv.driverData));
203
205
  this.pendingDrivers[InnerDriver.name].push(driverInstance);
204
206
  });
205
207
 
@@ -216,18 +218,14 @@ class AppiumDriver extends _baseDriver.BaseDriver {
216
218
  }
217
219
 
218
220
  this.attachUnexpectedShutdownHandler(driverInstance, innerSessionId);
219
-
220
- _logger.default.info(`New ${InnerDriver.name} session created successfully, session ` + `${innerSessionId} added to master session list`);
221
-
221
+ this.log.info(`New ${InnerDriver.name} session created successfully, session ` + `${innerSessionId} added to master session list`);
222
222
  driverInstance.startNewCommandTimeout();
223
223
 
224
224
  if (driverInstance.isW3CProtocol() && !_lodash.default.isEmpty(w3cSettings)) {
225
- _logger.default.info(`Applying the initial values to Appium settings parsed from W3C caps: ` + JSON.stringify(w3cSettings));
226
-
225
+ this.log.info(`Applying the initial values to Appium settings parsed from W3C caps: ` + JSON.stringify(w3cSettings));
227
226
  await driverInstance.updateSettings(w3cSettings);
228
227
  } else if (driverInstance.isMjsonwpProtocol() && !_lodash.default.isEmpty(jwpSettings)) {
229
- _logger.default.info(`Applying the initial values to Appium settings parsed from MJSONWP caps: ` + JSON.stringify(jwpSettings));
230
-
228
+ this.log.info(`Applying the initial values to Appium settings parsed from MJSONWP caps: ` + JSON.stringify(jwpSettings));
231
229
  await driverInstance.updateSettings(jwpSettings);
232
230
  }
233
231
  } catch (error) {
@@ -245,26 +243,25 @@ class AppiumDriver extends _baseDriver.BaseDriver {
245
243
 
246
244
  attachUnexpectedShutdownHandler(driver, innerSessionId) {
247
245
  const onShutdown = (cause = new Error('Unknown error')) => {
248
- _logger.default.warn(`Ending session, cause was '${cause.message}'`);
246
+ this.log.warn(`Ending session, cause was '${cause.message}'`);
249
247
 
250
248
  if (this.sessionPlugins[innerSessionId]) {
251
249
  for (const plugin of this.sessionPlugins[innerSessionId]) {
252
250
  if (_lodash.default.isFunction(plugin.onUnexpectedShutdown)) {
253
- _logger.default.debug(`Plugin ${plugin.name} defines an unexpected shutdown handler; calling it now`);
251
+ this.log.debug(`Plugin ${plugin.name} defines an unexpected shutdown handler; calling it now`);
254
252
 
255
253
  try {
256
254
  plugin.onUnexpectedShutdown(driver, cause);
257
255
  } catch (e) {
258
- _logger.default.warn(`Got an error when running plugin ${plugin.name} shutdown handler: ${e}`);
256
+ this.log.warn(`Got an error when running plugin ${plugin.name} shutdown handler: ${e}`);
259
257
  }
260
258
  } else {
261
- _logger.default.debug(`Plugin ${plugin.name} does not define an unexpected shutdown handler`);
259
+ this.log.debug(`Plugin ${plugin.name} does not define an unexpected shutdown handler`);
262
260
  }
263
261
  }
264
262
  }
265
263
 
266
- _logger.default.info(`Removing session '${innerSessionId}' from our master session list`);
267
-
264
+ this.log.info(`Removing session '${innerSessionId}' from our master session list`);
268
265
  delete this.sessions[innerSessionId];
269
266
  delete this.sessionPlugins[innerSessionId];
270
267
  };
@@ -272,14 +269,14 @@ class AppiumDriver extends _baseDriver.BaseDriver {
272
269
  if (_lodash.default.isFunction(driver.onUnexpectedShutdown)) {
273
270
  driver.onUnexpectedShutdown(onShutdown);
274
271
  } else {
275
- _logger.default.warn(`Failed to attach the unexpected shutdown listener. ` + `Is 'onUnexpectedShutdown' method available for '${driver.constructor.name}'?`);
272
+ this.log.warn(`Failed to attach the unexpected shutdown listener. ` + `Is 'onUnexpectedShutdown' method available for '${driver.constructor.name}'?`);
276
273
  }
277
274
  }
278
275
 
279
276
  async curSessionDataForDriver(InnerDriver) {
280
277
  const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);
281
278
 
282
- const data = _lodash.default.values(sessions).filter(s => s.constructor.name === InnerDriver.name).map(s => s.driverData);
279
+ const data = _lodash.default.compact(_lodash.default.values(sessions).filter(s => s.constructor.name === InnerDriver.name).map(s => s.driverData));
283
280
 
284
281
  for (let datum of data) {
285
282
  if (!datum) {
@@ -294,30 +291,32 @@ class AppiumDriver extends _baseDriver.BaseDriver {
294
291
  let protocol;
295
292
 
296
293
  try {
297
- let otherSessionsData = null;
298
- let dstSession = null;
299
- await sessionsListGuard.acquire(AppiumDriver.name, () => {
294
+ let otherSessionsData;
295
+ const dstSession = await sessionsListGuard.acquire(AppiumDriver.name, () => {
300
296
  if (!this.sessions[sessionId]) {
301
297
  return;
302
298
  }
303
299
 
304
300
  const curConstructorName = this.sessions[sessionId].constructor.name;
305
301
  otherSessionsData = _lodash.default.toPairs(this.sessions).filter(([key, value]) => value.constructor.name === curConstructorName && key !== sessionId).map(([, value]) => value.driverData);
306
- dstSession = this.sessions[sessionId];
302
+ const dstSession = this.sessions[sessionId];
307
303
  protocol = dstSession.protocol;
308
-
309
- _logger.default.info(`Removing session ${sessionId} from our master session list`);
310
-
304
+ this.log.info(`Removing session ${sessionId} from our master session list`);
311
305
  delete this.sessions[sessionId];
312
306
  delete this.sessionPlugins[sessionId];
307
+ return dstSession;
313
308
  });
309
+
310
+ if (!dstSession) {
311
+ throw new Error('Session not found');
312
+ }
313
+
314
314
  return {
315
315
  protocol,
316
316
  value: await dstSession.deleteSession(sessionId, otherSessionsData)
317
317
  };
318
318
  } catch (e) {
319
- _logger.default.error(`Had trouble ending session ${sessionId}: ${e.message}`);
320
-
319
+ this.log.error(`Had trouble ending session ${sessionId}: ${e.message}`);
321
320
  return {
322
321
  protocol,
323
322
  error: e
@@ -329,8 +328,7 @@ class AppiumDriver extends _baseDriver.BaseDriver {
329
328
  const sessionsCount = _lodash.default.size(this.sessions);
330
329
 
331
330
  if (0 === sessionsCount) {
332
- _logger.default.debug('There are no active sessions for cleanup');
333
-
331
+ this.log.debug('There are no active sessions for cleanup');
334
332
  return;
335
333
  }
336
334
 
@@ -338,16 +336,14 @@ class AppiumDriver extends _baseDriver.BaseDriver {
338
336
  force = false,
339
337
  reason
340
338
  } = opts;
341
-
342
- _logger.default.debug(`Cleaning up ${_support.util.pluralize('active session', sessionsCount, true)}`);
343
-
339
+ this.log.debug(`Cleaning up ${_support.util.pluralize('active session', sessionsCount, true)}`);
344
340
  const cleanupPromises = force ? _lodash.default.values(this.sessions).map(drv => drv.startUnexpectedShutdown(reason && new Error(reason))) : _lodash.default.keys(this.sessions).map(id => this.deleteSession(id));
345
341
 
346
342
  for (const cleanupPromise of cleanupPromises) {
347
343
  try {
348
344
  await cleanupPromise;
349
345
  } catch (e) {
350
- _logger.default.debug(e);
346
+ this.log.debug(e);
351
347
  }
352
348
  }
353
349
  }
@@ -420,7 +416,7 @@ class AppiumDriver extends _baseDriver.BaseDriver {
420
416
  };
421
417
 
422
418
  const defaultBehavior = async () => {
423
- plugins.length && _logger.default.info(`Executing default handling behavior for command '${cmd}'`);
419
+ plugins.length && this.log.info(`Executing default handling behavior for command '${cmd}'`);
424
420
  cmdHandledBy.default = true;
425
421
 
426
422
  if (reqForProxy) {
@@ -436,7 +432,7 @@ class AppiumDriver extends _baseDriver.BaseDriver {
436
432
  }
437
433
 
438
434
  if (isUmbrellaCmd) {
439
- return await super.executeCommand(cmd, ...args);
435
+ return await _baseDriver.BaseDriver.prototype.executeCommand.call(this, cmd, ...args);
440
436
  }
441
437
 
442
438
  return await dstSession.executeCommand(cmd, ...args);
@@ -462,8 +458,7 @@ class AppiumDriver extends _baseDriver.BaseDriver {
462
458
  if (cmd === _baseDriver.CREATE_SESSION_COMMAND && this.sessionlessPlugins.length && !res.error) {
463
459
  const sessionId = _lodash.default.first(res.value);
464
460
 
465
- _logger.default.info(`Promoting ${this.sessionlessPlugins.length} sessionless plugins to be attached ` + `to session ID ${sessionId}`);
466
-
461
+ this.log.info(`Promoting ${this.sessionlessPlugins.length} sessionless plugins to be attached ` + `to session ID ${sessionId}`);
467
462
  this.sessionPlugins[sessionId] = this.sessionlessPlugins;
468
463
  this.sessionlessPlugins = [];
469
464
  }
@@ -479,14 +474,13 @@ class AppiumDriver extends _baseDriver.BaseDriver {
479
474
  cmdHandledBy,
480
475
  plugins
481
476
  }) {
482
- plugins.length && _logger.default.info(`Plugins which can handle cmd '${cmd}': ${plugins.map(p => p.name)}`);
477
+ plugins.length && this.log.info(`Plugins which can handle cmd '${cmd}': ${plugins.map(p => p.name)}`);
483
478
 
484
479
  for (const plugin of plugins) {
485
480
  cmdHandledBy[plugin.name] = false;
486
481
 
487
482
  next = (_next => async () => {
488
- _logger.default.info(`Plugin ${plugin.name} is now handling cmd '${cmd}'`);
489
-
483
+ this.log.info(`Plugin ${plugin.name} is now handling cmd '${cmd}'`);
490
484
  cmdHandledBy[plugin.name] = true;
491
485
 
492
486
  if (plugin[cmd]) {
@@ -512,7 +506,7 @@ class AppiumDriver extends _baseDriver.BaseDriver {
512
506
  const didntHandle = Object.keys(cmdHandledBy).filter(k => !cmdHandledBy[k]);
513
507
 
514
508
  if (didntHandle.length > 0) {
515
- _logger.default.info(`Command '${cmd}' was *not* handled by the following behaviours or plugins, even ` + `though they were registered to handle it: ${JSON.stringify(didntHandle)}. The ` + `command *was* handled by these: ${JSON.stringify(didHandle)}.`);
509
+ this.log.info(`Command '${cmd}' was *not* handled by the following behaviours or plugins, even ` + `though they were registered to handle it: ${JSON.stringify(didntHandle)}. The ` + `command *was* handled by these: ${JSON.stringify(didHandle)}.`);
516
510
  }
517
511
  }
518
512
 
@@ -565,14 +559,13 @@ function isAppiumDriverCommand(cmd) {
565
559
  }
566
560
 
567
561
  class NoDriverProxyCommandError extends Error {
562
+ code = 'APPIUMERR_NO_DRIVER_PROXYCOMMAND';
563
+
568
564
  constructor() {
569
565
  super(`The default behavior for this command was to proxy, but the driver ` + `did not have the 'proxyCommand' method defined. To fully support ` + `plugins, drivers should have 'proxyCommand' set to a jwpProxy object's ` + `'command()' method, in addition to the normal 'proxyReqRes'`);
570
- (0, _defineProperty2.default)(this, "code", 'APPIUMERR_NO_DRIVER_PROXYCOMMAND');
571
566
  }
572
567
 
573
568
  }
574
569
 
575
- exports.NoDriverProxyCommandError = NoDriverProxyCommandError;require('source-map-support').install();
576
-
577
-
578
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/appium.js"],"names":["desiredCapabilityConstraints","automationName","presence","isString","platformName","sessionsListGuard","AsyncLock","pendingDriversGuard","AppiumDriver","BaseDriver","constructor","args","tmpDir","process","env","APPIUM_TMP_DIR","desiredCapConstraints","newCommandTimeoutMs","sessions","pendingDrivers","pluginClasses","sessionPlugins","sessionlessPlugins","isCommandsQueueEnabled","sessionExists","sessionId","dstSession","driverForSession","getStatus","build","_","clone","getSessions","acquire","name","toPairs","map","id","driver","capabilities","caps","printNewSessionAnnouncement","driverName","driverVersion","driverBaseVersion","log","info","APPIUM_VER","baseVersion","_findMatchingDriver","assignCliArgsToExtension","extType","extName","extInstance","allCliArgsForExt","isEmpty","defaults","cliArgs","omitBy","value","key","isEqual","createSession","jsonwpCaps","reqCaps","w3cCapabilities","defaultCapabilities","cloneDeep","defaultSettings","jwpSettings","Object","assign","w3cSettings","alwaysMatch","firstMatchEntry","firstMatch","protocol","innerSessionId","dCaps","parsedCaps","desiredCaps","processedJsonwpCapabilities","processedW3CCapabilities","error","InnerDriver","version","driverConfig","sessionOverride","deleteAllSessions","runningDriversData","otherPendingDriversData","driverInstance","relaxedSecurityEnabled","denyInsecure","a","allowInsecure","server","serverHost","address","serverPort","port","serverPath","basePath","curSessionDataForDriver","e","errors","SessionNotCreatedError","message","drv","driverData","push","pull","attachUnexpectedShutdownHandler","startNewCommandTimeout","isW3CProtocol","JSON","stringify","updateSettings","isMjsonwpProtocol","onShutdown","cause","Error","warn","plugin","isFunction","onUnexpectedShutdown","debug","data","values","filter","s","datum","deleteSession","otherSessionsData","curConstructorName","opts","sessionsCount","size","force","reason","util","pluralize","cleanupPromises","startUnexpectedShutdown","keys","cleanupPromise","pluginsForSession","createPluginInstances","pluginsToHandleCmd","cmd","p","handle","PluginClass","pluginName","executeCommand","isGetStatus","GET_STATUS_COMMAND","isDeleteSession","DELETE_SESSION_COMMAND","isUmbrellaCmd","isAppiumDriverCommand","isSessionCmd","reqForProxy","last","pop","plugins","cmdHandledBy","default","defaultBehavior","length","proxyCommand","NoDriverProxyCommandError","originalUrl","method","body","wrappedCmd","wrapCommandWithPlugins","next","res","executeWrappedCommand","logPluginHandlerReport","CREATE_SESSION_COMMAND","first","_next","didHandle","k","didntHandle","cmdRes","cmdErr","isPlainObject","has","proxyActive","getProxyAvoidList","canProxy"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport log from './logger';\nimport { getBuildInfo, updateBuildInfo, APPIUM_VER } from './config';\nimport { findMatchingDriver } from './drivers';\nimport { BaseDriver, errors, isSessionCommand,\n         CREATE_SESSION_COMMAND, DELETE_SESSION_COMMAND, GET_STATUS_COMMAND\n} from '@appium/base-driver';\nimport AsyncLock from 'async-lock';\nimport { parseCapsForInnerDriver, pullSettings } from './utils';\nimport { util } from '@appium/support';\nimport { getDefaultsForExtension } from './schema';\n\nconst desiredCapabilityConstraints = {\n  automationName: {\n    presence: true,\n    isString: true,\n  },\n  platformName: {\n    presence: true,\n    isString: true,\n  },\n};\n\nconst sessionsListGuard = new AsyncLock();\nconst pendingDriversGuard = new AsyncLock();\n\nclass AppiumDriver extends BaseDriver {\n  constructor (args) {\n    // It is necessary to set `--tmp` here since it should be set to\n    // process.env.APPIUM_TMP_DIR once at an initial point in the Appium lifecycle.\n    // The process argument will be referenced by BaseDriver.\n    // Please call @appium/support.tempDir module to apply this benefit.\n    if (args.tmpDir) {\n      process.env.APPIUM_TMP_DIR = args.tmpDir;\n    }\n\n    super(args);\n\n    this.desiredCapConstraints = desiredCapabilityConstraints;\n\n    // the main Appium Driver has no new command timeout\n    this.newCommandTimeoutMs = 0;\n\n    this.args = {...args};\n\n    // Access to sessions list must be guarded with a Semaphore, because\n    // it might be changed by other async calls at any time\n    // It is not recommended to access this property directly from the outside\n    this.sessions = {};\n\n    // Access to pending drivers list must be guarded with a Semaphore, because\n    // it might be changed by other async calls at any time\n    // It is not recommended to access this property directly from the outside\n    this.pendingDrivers = {};\n\n    /** @type {PluginExtensionClass[]} */\n    this.pluginClasses = []; // list of which plugins are active\n    this.sessionPlugins = {}; // map of sessions to actual plugin instances per session\n    this.sessionlessPlugins = []; // some commands are sessionless, so we need a set of plugins for them\n\n    // allow this to happen in the background, so no `await`\n    updateBuildInfo();\n  }\n\n  /** @type {import('./driver-config').default|undefined} */\n  driverConfig;\n\n  /** @type {import('express').Express|undefined} */\n  server;\n\n  /**\n   * Cancel commands queueing for the umbrella Appium driver\n   */\n  get isCommandsQueueEnabled () {\n    return false;\n  }\n\n  sessionExists (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && dstSession.sessionId !== null;\n  }\n\n  driverForSession (sessionId) {\n    return this.sessions[sessionId];\n  }\n\n  async getStatus () { // eslint-disable-line require-await\n    return {\n      build: _.clone(getBuildInfo()),\n    };\n  }\n\n  async getSessions () {\n    const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);\n    return _.toPairs(sessions)\n      .map(([id, driver]) => ({id, capabilities: driver.caps}));\n  }\n\n  printNewSessionAnnouncement (driverName, driverVersion, driverBaseVersion) {\n    log.info(driverVersion\n      ? `Appium v${APPIUM_VER} creating new ${driverName} (v${driverVersion}) session`\n      : `Appium v${APPIUM_VER} creating new ${driverName} session`\n    );\n    log.info(`Checking BaseDriver versions for Appium and ${driverName}`);\n    log.info(AppiumDriver.baseVersion\n      ? `Appium's BaseDriver version is ${AppiumDriver.baseVersion}`\n      : `Could not determine Appium's BaseDriver version`\n    );\n    log.info(driverBaseVersion\n      ? `${driverName}'s BaseDriver version is ${driverBaseVersion}`\n      : `Could not determine ${driverName}'s BaseDriver version`\n    );\n  }\n\n  /**\n   * This is just an alias for driver.js's method, which is necessary for\n   * mocking in the test suite\n   */\n  _findMatchingDriver (...args) {\n    return findMatchingDriver(...args);\n  }\n\n  /**\n   * Validate and assign CLI args for a driver or plugin\n   *\n   * If the extension has provided a schema, validation has already happened.\n   *\n   * Any arg which is equal to its default value will not be assigned to the extension.\n   * @param {import('./ext-config-io').ExtensionType} extType 'driver' or 'plugin'\n   * @param {string} extName the name of the extension\n   * @param {Object} extInstance the driver or plugin instance\n   */\n  assignCliArgsToExtension (extType, extName, extInstance) {\n    const allCliArgsForExt = this.args[extType]?.[extName];\n    if (!_.isEmpty(allCliArgsForExt)) {\n      const defaults = getDefaultsForExtension(extType, extName);\n      const cliArgs = _.isEmpty(defaults)\n        ? allCliArgsForExt\n        : _.omitBy(allCliArgsForExt, (value, key) => _.isEqual(defaults[key], value));\n      if (!_.isEmpty(cliArgs)) {\n        extInstance.cliArgs = cliArgs;\n      }\n    }\n  }\n\n  /**\n   * Create a new session\n   * @param {Object} jsonwpCaps JSONWP formatted desired capabilities\n   * @param {Object} reqCaps Required capabilities (JSONWP standard)\n   * @param {Object} w3cCapabilities W3C capabilities\n   * @return {Array} Unique session ID and capabilities\n   */\n  async createSession (jsonwpCaps, reqCaps, w3cCapabilities) {\n    const defaultCapabilities = _.cloneDeep(this.args.defaultCapabilities);\n    const defaultSettings = pullSettings(defaultCapabilities);\n    jsonwpCaps = _.cloneDeep(jsonwpCaps);\n    const jwpSettings = Object.assign({}, defaultSettings, pullSettings(jsonwpCaps));\n    w3cCapabilities = _.cloneDeep(w3cCapabilities);\n    // It is possible that the client only provides caps using JSONWP standard,\n    // although firstMatch/alwaysMatch properties are still present.\n    // In such case we assume the client understands W3C protocol and merge the given\n    // JSONWP caps to W3C caps\n    const w3cSettings = Object.assign({}, jwpSettings);\n    Object.assign(w3cSettings, pullSettings((w3cCapabilities || {}).alwaysMatch || {}));\n    for (const firstMatchEntry of ((w3cCapabilities || {}).firstMatch || [])) {\n      Object.assign(w3cSettings, pullSettings(firstMatchEntry));\n    }\n\n    let protocol;\n    let innerSessionId, dCaps;\n    try {\n      // Parse the caps into a format that the InnerDriver will accept\n      const parsedCaps = parseCapsForInnerDriver(\n        jsonwpCaps,\n        w3cCapabilities,\n        this.desiredCapConstraints,\n        defaultCapabilities\n      );\n\n      const {desiredCaps, processedJsonwpCapabilities, processedW3CCapabilities, error} = parsedCaps;\n      protocol = parsedCaps.protocol;\n\n      // If the parsing of the caps produced an error, throw it in here\n      if (error) {\n        throw error;\n      }\n\n      const {\n        driver: InnerDriver,\n        version: driverVersion,\n        driverName\n      } = this._findMatchingDriver(this.driverConfig, desiredCaps);\n      this.printNewSessionAnnouncement(InnerDriver.name, driverVersion, InnerDriver.baseVersion);\n\n      if (this.args.sessionOverride) {\n        await this.deleteAllSessions();\n      }\n\n      let runningDriversData, otherPendingDriversData;\n\n      const driverInstance = new InnerDriver(this.args, true);\n\n      // We want to assign security values directly on the driver. The driver\n      // should not read security values from `this.opts` because those values\n      // could have been set by a malicious user via capabilities, whereas we\n      // want a guarantee the values were set by the appium server admin\n      if (this.args.relaxedSecurityEnabled) {\n        log.info(`Applying relaxed security to '${InnerDriver.name}' as per ` +\n                 `server command line argument. All insecure features will be ` +\n                 `enabled unless explicitly disabled by --deny-insecure`);\n        driverInstance.relaxedSecurityEnabled = true;\n      }\n\n      if (!_.isEmpty(this.args.denyInsecure)) {\n        log.info('Explicitly preventing use of insecure features:');\n        this.args.denyInsecure.map((a) => log.info(`    ${a}`));\n        driverInstance.denyInsecure = this.args.denyInsecure;\n      }\n\n      if (!_.isEmpty(this.args.allowInsecure)) {\n        log.info('Explicitly enabling use of insecure features:');\n        this.args.allowInsecure.map((a) => log.info(`    ${a}`));\n        driverInstance.allowInsecure = this.args.allowInsecure;\n      }\n\n      // Likewise, any driver-specific CLI args that were passed in should be assigned directly to\n      // the driver so that they cannot be mimicked by a malicious user sending in capabilities\n      this.assignCliArgsToExtension('driver', driverName, driverInstance);\n\n\n      // This assignment is required for correct web sockets functionality inside the driver\n      driverInstance.server = this.server;\n\n      // Drivers/plugins might also want to know where they are hosted\n      driverInstance.serverHost = this.args.address;\n      driverInstance.serverPort = this.args.port;\n      driverInstance.serverPath = this.args.basePath;\n\n      try {\n        runningDriversData = await this.curSessionDataForDriver(InnerDriver);\n      } catch (e) {\n        throw new errors.SessionNotCreatedError(e.message);\n      }\n      await pendingDriversGuard.acquire(AppiumDriver.name, () => {\n        this.pendingDrivers[InnerDriver.name] = this.pendingDrivers[InnerDriver.name] || [];\n        otherPendingDriversData = this.pendingDrivers[InnerDriver.name].map((drv) => drv.driverData);\n        this.pendingDrivers[InnerDriver.name].push(driverInstance);\n      });\n\n      try {\n        [innerSessionId, dCaps] = await driverInstance.createSession(\n          processedJsonwpCapabilities,\n          reqCaps,\n          processedW3CCapabilities,\n          [...runningDriversData, ...otherPendingDriversData]\n        );\n        protocol = driverInstance.protocol;\n        await sessionsListGuard.acquire(AppiumDriver.name, () => {\n          this.sessions[innerSessionId] = driverInstance;\n        });\n      } finally {\n        await pendingDriversGuard.acquire(AppiumDriver.name, () => {\n          _.pull(this.pendingDrivers[InnerDriver.name], driverInstance);\n        });\n      }\n\n      this.attachUnexpectedShutdownHandler(driverInstance, innerSessionId);\n\n      log.info(`New ${InnerDriver.name} session created successfully, session ` +\n              `${innerSessionId} added to master session list`);\n\n      // set the New Command Timeout for the inner driver\n      driverInstance.startNewCommandTimeout();\n\n      // apply initial values to Appium settings (if provided)\n      if (driverInstance.isW3CProtocol() && !_.isEmpty(w3cSettings)) {\n        log.info(`Applying the initial values to Appium settings parsed from W3C caps: ` +\n          JSON.stringify(w3cSettings));\n        await driverInstance.updateSettings(w3cSettings);\n      } else if (driverInstance.isMjsonwpProtocol() && !_.isEmpty(jwpSettings)) {\n        log.info(`Applying the initial values to Appium settings parsed from MJSONWP caps: ` +\n          JSON.stringify(jwpSettings));\n        await driverInstance.updateSettings(jwpSettings);\n      }\n    } catch (error) {\n      return {\n        protocol,\n        error,\n      };\n    }\n\n    return {\n      protocol,\n      value: [innerSessionId, dCaps, protocol]\n    };\n  }\n\n  attachUnexpectedShutdownHandler (driver, innerSessionId) {\n    const onShutdown = (cause = new Error('Unknown error')) => {\n      log.warn(`Ending session, cause was '${cause.message}'`);\n\n      if (this.sessionPlugins[innerSessionId]) {\n        for (const plugin of this.sessionPlugins[innerSessionId]) {\n          if (_.isFunction(plugin.onUnexpectedShutdown)) {\n            log.debug(`Plugin ${plugin.name} defines an unexpected shutdown handler; calling it now`);\n            try {\n              plugin.onUnexpectedShutdown(driver, cause);\n            } catch (e) {\n              log.warn(`Got an error when running plugin ${plugin.name} shutdown handler: ${e}`);\n            }\n          } else {\n            log.debug(`Plugin ${plugin.name} does not define an unexpected shutdown handler`);\n          }\n        }\n      }\n\n      log.info(`Removing session '${innerSessionId}' from our master session list`);\n      delete this.sessions[innerSessionId];\n      delete this.sessionPlugins[innerSessionId];\n    };\n\n    if (_.isFunction(driver.onUnexpectedShutdown)) {\n      driver.onUnexpectedShutdown(onShutdown);\n    } else {\n      log.warn(`Failed to attach the unexpected shutdown listener. ` +\n        `Is 'onUnexpectedShutdown' method available for '${driver.constructor.name}'?`);\n    }\n  }\n\n  async curSessionDataForDriver (InnerDriver) {\n    const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);\n    const data = _.values(sessions)\n                   .filter((s) => s.constructor.name === InnerDriver.name)\n                   .map((s) => s.driverData);\n    for (let datum of data) {\n      if (!datum) {\n        throw new Error(`Problem getting session data for driver type ` +\n                        `${InnerDriver.name}; does it implement 'get ` +\n                        `driverData'?`);\n      }\n    }\n    return data;\n  }\n\n  async deleteSession (sessionId) {\n    let protocol;\n    try {\n      let otherSessionsData = null;\n      let dstSession = null;\n      await sessionsListGuard.acquire(AppiumDriver.name, () => {\n        if (!this.sessions[sessionId]) {\n          return;\n        }\n        const curConstructorName = this.sessions[sessionId].constructor.name;\n        otherSessionsData = _.toPairs(this.sessions)\n              .filter(([key, value]) => value.constructor.name === curConstructorName && key !== sessionId)\n              .map(([, value]) => value.driverData);\n        dstSession = this.sessions[sessionId];\n        protocol = dstSession.protocol;\n        log.info(`Removing session ${sessionId} from our master session list`);\n        // regardless of whether the deleteSession completes successfully or not\n        // make the session unavailable, because who knows what state it might\n        // be in otherwise\n        delete this.sessions[sessionId];\n        delete this.sessionPlugins[sessionId];\n      });\n      return {\n        protocol,\n        value: await dstSession.deleteSession(sessionId, otherSessionsData),\n      };\n    } catch (e) {\n      log.error(`Had trouble ending session ${sessionId}: ${e.message}`);\n      return {\n        protocol,\n        error: e,\n      };\n    }\n  }\n\n  async deleteAllSessions (opts = {}) {\n    const sessionsCount = _.size(this.sessions);\n    if (0 === sessionsCount) {\n      log.debug('There are no active sessions for cleanup');\n      return;\n    }\n\n    const {\n      force = false,\n      reason,\n    } = opts;\n    log.debug(`Cleaning up ${util.pluralize('active session', sessionsCount, true)}`);\n    const cleanupPromises = force\n      ? _.values(this.sessions).map((drv) => drv.startUnexpectedShutdown(reason && new Error(reason)))\n      : _.keys(this.sessions).map((id) => this.deleteSession(id));\n    for (const cleanupPromise of cleanupPromises) {\n      try {\n        await cleanupPromise;\n      } catch (e) {\n        log.debug(e);\n      }\n    }\n  }\n\n  /**\n   * Get the appropriate plugins for a session (or sessionless plugins)\n   *\n   * @param {?string} sessionId - the sessionId (or null) to use to find plugins\n   * @returns {Array} - array of plugin instances\n   */\n  pluginsForSession (sessionId = null) {\n    if (sessionId) {\n      if (!this.sessionPlugins[sessionId]) {\n        this.sessionPlugins[sessionId] = this.createPluginInstances();\n      }\n      return this.sessionPlugins[sessionId];\n    }\n\n    if (_.isEmpty(this.sessionlessPlugins)) {\n      this.sessionlessPlugins = this.createPluginInstances();\n    }\n    return this.sessionlessPlugins;\n  }\n\n  /**\n   * To get plugins for a command, we either get the plugin instances associated with the\n   * particular command's session, or in the case of sessionless plugins, pull from the set of\n   * plugin instances reserved for sessionless commands (and we lazily create plugin instances on\n   * first use)\n   *\n   * @param {string} cmd - the name of the command to find a plugin to handle\n   * @param {?string} sessionId - the particular session for which to find a plugin, or null if\n   * sessionless\n   */\n  pluginsToHandleCmd (cmd, sessionId = null) {\n    // to handle a given command, a plugin should either implement that command as a plugin\n    // instance method or it should implement a generic 'handle' method\n    return this.pluginsForSession(sessionId)\n      .filter((p) => _.isFunction(p[cmd]) || _.isFunction(p.handle));\n  }\n\n  createPluginInstances () {\n    return this.pluginClasses.map((PluginClass) => {\n      const name = PluginClass.pluginName;\n      const plugin = new PluginClass(name);\n      this.assignCliArgsToExtension('plugin', name, plugin);\n      return plugin;\n    });\n  }\n\n  async executeCommand (cmd, ...args) {\n    // We have basically three cases for how to handle commands:\n    // 1. handle getStatus (we do this as a special out of band case so it doesn't get added to an\n    //    execution queue, and can be called while e.g. createSession is in progress)\n    // 2. handle commands that this umbrella driver should handle, rather than the actual session\n    //    driver (for example, deleteSession, or other non-session commands)\n    // 3. handle session driver commands.\n    // The tricky part is that because we support command plugins, we need to wrap any of these\n    // cases with plugin handling.\n\n    const isGetStatus = cmd === GET_STATUS_COMMAND;\n    const isDeleteSession = cmd === DELETE_SESSION_COMMAND;\n    const isUmbrellaCmd = !isGetStatus && isAppiumDriverCommand(cmd);\n    const isSessionCmd = !isUmbrellaCmd || isDeleteSession;\n\n    // if a plugin override proxying for this command and that is why we are here instead of just\n    // letting the protocol proxy the command entirely, determine that, get the request object for\n    // use later on, then clean up the args\n    const reqForProxy = _.last(args)?.reqForProxy;\n    if (reqForProxy) {\n      args.pop();\n    }\n\n\n    // first do some error checking. If we're requesting a session command execution, then make\n    // sure that session actually exists on the session driver, and set the session driver itself\n    let sessionId = null;\n    let dstSession = null;\n    let protocol = null;\n    let driver = this;\n    if (isSessionCmd) {\n      sessionId = _.last(args);\n      dstSession = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions[sessionId]);\n      if (!dstSession) {\n        throw new Error(`The session with id '${sessionId}' does not exist`);\n      }\n      // now save the response protocol given that the session driver's protocol might differ\n      protocol = dstSession.protocol;\n      if (!isUmbrellaCmd) {\n        driver = dstSession;\n      }\n    }\n\n    // get any plugins which are registered as handling this command\n    const plugins = this.pluginsToHandleCmd(cmd, sessionId);\n\n    // now we define a 'cmdHandledBy' object which will keep track of which plugins have handled this\n    // command. we care about this because (a) multiple plugins can handle the same command, and\n    // (b) there's no guarantee that a plugin will actually call the next() method which runs the\n    // original command execution. This results in a situation where the command might be handled\n    // by some but not all plugins, or by plugin(s) but not by the default behavior. So start out\n    // this object declaring that the default handler has not been executed.\n    const cmdHandledBy = {default: false};\n\n    // now we define an async function which will be passed to plugins, and successively wrapped\n    // if there is more than one plugin that can handle the command. To start off with, the async\n    // function is defined as calling the default behavior, i.e., whichever of the 3 cases above is\n    // the appropriate one\n    const defaultBehavior = async () => {\n      // if we're running with plugins, make sure we log that the default behavior is actually\n      // happening so we can tell when the plugin call chain is unwrapping to the default behavior\n      // if that's what happens\n      plugins.length && log.info(`Executing default handling behavior for command '${cmd}'`);\n\n      // if we make it here, we know that the default behavior is handled\n      cmdHandledBy.default = true;\n\n      if (reqForProxy) {\n        // we would have proxied this command had a plugin not handled it, so the default behavior\n        // is to do the proxy and retrieve the result internally so it can be passed to the plugin\n        // in case it calls 'await next()'. This requires that the driver have defined\n        // 'proxyCommand' and not just 'proxyReqRes'.\n        if (!dstSession.proxyCommand) {\n          throw new NoDriverProxyCommandError();\n        }\n        return await dstSession.proxyCommand(reqForProxy.originalUrl, reqForProxy.method,\n          reqForProxy.body);\n      }\n\n      if (isGetStatus) {\n        return await this.getStatus();\n      }\n\n      if (isUmbrellaCmd) {\n        // some commands, like deleteSession, we want to make sure to handle on *this* driver,\n        // not the platform driver\n        return await super.executeCommand(cmd, ...args);\n      }\n\n      // here we know that we are executing a session command, and have a valid session driver\n      return await dstSession.executeCommand(cmd, ...args);\n    };\n\n    // now take our default behavior, wrap it with any number of plugin behaviors, and run it\n    const wrappedCmd = this.wrapCommandWithPlugins({\n      driver, cmd, args, plugins, cmdHandledBy, next: defaultBehavior\n    });\n    const res = await this.executeWrappedCommand({wrappedCmd, protocol});\n\n    // if we had plugins, make sure to log out the helpful report about which plugins ended up\n    // handling the command and which didn't\n    this.logPluginHandlerReport(plugins, {cmd, cmdHandledBy});\n\n    // And finally, if the command was createSession, we want to migrate any plugins which were\n    // previously sessionless to use the new sessionId, so that plugins can share state between\n    // their createSession method and other instance methods\n    if (cmd === CREATE_SESSION_COMMAND && this.sessionlessPlugins.length && !res.error) {\n      const sessionId = _.first(res.value);\n      log.info(`Promoting ${this.sessionlessPlugins.length} sessionless plugins to be attached ` +\n               `to session ID ${sessionId}`);\n      this.sessionPlugins[sessionId] = this.sessionlessPlugins;\n      this.sessionlessPlugins = [];\n    }\n\n    return res;\n  }\n\n  wrapCommandWithPlugins ({driver, cmd, args, next, cmdHandledBy, plugins}) {\n    plugins.length && log.info(`Plugins which can handle cmd '${cmd}': ${plugins.map((p) => p.name)}`);\n\n    // now we can go through each plugin and wrap `next` around its own handler, passing the *old*\n    // next in so that it can call it if it wants to\n    for (const plugin of plugins) {\n      // need an IIFE here because we want the value of next that's passed to plugin.handle to be\n      // exactly the value of next here before reassignment; we don't want it to be lazily\n      // evaluated, otherwise we end up with infinite recursion of the last `next` to be defined.\n      cmdHandledBy[plugin.name] = false; // we see a new plugin, so add it to the 'cmdHandledBy' object\n      next = ((_next) => async () => {\n        log.info(`Plugin ${plugin.name} is now handling cmd '${cmd}'`);\n        cmdHandledBy[plugin.name] = true; // if we make it here, this plugin has attempted to handle cmd\n        // first attempt to handle the command via a command-specific handler on the plugin\n        if (plugin[cmd]) {\n          return await plugin[cmd](_next, driver, ...args);\n        }\n        // otherwise, call the generic 'handle' method\n        return await plugin.handle(_next, driver, cmd, ...args);\n      })(next);\n    }\n\n    return next;\n  }\n\n  logPluginHandlerReport (plugins, {cmd, cmdHandledBy}) {\n    if (!plugins.length) {\n      return;\n    }\n\n    // at the end of the day, we have an object representing which plugins ended up getting\n    // their code run as part of handling this command. Because plugins can choose *not* to\n    // pass control to other plugins or to the default driver behavior, this is information\n    // which is probably useful to the user (especially in situations where plugins might not\n    // interact well together, and it would be hard to debug otherwise without this kind of\n    // message).\n    const didHandle = Object.keys(cmdHandledBy).filter((k) => cmdHandledBy[k]);\n    const didntHandle = Object.keys(cmdHandledBy).filter((k) => !cmdHandledBy[k]);\n    if (didntHandle.length > 0) {\n      log.info(`Command '${cmd}' was *not* handled by the following behaviours or plugins, even ` +\n               `though they were registered to handle it: ${JSON.stringify(didntHandle)}. The ` +\n               `command *was* handled by these: ${JSON.stringify(didHandle)}.`);\n    }\n  }\n\n  async executeWrappedCommand ({wrappedCmd, protocol}) {\n    let cmdRes, cmdErr, res = {};\n    try {\n      // At this point, `wrappedCmd` defines a whole sequence of plugin handlers, culminating in\n      // our default handler. Whatever it returns is what we're going to want to send back to the\n      // user.\n      cmdRes = await wrappedCmd();\n    } catch (e) {\n      cmdErr = e;\n    }\n\n    // Sadly, we don't know exactly what kind of object will be returned. It will either be a bare\n    // object, or a protocol-aware object with protocol and error/value keys. So we need to sniff\n    // it and make sure we don't double-wrap it if it's the latter kind.\n    if (_.isPlainObject(cmdRes) && _.has(cmdRes, 'protocol')) {\n      res = cmdRes;\n    } else {\n      res.value = cmdRes;\n      res.error = cmdErr;\n      res.protocol = protocol;\n    }\n    return res;\n  }\n\n  proxyActive (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && _.isFunction(dstSession.proxyActive) && dstSession.proxyActive(sessionId);\n  }\n\n  getProxyAvoidList (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession ? dstSession.getProxyAvoidList() : [];\n  }\n\n  canProxy (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && dstSession.canProxy(sessionId);\n  }\n}\n\n// help decide which commands should be proxied to sub-drivers and which\n// should be handled by this, our umbrella driver\nfunction isAppiumDriverCommand (cmd) {\n  return !isSessionCommand(cmd) || cmd === 'deleteSession';\n}\n\n/**\n * Thrown when Appium tried to proxy a command using a driver's `proxyCommand` method but the\n * method did not exist\n */\nexport class NoDriverProxyCommandError extends Error {\n  /**\n   * @type {Readonly<string>}\n   */\n  code = 'APPIUMERR_NO_DRIVER_PROXYCOMMAND';\n\n  constructor () {\n    super(`The default behavior for this command was to proxy, but the driver ` +\n          `did not have the 'proxyCommand' method defined. To fully support ` +\n          `plugins, drivers should have 'proxyCommand' set to a jwpProxy object's ` +\n          `'command()' method, in addition to the normal 'proxyReqRes'`);\n  }\n}\n\nexport { AppiumDriver };\n\n\n/**\n * @typedef {Object} StaticExtMembers\n * @property {(app: import('express').Express, httpServer: import('http').Server) => import('type-fest').Promisable<void>} [updateServer]\n * @property {import('@appium/base-driver').MethodMap} [newMethodMap]\n */\n\n/**\n * @typedef {Object} StaticPluginMembers\n * @property {string} pluginName\n */\n\n/**\n * @typedef {import('type-fest').Class<unknown> & StaticPluginMembers & StaticExtMembers} PluginExtensionClass\n */\n\n/**\n * @typedef {import('type-fest').Class<unknown> & StaticExtMembers} DriverExtensionClass\n */\n"],"file":"lib/appium.js","sourceRoot":"../.."}
570
+ exports.NoDriverProxyCommandError = NoDriverProxyCommandError;
571
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../lib/appium.js"],"names":["desiredCapabilityConstraints","Object","freeze","automationName","presence","isString","platformName","sessionsListGuard","AsyncLock","pendingDriversGuard","AppiumDriver","DriverCore","sessions","pendingDrivers","newCommandTimeoutMs","pluginClasses","sessionPlugins","sessionlessPlugins","driverConfig","server","constructor","opts","tmpDir","process","env","APPIUM_TMP_DIR","desiredCapConstraints","args","catch","err","log","debug","_log","instanceName","name","node","getObjectId","substring","logger","getLogger","isCommandsQueueEnabled","sessionExists","sessionId","dstSession","driverForSession","getStatus","build","_","clone","getSessions","acquire","toPairs","map","id","driver","capabilities","caps","printNewSessionAnnouncement","driverName","driverVersion","driverBaseVersion","info","APPIUM_VER","baseVersion","assignCliArgsToExtension","extType","extName","extInstance","allCliArgsForExt","isEmpty","defaults","cliArgs","omitBy","value","key","isEqual","createSession","jsonwpCaps","reqCaps","w3cCapabilities","driverData","defaultCapabilities","cloneDeep","defaultSettings","jwpSettings","w3cSettings","alwaysMatch","firstMatchEntry","firstMatch","assign","protocol","innerSessionId","dCaps","parsedCaps","desiredCaps","processedJsonwpCapabilities","processedW3CCapabilities","error","InnerDriver","version","findMatchingDriver","sessionOverride","deleteAllSessions","runningDriversData","otherPendingDriversData","driverInstance","relaxedSecurityEnabled","denyInsecure","a","allowInsecure","assignServer","address","port","basePath","curSessionDataForDriver","e","errors","SessionNotCreatedError","message","compact","drv","push","pull","attachUnexpectedShutdownHandler","startNewCommandTimeout","isW3CProtocol","JSON","stringify","updateSettings","isMjsonwpProtocol","onShutdown","cause","Error","warn","plugin","isFunction","onUnexpectedShutdown","data","values","filter","s","datum","deleteSession","otherSessionsData","curConstructorName","sessionsCount","size","force","reason","util","pluralize","cleanupPromises","startUnexpectedShutdown","keys","cleanupPromise","pluginsForSession","createPluginInstances","pluginsToHandleCmd","cmd","p","handle","PluginClass","pluginName","executeCommand","isGetStatus","GET_STATUS_COMMAND","isDeleteSession","DELETE_SESSION_COMMAND","isUmbrellaCmd","isAppiumDriverCommand","isSessionCmd","reqForProxy","last","pop","plugins","cmdHandledBy","default","defaultBehavior","length","proxyCommand","NoDriverProxyCommandError","originalUrl","method","body","BaseDriver","prototype","call","wrappedCmd","wrapCommandWithPlugins","next","res","executeWrappedCommand","logPluginHandlerReport","CREATE_SESSION_COMMAND","first","_next","didHandle","k","didntHandle","cmdRes","cmdErr","isPlainObject","has","proxyActive","getProxyAvoidList","canProxy","code"],"mappings":"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","sourcesContent":["/* eslint-disable no-unused-vars */\nimport _ from 'lodash';\nimport { getBuildInfo, updateBuildInfo, APPIUM_VER } from './config';\nimport { BaseDriver, DriverCore, errors, isSessionCommand,\n         CREATE_SESSION_COMMAND, DELETE_SESSION_COMMAND, GET_STATUS_COMMAND\n} from '@appium/base-driver';\nimport AsyncLock from 'async-lock';\nimport { parseCapsForInnerDriver, pullSettings } from './utils';\nimport { util, node, logger } from '@appium/support';\nimport { getDefaultsForExtension } from './schema';\n\n/**\n * Invariant set of base constraints\n * @type {Readonly<Constraints>}\n */\nconst desiredCapabilityConstraints = Object.freeze({\n  automationName: {\n    presence: true,\n    isString: true,\n  },\n  platformName: {\n    presence: true,\n    isString: true,\n  },\n});\n\nconst sessionsListGuard = new AsyncLock();\nconst pendingDriversGuard = new AsyncLock();\n\n/**\n * @implements {SessionHandler}\n */\nclass AppiumDriver extends DriverCore {\n  /**\n   * Access to sessions list must be guarded with a Semaphore, because\n   * it might be changed by other async calls at any time\n   * It is not recommended to access this property directly from the outside\n   * @type {Record<string,ExternalDriver>}\n   */\n  sessions = {};\n\n  /**\n   * Access to pending drivers list must be guarded with a Semaphore, because\n   * it might be changed by other async calls at any time\n   * It is not recommended to access this property directly from the outside\n   * @type {Record<string,ExternalDriver[]>}\n   */\n  pendingDrivers = {};\n\n  /**\n   * Note that {@linkcode AppiumDriver} has no `newCommandTimeout` method.\n   * `AppiumDriver` does not set and observe its own timeouts; individual\n   * sessions (managed drivers) do instead.\n   */\n  newCommandTimeoutMs = 0;\n\n  /**\n   * List of active plugins\n   * @type {PluginClass[]}\n   */\n  pluginClasses = [];\n\n  /**\n   * map of sessions to actual plugin instances per session\n   * @type {Record<string,InstanceType<PluginClass>[]>}\n   */\n  sessionPlugins = {};\n\n  /**\n   * some commands are sessionless, so we need a set of plugins for them\n   * @type {InstanceType<PluginClass>[]}\n   */\n  sessionlessPlugins = [];;\n\n  /** @type {DriverConfig} */\n  driverConfig;\n\n  /** @type {AppiumServer} */\n  server;\n\n  /**\n   * @param {DriverOpts} opts\n   */\n  constructor (opts) {\n    // It is necessary to set `--tmp` here since it should be set to\n    // process.env.APPIUM_TMP_DIR once at an initial point in the Appium lifecycle.\n    // The process argument will be referenced by BaseDriver.\n    // Please call @appium/support.tempDir module to apply this benefit.\n    if (opts.tmpDir) {\n      process.env.APPIUM_TMP_DIR = opts.tmpDir;\n    }\n\n    super(opts);\n\n    this.desiredCapConstraints = desiredCapabilityConstraints;\n\n    this.args = {...opts};\n\n    // allow this to happen in the background, so no `await`\n    // catch this to avoid an unhandled rejection\n    // eslint-disable-next-line promise/prefer-await-to-then,promise/prefer-await-to-callbacks\n    updateBuildInfo().catch((err) => {\n      this.log.debug(err);\n    });\n  }\n\n  /**\n   * Retrieves logger instance for the current umbrella driver instance\n   */\n  get log () {\n    if (!this._log) {\n      const instanceName = `${this.constructor.name}@${node.getObjectId(this).substring(0, 4)}`;\n      this._log = logger.getLogger(instanceName);\n    }\n    return this._log;\n  }\n\n\n  /**\n   * Cancel commands queueing for the umbrella Appium driver\n   */\n  get isCommandsQueueEnabled () {\n    return false;\n  }\n\n  sessionExists (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && dstSession.sessionId !== null;\n  }\n\n  driverForSession (sessionId) {\n    return this.sessions[sessionId];\n  }\n\n  async getStatus () { // eslint-disable-line require-await\n    return {\n      build: _.clone(getBuildInfo()),\n    };\n  }\n\n  async getSessions () {\n    const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);\n    return _.toPairs(sessions)\n      .map(([id, driver]) => ({id, capabilities: driver.caps}));\n  }\n\n  printNewSessionAnnouncement (driverName, driverVersion, driverBaseVersion) {\n    this.log.info(driverVersion\n      ? `Appium v${APPIUM_VER} creating new ${driverName} (v${driverVersion}) session`\n      : `Appium v${APPIUM_VER} creating new ${driverName} session`\n    );\n    this.log.info(`Checking BaseDriver versions for Appium and ${driverName}`);\n    this.log.info(AppiumDriver.baseVersion\n      ? `Appium's BaseDriver version is ${AppiumDriver.baseVersion}`\n      : `Could not determine Appium's BaseDriver version`\n    );\n    this.log.info(driverBaseVersion\n      ? `${driverName}'s BaseDriver version is ${driverBaseVersion}`\n      : `Could not determine ${driverName}'s BaseDriver version`\n    );\n  }\n\n  /**\n   * Validate and assign CLI args for a driver or plugin\n   *\n   * If the extension has provided a schema, validation has already happened.\n   *\n   * Any arg which is equal to its default value will not be assigned to the extension.\n   * @param {ExtensionType} extType 'driver' or 'plugin'\n   * @param {string} extName the name of the extension\n   * @param {Object} extInstance the driver or plugin instance\n   */\n  assignCliArgsToExtension (extType, extName, extInstance) {\n    const allCliArgsForExt = /** @type {Record<string,unknown>|undefined} */(this.args[extType]?.[extName]);\n    if (!_.isEmpty(allCliArgsForExt)) {\n      const defaults = getDefaultsForExtension(extType, extName);\n      const cliArgs = _.isEmpty(defaults)\n        ? allCliArgsForExt\n        : _.omitBy(allCliArgsForExt, (value, key) => _.isEqual(defaults[key], value));\n      if (!_.isEmpty(cliArgs)) {\n        extInstance.cliArgs = cliArgs;\n      }\n    }\n  }\n\n  /**\n   * Create a new session\n   * @param {W3CCapabilities} jsonwpCaps JSONWP formatted desired capabilities\n   * @param {W3CCapabilities} reqCaps Required capabilities (JSONWP standard)\n   * @param {W3CCapabilities} w3cCapabilities W3C capabilities\n   * @param {import('@appium/types').DriverData[]} [driverData]\n   */\n  async createSession (jsonwpCaps, reqCaps, w3cCapabilities, driverData) {\n    const defaultCapabilities = _.cloneDeep(this.args.defaultCapabilities);\n    const defaultSettings = pullSettings(defaultCapabilities);\n    jsonwpCaps = _.cloneDeep(jsonwpCaps);\n    const jwpSettings = {...defaultSettings, ...pullSettings(jsonwpCaps)};\n    w3cCapabilities = _.cloneDeep(w3cCapabilities);\n    // It is possible that the client only provides caps using JSONWP standard,\n    // although firstMatch/alwaysMatch properties are still present.\n    // In such case we assume the client understands W3C protocol and merge the given\n    // JSONWP caps to W3C caps\n    const w3cSettings = {\n      ...jwpSettings,\n      ...pullSettings((w3cCapabilities ?? {}).alwaysMatch ?? {})\n    };\n    for (const firstMatchEntry of ((w3cCapabilities ?? {}).firstMatch ?? [])) {\n      Object.assign(w3cSettings, pullSettings(firstMatchEntry));\n    }\n\n    let protocol;\n    let innerSessionId, dCaps;\n    try {\n      // Parse the caps into a format that the InnerDriver will accept\n      const parsedCaps = parseCapsForInnerDriver(\n        jsonwpCaps,\n        w3cCapabilities,\n        this.desiredCapConstraints,\n        defaultCapabilities\n      );\n\n      const {desiredCaps, processedJsonwpCapabilities, processedW3CCapabilities} = /** @type {import('./utils').ParsedDriverCaps} */(parsedCaps);\n      protocol = parsedCaps.protocol;\n      const error = /** @type {import('./utils').InvalidCaps} */(parsedCaps).error;\n      // If the parsing of the caps produced an error, throw it in here\n      if (error) {\n        throw error;\n      }\n\n      const {\n        driver: InnerDriver,\n        version: driverVersion,\n        driverName\n      } = this.driverConfig.findMatchingDriver(desiredCaps);\n      this.printNewSessionAnnouncement(InnerDriver.name, driverVersion, InnerDriver.baseVersion);\n\n      if (this.args.sessionOverride) {\n        await this.deleteAllSessions();\n      }\n\n      /**\n       * @type {DriverData[]}\n       */\n      let runningDriversData = [];\n      /**\n       * @type {DriverData[]}\n       */\n      let otherPendingDriversData = [];\n\n      const driverInstance = new InnerDriver(this.args, true);\n\n      // We want to assign security values directly on the driver. The driver\n      // should not read security values from `this.opts` because those values\n      // could have been set by a malicious user via capabilities, whereas we\n      // want a guarantee the values were set by the appium server admin\n      if (this.args.relaxedSecurityEnabled) {\n        this.log.info(`Applying relaxed security to '${InnerDriver.name}' as per ` +\n          `server command line argument. All insecure features will be ` +\n          `enabled unless explicitly disabled by --deny-insecure`);\n        driverInstance.relaxedSecurityEnabled = true;\n      }\n\n      if (!_.isEmpty(this.args.denyInsecure)) {\n        this.log.info('Explicitly preventing use of insecure features:');\n        this.args.denyInsecure.map((a) => this.log.info(`    ${a}`));\n        driverInstance.denyInsecure = this.args.denyInsecure;\n      }\n\n      if (!_.isEmpty(this.args.allowInsecure)) {\n        this.log.info('Explicitly enabling use of insecure features:');\n        this.args.allowInsecure.map((a) => this.log.info(`    ${a}`));\n        driverInstance.allowInsecure = this.args.allowInsecure;\n      }\n\n      // Likewise, any driver-specific CLI args that were passed in should be assigned directly to\n      // the driver so that they cannot be mimicked by a malicious user sending in capabilities\n      this.assignCliArgsToExtension('driver', driverName, driverInstance);\n\n\n      // This assignment is required for correct web sockets functionality inside the driver\n      // Drivers/plugins might also want to know where they are hosted\n      driverInstance.assignServer(this.server, this.args.address, this.args.port, this.args.basePath);\n\n      try {\n        runningDriversData = await this.curSessionDataForDriver(InnerDriver) ?? [];\n      } catch (e) {\n        throw new errors.SessionNotCreatedError(e.message);\n      }\n      await pendingDriversGuard.acquire(AppiumDriver.name, () => {\n        this.pendingDrivers[InnerDriver.name] = this.pendingDrivers[InnerDriver.name] || [];\n        otherPendingDriversData = _.compact(this.pendingDrivers[InnerDriver.name].map((drv) => drv.driverData));\n        this.pendingDrivers[InnerDriver.name].push(driverInstance);\n      });\n\n      try {\n        [innerSessionId, dCaps] = await driverInstance.createSession(\n          processedJsonwpCapabilities,\n          reqCaps,\n          processedW3CCapabilities,\n          [...runningDriversData, ...otherPendingDriversData]\n        );\n        protocol = driverInstance.protocol;\n        await sessionsListGuard.acquire(AppiumDriver.name, () => {\n          this.sessions[innerSessionId] = driverInstance;\n        });\n      } finally {\n        await pendingDriversGuard.acquire(AppiumDriver.name, () => {\n          _.pull(this.pendingDrivers[InnerDriver.name], driverInstance);\n        });\n      }\n\n      this.attachUnexpectedShutdownHandler(driverInstance, innerSessionId);\n\n      this.log.info(`New ${InnerDriver.name} session created successfully, session ` +\n        `${innerSessionId} added to master session list`);\n\n      // set the New Command Timeout for the inner driver\n      driverInstance.startNewCommandTimeout();\n\n      // apply initial values to Appium settings (if provided)\n      if (driverInstance.isW3CProtocol() && !_.isEmpty(w3cSettings)) {\n        this.log.info(`Applying the initial values to Appium settings parsed from W3C caps: ` +\n          JSON.stringify(w3cSettings));\n        await driverInstance.updateSettings(w3cSettings);\n      } else if (driverInstance.isMjsonwpProtocol() && !_.isEmpty(jwpSettings)) {\n        this.log.info(`Applying the initial values to Appium settings parsed from MJSONWP caps: ` +\n          JSON.stringify(jwpSettings));\n        await driverInstance.updateSettings(jwpSettings);\n      }\n    } catch (error) {\n      return {\n        protocol,\n        error,\n      };\n    }\n\n    return {\n      protocol,\n      value: [innerSessionId, dCaps, protocol]\n    };\n  }\n\n  attachUnexpectedShutdownHandler (driver, innerSessionId) {\n    const onShutdown = (cause = new Error('Unknown error')) => {\n      this.log.warn(`Ending session, cause was '${cause.message}'`);\n\n      if (this.sessionPlugins[innerSessionId]) {\n        for (const plugin of this.sessionPlugins[innerSessionId]) {\n          if (_.isFunction(plugin.onUnexpectedShutdown)) {\n            this.log.debug(`Plugin ${plugin.name} defines an unexpected shutdown handler; calling it now`);\n            try {\n              plugin.onUnexpectedShutdown(driver, cause);\n            } catch (e) {\n              this.log.warn(`Got an error when running plugin ${plugin.name} shutdown handler: ${e}`);\n            }\n          } else {\n            this.log.debug(`Plugin ${plugin.name} does not define an unexpected shutdown handler`);\n          }\n        }\n      }\n\n      this.log.info(`Removing session '${innerSessionId}' from our master session list`);\n      delete this.sessions[innerSessionId];\n      delete this.sessionPlugins[innerSessionId];\n    };\n\n    if (_.isFunction(driver.onUnexpectedShutdown)) {\n      driver.onUnexpectedShutdown(onShutdown);\n    } else {\n      this.log.warn(`Failed to attach the unexpected shutdown listener. ` +\n        `Is 'onUnexpectedShutdown' method available for '${driver.constructor.name}'?`);\n    }\n  }\n\n  /**\n   *\n   * @param {import('../types/extension').DriverClass} InnerDriver\n   * @returns {Promise<DriverData[]>}}\n   */\n  async curSessionDataForDriver (InnerDriver) {\n    const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);\n    const data = _.compact(_.values(sessions)\n                   .filter((s) => s.constructor.name === InnerDriver.name)\n                   .map((s) => s.driverData));\n    for (let datum of data) {\n      if (!datum) {\n        throw new Error(`Problem getting session data for driver type ` +\n                        `${InnerDriver.name}; does it implement 'get ` +\n                        `driverData'?`);\n      }\n    }\n    return data;\n  }\n\n  /**\n   * @param {string} sessionId\n   */\n  async deleteSession (sessionId) {\n    let protocol;\n    try {\n      let otherSessionsData;\n      const dstSession = await sessionsListGuard.acquire(AppiumDriver.name, () => {\n        if (!this.sessions[sessionId]) {\n          return;\n        }\n        const curConstructorName = this.sessions[sessionId].constructor.name;\n        otherSessionsData = _.toPairs(this.sessions)\n              .filter(([key, value]) => value.constructor.name === curConstructorName && key !== sessionId)\n              .map(([, value]) => value.driverData);\n        const dstSession = this.sessions[sessionId];\n        protocol = dstSession.protocol;\n        this.log.info(`Removing session ${sessionId} from our master session list`);\n        // regardless of whether the deleteSession completes successfully or not\n        // make the session unavailable, because who knows what state it might\n        // be in otherwise\n        delete this.sessions[sessionId];\n        delete this.sessionPlugins[sessionId];\n        return dstSession;\n      });\n      // this may not be correct, but if `dstSession` was falsy, the call to `deleteSession()` would\n      // throw anyway.\n      if (!dstSession) {\n        throw new Error('Session not found');\n      }\n      return {\n        protocol,\n        value: await dstSession.deleteSession(sessionId, otherSessionsData),\n      };\n    } catch (e) {\n      this.log.error(`Had trouble ending session ${sessionId}: ${e.message}`);\n      return {\n        protocol,\n        error: e,\n      };\n    }\n  }\n\n  async deleteAllSessions (opts = {}) {\n    const sessionsCount = _.size(this.sessions);\n    if (0 === sessionsCount) {\n      this.log.debug('There are no active sessions for cleanup');\n      return;\n    }\n\n    const {\n      force = false,\n      reason,\n    } = opts;\n    this.log.debug(`Cleaning up ${util.pluralize('active session', sessionsCount, true)}`);\n    const cleanupPromises = force\n      ? _.values(this.sessions).map((drv) => drv.startUnexpectedShutdown(reason && new Error(reason)))\n      : _.keys(this.sessions).map((id) => this.deleteSession(id));\n    for (const cleanupPromise of cleanupPromises) {\n      try {\n        await cleanupPromise;\n      } catch (e) {\n        this.log.debug(e);\n      }\n    }\n  }\n\n  /**\n   * Get the appropriate plugins for a session (or sessionless plugins)\n   *\n   * @param {?string} sessionId - the sessionId (or null) to use to find plugins\n   * @returns {Array} - array of plugin instances\n   */\n  pluginsForSession (sessionId = null) {\n    if (sessionId) {\n      if (!this.sessionPlugins[sessionId]) {\n        this.sessionPlugins[sessionId] = this.createPluginInstances();\n      }\n      return this.sessionPlugins[sessionId];\n    }\n\n    if (_.isEmpty(this.sessionlessPlugins)) {\n      this.sessionlessPlugins = this.createPluginInstances();\n    }\n    return this.sessionlessPlugins;\n  }\n\n  /**\n   * To get plugins for a command, we either get the plugin instances associated with the\n   * particular command's session, or in the case of sessionless plugins, pull from the set of\n   * plugin instances reserved for sessionless commands (and we lazily create plugin instances on\n   * first use)\n   *\n   * @param {string} cmd - the name of the command to find a plugin to handle\n   * @param {?string} sessionId - the particular session for which to find a plugin, or null if\n   * sessionless\n   */\n  pluginsToHandleCmd (cmd, sessionId = null) {\n    // to handle a given command, a plugin should either implement that command as a plugin\n    // instance method or it should implement a generic 'handle' method\n    return this.pluginsForSession(sessionId)\n      .filter((p) => _.isFunction(p[cmd]) || _.isFunction(p.handle));\n  }\n\n  createPluginInstances () {\n    return this.pluginClasses.map((PluginClass) => {\n      const name = PluginClass.pluginName;\n      const plugin = new PluginClass(name);\n      this.assignCliArgsToExtension('plugin', name, plugin);\n      return plugin;\n    });\n  }\n\n  /**\n   *\n   * @param {string} cmd\n   * @param  {...any} args\n   * @returns {Promise<{value: any, error?: Error, protocol: string} | import('type-fest').AsyncReturnType<import('@appium/types').Driver['executeCommand']>>}\n   */\n  async executeCommand (cmd, ...args) {\n    // We have basically three cases for how to handle commands:\n    // 1. handle getStatus (we do this as a special out of band case so it doesn't get added to an\n    //    execution queue, and can be called while e.g. createSession is in progress)\n    // 2. handle commands that this umbrella driver should handle, rather than the actual session\n    //    driver (for example, deleteSession, or other non-session commands)\n    // 3. handle session driver commands.\n    // The tricky part is that because we support command plugins, we need to wrap any of these\n    // cases with plugin handling.\n\n    const isGetStatus = cmd === GET_STATUS_COMMAND;\n    const isDeleteSession = cmd === DELETE_SESSION_COMMAND;\n    const isUmbrellaCmd = !isGetStatus && isAppiumDriverCommand(cmd);\n    const isSessionCmd = !isUmbrellaCmd || isDeleteSession;\n\n    // if a plugin override proxying for this command and that is why we are here instead of just\n    // letting the protocol proxy the command entirely, determine that, get the request object for\n    // use later on, then clean up the args\n    const reqForProxy = _.last(args)?.reqForProxy;\n    if (reqForProxy) {\n      args.pop();\n    }\n\n\n    // first do some error checking. If we're requesting a session command execution, then make\n    // sure that session actually exists on the session driver, and set the session driver itself\n    let sessionId = null;\n    let dstSession = null;\n    let protocol = null;\n    /** @type {this | ExternalDriver} */\n    let driver = this;\n    if (isSessionCmd) {\n      sessionId = _.last(args);\n      dstSession = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions[sessionId]);\n      if (!dstSession) {\n        throw new Error(`The session with id '${sessionId}' does not exist`);\n      }\n      // now save the response protocol given that the session driver's protocol might differ\n      protocol = dstSession.protocol;\n      if (!isUmbrellaCmd) {\n        driver = dstSession;\n      }\n    }\n\n    // get any plugins which are registered as handling this command\n    const plugins = this.pluginsToHandleCmd(cmd, sessionId);\n\n    // now we define a 'cmdHandledBy' object which will keep track of which plugins have handled this\n    // command. we care about this because (a) multiple plugins can handle the same command, and\n    // (b) there's no guarantee that a plugin will actually call the next() method which runs the\n    // original command execution. This results in a situation where the command might be handled\n    // by some but not all plugins, or by plugin(s) but not by the default behavior. So start out\n    // this object declaring that the default handler has not been executed.\n    const cmdHandledBy = {default: false};\n\n    // now we define an async function which will be passed to plugins, and successively wrapped\n    // if there is more than one plugin that can handle the command. To start off with, the async\n    // function is defined as calling the default behavior, i.e., whichever of the 3 cases above is\n    // the appropriate one\n    const defaultBehavior = async () => {\n      // if we're running with plugins, make sure we log that the default behavior is actually\n      // happening so we can tell when the plugin call chain is unwrapping to the default behavior\n      // if that's what happens\n      plugins.length && this.log.info(`Executing default handling behavior for command '${cmd}'`);\n\n      // if we make it here, we know that the default behavior is handled\n      cmdHandledBy.default = true;\n\n      if (reqForProxy) {\n        // we would have proxied this command had a plugin not handled it, so the default behavior\n        // is to do the proxy and retrieve the result internally so it can be passed to the plugin\n        // in case it calls 'await next()'. This requires that the driver have defined\n        // 'proxyCommand' and not just 'proxyReqRes'.\n        if (!dstSession.proxyCommand) {\n          throw new NoDriverProxyCommandError();\n        }\n        return await dstSession.proxyCommand(reqForProxy.originalUrl, reqForProxy.method,\n          reqForProxy.body);\n      }\n\n      if (isGetStatus) {\n        return await this.getStatus();\n      }\n\n      if (isUmbrellaCmd) {\n        // some commands, like deleteSession, we want to make sure to handle on *this* driver,\n        // not the platform driver\n        return await BaseDriver.prototype.executeCommand.call(this, cmd, ...args);\n      }\n\n      // here we know that we are executing a session command, and have a valid session driver\n      return await dstSession.executeCommand(cmd, ...args);\n    };\n\n    // now take our default behavior, wrap it with any number of plugin behaviors, and run it\n    const wrappedCmd = this.wrapCommandWithPlugins({\n      driver, cmd, args, plugins, cmdHandledBy, next: defaultBehavior\n    });\n    const res = await this.executeWrappedCommand({wrappedCmd, protocol});\n\n    // if we had plugins, make sure to log out the helpful report about which plugins ended up\n    // handling the command and which didn't\n    this.logPluginHandlerReport(plugins, {cmd, cmdHandledBy});\n\n    // And finally, if the command was createSession, we want to migrate any plugins which were\n    // previously sessionless to use the new sessionId, so that plugins can share state between\n    // their createSession method and other instance methods\n    if (cmd === CREATE_SESSION_COMMAND && this.sessionlessPlugins.length && !res.error) {\n      const sessionId = _.first(res.value);\n      this.log.info(`Promoting ${this.sessionlessPlugins.length} sessionless plugins to be attached ` +\n        `to session ID ${sessionId}`);\n      this.sessionPlugins[sessionId] = this.sessionlessPlugins;\n      this.sessionlessPlugins = [];\n    }\n\n    return res;\n  }\n\n  wrapCommandWithPlugins ({driver, cmd, args, next, cmdHandledBy, plugins}) {\n    plugins.length && this.log.info(`Plugins which can handle cmd '${cmd}': ${plugins.map((p) => p.name)}`);\n\n    // now we can go through each plugin and wrap `next` around its own handler, passing the *old*\n    // next in so that it can call it if it wants to\n    for (const plugin of plugins) {\n      // need an IIFE here because we want the value of next that's passed to plugin.handle to be\n      // exactly the value of next here before reassignment; we don't want it to be lazily\n      // evaluated, otherwise we end up with infinite recursion of the last `next` to be defined.\n      cmdHandledBy[plugin.name] = false; // we see a new plugin, so add it to the 'cmdHandledBy' object\n      next = ((_next) => async () => {\n        this.log.info(`Plugin ${plugin.name} is now handling cmd '${cmd}'`);\n        cmdHandledBy[plugin.name] = true; // if we make it here, this plugin has attempted to handle cmd\n        // first attempt to handle the command via a command-specific handler on the plugin\n        if (plugin[cmd]) {\n          return await plugin[cmd](_next, driver, ...args);\n        }\n        // otherwise, call the generic 'handle' method\n        return await plugin.handle(_next, driver, cmd, ...args);\n      })(next);\n    }\n\n    return next;\n  }\n\n  logPluginHandlerReport (plugins, {cmd, cmdHandledBy}) {\n    if (!plugins.length) {\n      return;\n    }\n\n    // at the end of the day, we have an object representing which plugins ended up getting\n    // their code run as part of handling this command. Because plugins can choose *not* to\n    // pass control to other plugins or to the default driver behavior, this is information\n    // which is probably useful to the user (especially in situations where plugins might not\n    // interact well together, and it would be hard to debug otherwise without this kind of\n    // message).\n    const didHandle = Object.keys(cmdHandledBy).filter((k) => cmdHandledBy[k]);\n    const didntHandle = Object.keys(cmdHandledBy).filter((k) => !cmdHandledBy[k]);\n    if (didntHandle.length > 0) {\n      this.log.info(`Command '${cmd}' was *not* handled by the following behaviours or plugins, even ` +\n        `though they were registered to handle it: ${JSON.stringify(didntHandle)}. The ` +\n        `command *was* handled by these: ${JSON.stringify(didHandle)}.`);\n    }\n  }\n\n  async executeWrappedCommand ({wrappedCmd, protocol}) {\n    let cmdRes, cmdErr, res = {};\n    try {\n      // At this point, `wrappedCmd` defines a whole sequence of plugin handlers, culminating in\n      // our default handler. Whatever it returns is what we're going to want to send back to the\n      // user.\n      cmdRes = await wrappedCmd();\n    } catch (e) {\n      cmdErr = e;\n    }\n\n    // Sadly, we don't know exactly what kind of object will be returned. It will either be a bare\n    // object, or a protocol-aware object with protocol and error/value keys. So we need to sniff\n    // it and make sure we don't double-wrap it if it's the latter kind.\n    if (_.isPlainObject(cmdRes) && _.has(cmdRes, 'protocol')) {\n      res = cmdRes;\n    } else {\n      res.value = cmdRes;\n      res.error = cmdErr;\n      res.protocol = protocol;\n    }\n    return res;\n  }\n\n  proxyActive (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && _.isFunction(dstSession.proxyActive) && dstSession.proxyActive(sessionId);\n  }\n\n  getProxyAvoidList (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession ? dstSession.getProxyAvoidList() : [];\n  }\n\n  canProxy (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && dstSession.canProxy(sessionId);\n  }\n}\n\n// help decide which commands should be proxied to sub-drivers and which\n// should be handled by this, our umbrella driver\nfunction isAppiumDriverCommand (cmd) {\n  return !isSessionCommand(cmd) || cmd === 'deleteSession';\n}\n\n/**\n * Thrown when Appium tried to proxy a command using a driver's `proxyCommand` method but the\n * method did not exist\n */\nexport class NoDriverProxyCommandError extends Error {\n  /**\n   * @type {Readonly<string>}\n   */\n  code = 'APPIUMERR_NO_DRIVER_PROXYCOMMAND';\n\n  constructor () {\n    super(`The default behavior for this command was to proxy, but the driver ` +\n          `did not have the 'proxyCommand' method defined. To fully support ` +\n          `plugins, drivers should have 'proxyCommand' set to a jwpProxy object's ` +\n          `'command()' method, in addition to the normal 'proxyReqRes'`);\n  }\n}\n\nexport { AppiumDriver };\n\n/**\n * @typedef {import('@appium/types').ExternalDriver} ExternalDriver\n * @typedef {import('@appium/types').W3CCapabilities} W3CCapabilities\n * @typedef {import('@appium/types').DriverData} DriverData\n * @typedef {import('@appium/types').DriverOpts} DriverOpts\n * @typedef {import('@appium/types').Constraints} Constraints\n * @typedef {import('@appium/types').AppiumServer} AppiumServer\n * @typedef {import('../types').ExtensionType} ExtensionType\n * @typedef {import('../types/extension').PluginClass} PluginClass\n * @typedef {import('./extension/driver-config').DriverConfig} DriverConfig\n */\n\n/**\n * Used by {@linkcode AppiumDriver.createSession} and {@linkcode AppiumDriver.deleteSession} to describe\n * result.\n * @template V\n * @typedef SessionHandlerResult\n * @property {V} [value]\n * @property {Error} [error]\n * @property {string} [protocol]\n */\n\n/**\n * @typedef {import('@appium/types').SessionHandler<SessionHandlerResult<any[]>,SessionHandlerResult<void>>} SessionHandler\n */\n"]}