appium 2.0.0-beta.6 → 2.0.0-beta.60

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 (204) hide show
  1. package/LICENSE +1 -1
  2. package/README.md +156 -65
  3. package/build/lib/appium.d.ts +229 -0
  4. package/build/lib/appium.d.ts.map +1 -0
  5. package/build/lib/appium.js +678 -439
  6. package/build/lib/appium.js.map +1 -0
  7. package/build/lib/cli/args.d.ts +17 -0
  8. package/build/lib/cli/args.d.ts.map +1 -0
  9. package/build/lib/cli/args.js +263 -319
  10. package/build/lib/cli/args.js.map +1 -0
  11. package/build/lib/cli/driver-command.d.ts +102 -0
  12. package/build/lib/cli/driver-command.d.ts.map +1 -0
  13. package/build/lib/cli/driver-command.js +131 -81
  14. package/build/lib/cli/driver-command.js.map +1 -0
  15. package/build/lib/cli/extension-command.d.ts +402 -0
  16. package/build/lib/cli/extension-command.d.ts.map +1 -0
  17. package/build/lib/cli/extension-command.js +799 -383
  18. package/build/lib/cli/extension-command.js.map +1 -0
  19. package/build/lib/cli/extension.d.ts +23 -0
  20. package/build/lib/cli/extension.d.ts.map +1 -0
  21. package/build/lib/cli/extension.js +71 -54
  22. package/build/lib/cli/extension.js.map +1 -0
  23. package/build/lib/cli/parser.d.ts +84 -0
  24. package/build/lib/cli/parser.d.ts.map +1 -0
  25. package/build/lib/cli/parser.js +240 -128
  26. package/build/lib/cli/parser.js.map +1 -0
  27. package/build/lib/cli/plugin-command.d.ts +99 -0
  28. package/build/lib/cli/plugin-command.d.ts.map +1 -0
  29. package/build/lib/cli/plugin-command.js +125 -81
  30. package/build/lib/cli/plugin-command.js.map +1 -0
  31. package/build/lib/cli/utils.d.ts +29 -0
  32. package/build/lib/cli/utils.d.ts.map +1 -0
  33. package/build/lib/cli/utils.js +72 -51
  34. package/build/lib/cli/utils.js.map +1 -0
  35. package/build/lib/config-file.d.ts +100 -0
  36. package/build/lib/config-file.d.ts.map +1 -0
  37. package/build/lib/config-file.js +207 -0
  38. package/build/lib/config-file.js.map +1 -0
  39. package/build/lib/config.d.ts +49 -0
  40. package/build/lib/config.d.ts.map +1 -0
  41. package/build/lib/config.js +267 -202
  42. package/build/lib/config.js.map +1 -0
  43. package/build/lib/constants.d.ts +56 -0
  44. package/build/lib/constants.d.ts.map +1 -0
  45. package/build/lib/constants.js +73 -0
  46. package/build/lib/constants.js.map +1 -0
  47. package/build/lib/extension/driver-config.d.ts +82 -0
  48. package/build/lib/extension/driver-config.d.ts.map +1 -0
  49. package/build/lib/extension/driver-config.js +210 -0
  50. package/build/lib/extension/driver-config.js.map +1 -0
  51. package/build/lib/extension/extension-config.d.ts +270 -0
  52. package/build/lib/extension/extension-config.d.ts.map +1 -0
  53. package/build/lib/extension/extension-config.js +601 -0
  54. package/build/lib/extension/extension-config.js.map +1 -0
  55. package/build/lib/extension/index.d.ts +48 -0
  56. package/build/lib/extension/index.d.ts.map +1 -0
  57. package/build/lib/extension/index.js +105 -0
  58. package/build/lib/extension/index.js.map +1 -0
  59. package/build/lib/extension/manifest-migrations.d.ts +27 -0
  60. package/build/lib/extension/manifest-migrations.d.ts.map +1 -0
  61. package/build/lib/extension/manifest-migrations.js +134 -0
  62. package/build/lib/extension/manifest-migrations.js.map +1 -0
  63. package/build/lib/extension/manifest.d.ts +145 -0
  64. package/build/lib/extension/manifest.d.ts.map +1 -0
  65. package/build/lib/extension/manifest.js +528 -0
  66. package/build/lib/extension/manifest.js.map +1 -0
  67. package/build/lib/extension/package-changed.d.ts +11 -0
  68. package/build/lib/extension/package-changed.d.ts.map +1 -0
  69. package/build/lib/extension/package-changed.js +62 -0
  70. package/build/lib/extension/package-changed.js.map +1 -0
  71. package/build/lib/extension/plugin-config.d.ts +56 -0
  72. package/build/lib/extension/plugin-config.d.ts.map +1 -0
  73. package/build/lib/extension/plugin-config.js +102 -0
  74. package/build/lib/extension/plugin-config.js.map +1 -0
  75. package/build/lib/grid-register.d.ts +10 -0
  76. package/build/lib/grid-register.d.ts.map +1 -0
  77. package/build/lib/grid-register.js +122 -144
  78. package/build/lib/grid-register.js.map +1 -0
  79. package/build/lib/logger.d.ts +3 -0
  80. package/build/lib/logger.d.ts.map +1 -0
  81. package/build/lib/logger.js +5 -17
  82. package/build/lib/logger.js.map +1 -0
  83. package/build/lib/logsink.d.ts +4 -0
  84. package/build/lib/logsink.d.ts.map +1 -0
  85. package/build/lib/logsink.js +190 -187
  86. package/build/lib/logsink.js.map +1 -0
  87. package/build/lib/main.d.ts +62 -0
  88. package/build/lib/main.d.ts.map +1 -0
  89. package/build/lib/main.js +339 -229
  90. package/build/lib/main.js.map +1 -0
  91. package/build/lib/schema/arg-spec.d.ts +143 -0
  92. package/build/lib/schema/arg-spec.d.ts.map +1 -0
  93. package/build/lib/schema/arg-spec.js +164 -0
  94. package/build/lib/schema/arg-spec.js.map +1 -0
  95. package/build/lib/schema/cli-args.d.ts +19 -0
  96. package/build/lib/schema/cli-args.d.ts.map +1 -0
  97. package/build/lib/schema/cli-args.js +217 -0
  98. package/build/lib/schema/cli-args.js.map +1 -0
  99. package/build/lib/schema/cli-transformers.d.ts +5 -0
  100. package/build/lib/schema/cli-transformers.d.ts.map +1 -0
  101. package/build/lib/schema/cli-transformers.js +124 -0
  102. package/build/lib/schema/cli-transformers.js.map +1 -0
  103. package/build/lib/schema/index.d.ts +3 -0
  104. package/build/lib/schema/index.d.ts.map +1 -0
  105. package/build/lib/schema/index.js +19 -0
  106. package/build/lib/schema/index.js.map +1 -0
  107. package/build/lib/schema/keywords.d.ts +24 -0
  108. package/build/lib/schema/keywords.d.ts.map +1 -0
  109. package/build/lib/schema/keywords.js +128 -0
  110. package/build/lib/schema/keywords.js.map +1 -0
  111. package/build/lib/schema/schema.d.ts +260 -0
  112. package/build/lib/schema/schema.d.ts.map +1 -0
  113. package/build/lib/schema/schema.js +640 -0
  114. package/build/lib/schema/schema.js.map +1 -0
  115. package/build/lib/utils.d.ts +266 -0
  116. package/build/lib/utils.d.ts.map +1 -0
  117. package/build/lib/utils.js +349 -273
  118. package/build/lib/utils.js.map +1 -0
  119. package/build/types/cli.d.ts +134 -0
  120. package/build/types/cli.d.ts.map +1 -0
  121. package/build/types/cli.js +3 -0
  122. package/build/types/cli.js.map +1 -0
  123. package/build/types/index.d.ts +15 -0
  124. package/build/types/index.d.ts.map +1 -0
  125. package/build/types/index.js +19 -0
  126. package/build/types/index.js.map +1 -0
  127. package/build/types/manifest/base.d.ts +135 -0
  128. package/build/types/manifest/base.d.ts.map +1 -0
  129. package/build/types/manifest/base.js +3 -0
  130. package/build/types/manifest/base.js.map +1 -0
  131. package/build/types/manifest/index.d.ts +21 -0
  132. package/build/types/manifest/index.d.ts.map +1 -0
  133. package/build/types/manifest/index.js +42 -0
  134. package/build/types/manifest/index.js.map +1 -0
  135. package/build/types/manifest/v3.d.ts +139 -0
  136. package/build/types/manifest/v3.d.ts.map +1 -0
  137. package/build/types/manifest/v3.js +3 -0
  138. package/build/types/manifest/v3.js.map +1 -0
  139. package/build/types/manifest/v4.d.ts +139 -0
  140. package/build/types/manifest/v4.d.ts.map +1 -0
  141. package/build/types/manifest/v4.js +3 -0
  142. package/build/types/manifest/v4.js.map +1 -0
  143. package/driver.d.ts +1 -0
  144. package/driver.js +14 -0
  145. package/index.js +11 -0
  146. package/lib/appium.js +558 -186
  147. package/lib/cli/args.js +277 -422
  148. package/lib/cli/driver-command.js +132 -24
  149. package/lib/cli/extension-command.js +751 -272
  150. package/lib/cli/extension.js +49 -18
  151. package/lib/cli/parser.js +263 -83
  152. package/lib/cli/plugin-command.js +122 -22
  153. package/lib/cli/utils.js +24 -10
  154. package/lib/config-file.js +220 -0
  155. package/lib/config.js +246 -111
  156. package/lib/constants.js +79 -0
  157. package/lib/extension/driver-config.js +247 -0
  158. package/lib/extension/extension-config.js +709 -0
  159. package/lib/extension/index.js +116 -0
  160. package/lib/extension/manifest-migrations.js +136 -0
  161. package/lib/extension/manifest.js +580 -0
  162. package/lib/extension/package-changed.js +64 -0
  163. package/lib/extension/plugin-config.js +112 -0
  164. package/lib/grid-register.js +49 -35
  165. package/lib/logger.js +1 -2
  166. package/lib/logsink.js +64 -38
  167. package/lib/main.js +318 -103
  168. package/lib/schema/arg-spec.js +229 -0
  169. package/lib/schema/cli-args.js +238 -0
  170. package/lib/schema/cli-transformers.js +119 -0
  171. package/lib/schema/index.js +2 -0
  172. package/lib/schema/keywords.js +136 -0
  173. package/lib/schema/schema.js +725 -0
  174. package/lib/utils.js +289 -167
  175. package/package.json +84 -84
  176. package/plugin.d.ts +1 -0
  177. package/plugin.js +13 -0
  178. package/scripts/autoinstall-extensions.js +243 -0
  179. package/support.d.ts +1 -0
  180. package/support.js +13 -0
  181. package/tsconfig.json +25 -0
  182. package/types/cli.ts +193 -0
  183. package/types/index.ts +20 -0
  184. package/types/manifest/README.md +30 -0
  185. package/types/manifest/base.ts +158 -0
  186. package/types/manifest/index.ts +28 -0
  187. package/types/manifest/v3.ts +161 -0
  188. package/types/manifest/v4.ts +161 -0
  189. package/CHANGELOG.md +0 -3515
  190. package/bin/ios-webkit-debug-proxy-launcher.js +0 -71
  191. package/build/lib/cli/npm.js +0 -208
  192. package/build/lib/cli/parser-helpers.js +0 -82
  193. package/build/lib/driver-config.js +0 -77
  194. package/build/lib/drivers.js +0 -96
  195. package/build/lib/extension-config.js +0 -253
  196. package/build/lib/plugin-config.js +0 -59
  197. package/build/lib/plugins.js +0 -14
  198. package/lib/cli/npm.js +0 -184
  199. package/lib/cli/parser-helpers.js +0 -79
  200. package/lib/driver-config.js +0 -46
  201. package/lib/drivers.js +0 -81
  202. package/lib/extension-config.js +0 -209
  203. package/lib/plugin-config.js +0 -34
  204. package/lib/plugins.js +0 -10
@@ -1,459 +1,698 @@
1
1
  "use strict";
2
-
3
- var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
-
5
- Object.defineProperty(exports, "__esModule", {
6
- value: true
7
- });
8
- exports.AppiumDriver = void 0;
9
-
10
- require("source-map-support/register");
11
-
12
- var _lodash = _interopRequireDefault(require("lodash"));
13
-
14
- var _logger = _interopRequireDefault(require("./logger"));
15
-
16
- var _config = require("./config");
17
-
18
- var _drivers = require("./drivers");
19
-
20
- var _appiumBaseDriver = require("appium-base-driver");
21
-
22
- var _bluebird = _interopRequireDefault(require("bluebird"));
23
-
24
- var _asyncLock = _interopRequireDefault(require("async-lock"));
25
-
26
- var _utils = require("./utils");
27
-
28
- var _appiumSupport = require("appium-support");
29
-
30
- const desiredCapabilityConstraints = {
31
- automationName: {
32
- presence: true,
33
- isString: true
34
- },
35
- platformName: {
36
- presence: true,
37
- isString: true
38
- }
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
39
4
  };
40
- const sessionsListGuard = new _asyncLock.default();
41
- const pendingDriversGuard = new _asyncLock.default();
42
-
43
- class AppiumDriver extends _appiumBaseDriver.BaseDriver {
44
- constructor(args) {
45
- if (args.tmpDir) {
46
- process.env.APPIUM_TMP_DIR = args.tmpDir;
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.AppiumDriver = exports.NoDriverProxyCommandError = void 0;
7
+ /* eslint-disable no-unused-vars */
8
+ const lodash_1 = __importDefault(require("lodash"));
9
+ const config_1 = require("./config");
10
+ const base_driver_1 = require("@appium/base-driver");
11
+ const async_lock_1 = __importDefault(require("async-lock"));
12
+ const utils_1 = require("./utils");
13
+ const support_1 = require("@appium/support");
14
+ const schema_1 = require("./schema");
15
+ const constants_1 = require("./constants");
16
+ const desiredCapabilityConstraints = /** @type {const} */ ({
17
+ automationName: {
18
+ presence: true,
19
+ isString: true,
20
+ },
21
+ platformName: {
22
+ presence: true,
23
+ isString: true,
24
+ },
25
+ });
26
+ const sessionsListGuard = new async_lock_1.default();
27
+ const pendingDriversGuard = new async_lock_1.default();
28
+ /**
29
+ * @extends {DriverCore<AppiumDriverConstraints>}
30
+ */
31
+ class AppiumDriver extends base_driver_1.DriverCore {
32
+ /**
33
+ * @param {import('@appium/types').DriverOpts<AppiumDriverConstraints>} opts
34
+ */
35
+ constructor(opts) {
36
+ // It is necessary to set `--tmp` here since it should be set to
37
+ // process.env.APPIUM_TMP_DIR once at an initial point in the Appium lifecycle.
38
+ // The process argument will be referenced by BaseDriver.
39
+ // Please call @appium/support.tempDir module to apply this benefit.
40
+ if (opts.tmpDir) {
41
+ process.env.APPIUM_TMP_DIR = opts.tmpDir;
42
+ }
43
+ super(opts);
44
+ this.args = { ...opts };
45
+ this.sessions = {};
46
+ this.pendingDrivers = {};
47
+ this.newCommandTimeoutMs = 0;
48
+ this.pluginClasses = new Map();
49
+ this.sessionPlugins = {};
50
+ this.sessionlessPlugins = [];
51
+ this.desiredCapConstraints = desiredCapabilityConstraints;
52
+ // allow this to happen in the background, so no `await`
53
+ (async () => {
54
+ try {
55
+ await (0, config_1.updateBuildInfo)();
56
+ }
57
+ catch (e) {
58
+ // make sure we catch any possible errors to avoid unhandled rejections
59
+ this.log.debug(`Cannot fetch Appium build info: ${e.message}`);
60
+ }
61
+ })();
47
62
  }
48
-
49
- super(args);
50
- this.desiredCapConstraints = desiredCapabilityConstraints;
51
- this.newCommandTimeoutMs = 0;
52
- this.args = Object.assign({}, args);
53
- this.sessions = {};
54
- this.pendingDrivers = {};
55
- this.plugins = [];
56
- (0, _config.updateBuildInfo)();
57
- }
58
-
59
- get isCommandsQueueEnabled() {
60
- return false;
61
- }
62
-
63
- sessionExists(sessionId) {
64
- const dstSession = this.sessions[sessionId];
65
- return dstSession && dstSession.sessionId !== null;
66
- }
67
-
68
- driverForSession(sessionId) {
69
- return this.sessions[sessionId];
70
- }
71
-
72
- async getStatus() {
73
- return {
74
- build: _lodash.default.clone((0, _config.getBuildInfo)())
75
- };
76
- }
77
-
78
- async getSessions() {
79
- const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);
80
- return _lodash.default.toPairs(sessions).map(([id, driver]) => ({
81
- id,
82
- capabilities: driver.caps
83
- }));
84
- }
85
-
86
- printNewSessionAnnouncement(driverName, driverVersion) {
87
- const introString = driverVersion ? `Appium v${_config.APPIUM_VER} creating new ${driverName} (v${driverVersion}) session` : `Appium v${_config.APPIUM_VER} creating new ${driverName} session`;
88
-
89
- _logger.default.info(introString);
90
- }
91
-
92
- _findMatchingDriver(...args) {
93
- return (0, _drivers.findMatchingDriver)(...args);
94
- }
95
-
96
- async createSession(jsonwpCaps, reqCaps, w3cCapabilities) {
97
- const defaultCapabilities = _lodash.default.cloneDeep(this.args.defaultCapabilities);
98
-
99
- const defaultSettings = (0, _utils.pullSettings)(defaultCapabilities);
100
- jsonwpCaps = _lodash.default.cloneDeep(jsonwpCaps);
101
- const jwpSettings = Object.assign({}, defaultSettings, (0, _utils.pullSettings)(jsonwpCaps));
102
- w3cCapabilities = _lodash.default.cloneDeep(w3cCapabilities);
103
- const w3cSettings = Object.assign({}, jwpSettings);
104
- Object.assign(w3cSettings, (0, _utils.pullSettings)((w3cCapabilities || {}).alwaysMatch || {}));
105
-
106
- for (const firstMatchEntry of (w3cCapabilities || {}).firstMatch || []) {
107
- Object.assign(w3cSettings, (0, _utils.pullSettings)(firstMatchEntry));
63
+ /**
64
+ * Retrieves logger instance for the current umbrella driver instance
65
+ */
66
+ get log() {
67
+ if (!this._log) {
68
+ const instanceName = `${this.constructor.name}@${support_1.node.getObjectId(this).substring(0, 4)}`;
69
+ this._log = support_1.logger.getLogger(instanceName);
70
+ }
71
+ return this._log;
108
72
  }
109
-
110
- let protocol;
111
- let innerSessionId, dCaps;
112
-
113
- try {
114
- const parsedCaps = (0, _utils.parseCapsForInnerDriver)(jsonwpCaps, w3cCapabilities, this.desiredCapConstraints, defaultCapabilities);
115
- const {
116
- desiredCaps,
117
- processedJsonwpCapabilities,
118
- processedW3CCapabilities,
119
- error
120
- } = parsedCaps;
121
- protocol = parsedCaps.protocol;
122
-
123
- if (error) {
124
- throw error;
125
- }
126
-
127
- const {
128
- driver: InnerDriver,
129
- version: driverVersion
130
- } = this._findMatchingDriver(this.driverConfig, desiredCaps);
131
-
132
- this.printNewSessionAnnouncement(InnerDriver.name, driverVersion);
133
-
134
- if (this.args.sessionOverride) {
135
- await this.deleteAllSessions();
136
- }
137
-
138
- let runningDriversData, otherPendingDriversData;
139
- const d = new InnerDriver(this.args);
140
-
141
- if (this.args.relaxedSecurityEnabled) {
142
- _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`);
143
-
144
- d.relaxedSecurityEnabled = true;
145
- }
146
-
147
- if (!_lodash.default.isEmpty(this.args.denyInsecure)) {
148
- _logger.default.info('Explicitly preventing use of insecure features:');
149
-
150
- this.args.denyInsecure.map(a => _logger.default.info(` ${a}`));
151
- d.denyInsecure = this.args.denyInsecure;
152
- }
153
-
154
- if (!_lodash.default.isEmpty(this.args.allowInsecure)) {
155
- _logger.default.info('Explicitly enabling use of insecure features:');
156
-
157
- this.args.allowInsecure.map(a => _logger.default.info(` ${a}`));
158
- d.allowInsecure = this.args.allowInsecure;
159
- }
160
-
161
- d.server = this.server;
162
-
163
- try {
164
- runningDriversData = await this.curSessionDataForDriver(InnerDriver);
165
- } catch (e) {
166
- throw new _appiumBaseDriver.errors.SessionNotCreatedError(e.message);
167
- }
168
-
169
- await pendingDriversGuard.acquire(AppiumDriver.name, () => {
170
- this.pendingDrivers[InnerDriver.name] = this.pendingDrivers[InnerDriver.name] || [];
171
- otherPendingDriversData = this.pendingDrivers[InnerDriver.name].map(drv => drv.driverData);
172
- this.pendingDrivers[InnerDriver.name].push(d);
173
- });
174
-
175
- try {
176
- [innerSessionId, dCaps] = await d.createSession(processedJsonwpCapabilities, reqCaps, processedW3CCapabilities, [...runningDriversData, ...otherPendingDriversData]);
177
- protocol = d.protocol;
178
- await sessionsListGuard.acquire(AppiumDriver.name, () => {
179
- this.sessions[innerSessionId] = d;
180
- });
181
- } finally {
182
- await pendingDriversGuard.acquire(AppiumDriver.name, () => {
183
- _lodash.default.pull(this.pendingDrivers[InnerDriver.name], d);
184
- });
185
- }
186
-
187
- this.attachUnexpectedShutdownHandler(d, innerSessionId);
188
-
189
- _logger.default.info(`New ${InnerDriver.name} session created successfully, session ` + `${innerSessionId} added to master session list`);
190
-
191
- d.startNewCommandTimeout();
192
-
193
- if (d.isW3CProtocol() && !_lodash.default.isEmpty(w3cSettings)) {
194
- _logger.default.info(`Applying the initial values to Appium settings parsed from W3C caps: ` + JSON.stringify(w3cSettings));
195
-
196
- await d.updateSettings(w3cSettings);
197
- } else if (d.isMjsonwpProtocol() && !_lodash.default.isEmpty(jwpSettings)) {
198
- _logger.default.info(`Applying the initial values to Appium settings parsed from MJSONWP caps: ` + JSON.stringify(jwpSettings));
199
-
200
- await d.updateSettings(jwpSettings);
201
- }
202
- } catch (error) {
203
- return {
204
- protocol,
205
- error
206
- };
73
+ /**
74
+ * Cancel commands queueing for the umbrella Appium driver
75
+ */
76
+ get isCommandsQueueEnabled() {
77
+ return false;
78
+ }
79
+ sessionExists(sessionId) {
80
+ const dstSession = this.sessions[sessionId];
81
+ return dstSession && dstSession.sessionId !== null;
82
+ }
83
+ driverForSession(sessionId) {
84
+ return this.sessions[sessionId];
85
+ }
86
+ // eslint-disable-next-line require-await
87
+ async getStatus() {
88
+ return {
89
+ build: lodash_1.default.clone((0, config_1.getBuildInfo)()),
90
+ };
91
+ }
92
+ async getSessions() {
93
+ return lodash_1.default.toPairs(this.sessions).map(([id, driver]) => ({
94
+ id,
95
+ capabilities: /** @type {import('@appium/types').DriverCaps} */ (driver.caps),
96
+ }));
97
+ }
98
+ printNewSessionAnnouncement(driverName, driverVersion, driverBaseVersion) {
99
+ this.log.info(driverVersion
100
+ ? `Appium v${config_1.APPIUM_VER} creating new ${driverName} (v${driverVersion}) session`
101
+ : `Appium v${config_1.APPIUM_VER} creating new ${driverName} session`);
102
+ this.log.info(`Checking BaseDriver versions for Appium and ${driverName}`);
103
+ this.log.info(AppiumDriver.baseVersion
104
+ ? `Appium's BaseDriver version is ${AppiumDriver.baseVersion}`
105
+ : `Could not determine Appium's BaseDriver version`);
106
+ this.log.info(driverBaseVersion
107
+ ? `${driverName}'s BaseDriver version is ${driverBaseVersion}`
108
+ : `Could not determine ${driverName}'s BaseDriver version`);
109
+ }
110
+ /**
111
+ * Retrieves all CLI arguments for a specific plugin.
112
+ * @param {string} extName - Plugin name
113
+ * @returns {StringRecord} Arguments object. If none, an empty object.
114
+ */
115
+ getCliArgsForPlugin(extName) {
116
+ return /** @type {StringRecord} */ (this.args.plugin?.[extName] ?? {});
117
+ }
118
+ /**
119
+ * Retrieves CLI args for a specific driver.
120
+ *
121
+ * _Any arg which is equal to its default value will not be present in the returned object._
122
+ *
123
+ * _Note that this behavior currently (May 18 2022) differs from how plugins are handled_ (see {@linkcode AppiumDriver.getCliArgsForPlugin}).
124
+ * @param {string} extName - Driver name
125
+ * @returns {StringRecord|undefined} Arguments object. If none, `undefined`
126
+ */
127
+ getCliArgsForDriver(extName) {
128
+ const allCliArgsForExt = /** @type {StringRecord|undefined} */ (this.args.driver?.[extName]);
129
+ if (!lodash_1.default.isEmpty(allCliArgsForExt)) {
130
+ const defaults = (0, schema_1.getDefaultsForExtension)(constants_1.DRIVER_TYPE, extName);
131
+ const cliArgs = lodash_1.default.isEmpty(defaults)
132
+ ? allCliArgsForExt
133
+ : lodash_1.default.omitBy(allCliArgsForExt, (value, key) => lodash_1.default.isEqual(defaults[key], value));
134
+ if (!lodash_1.default.isEmpty(cliArgs)) {
135
+ return cliArgs;
136
+ }
137
+ }
138
+ }
139
+ /**
140
+ * Create a new session
141
+ * @param {W3CAppiumDriverCaps} jsonwpCaps JSONWP formatted desired capabilities
142
+ * @param {W3CAppiumDriverCaps} [reqCaps] Required capabilities (JSONWP standard)
143
+ * @param {W3CAppiumDriverCaps} [w3cCapabilities] W3C capabilities
144
+ * @returns {Promise<SessionHandlerCreateResult>}
145
+ */
146
+ async createSession(jsonwpCaps, reqCaps, w3cCapabilities) {
147
+ const defaultCapabilities = lodash_1.default.cloneDeep(this.args.defaultCapabilities);
148
+ const defaultSettings = (0, utils_1.pullSettings)(defaultCapabilities);
149
+ jsonwpCaps = lodash_1.default.cloneDeep(jsonwpCaps);
150
+ const jwpSettings = { ...defaultSettings, ...(0, utils_1.pullSettings)(jsonwpCaps) };
151
+ w3cCapabilities = lodash_1.default.cloneDeep(w3cCapabilities);
152
+ // It is possible that the client only provides caps using JSONWP standard,
153
+ // although firstMatch/alwaysMatch properties are still present.
154
+ // In such case we assume the client understands W3C protocol and merge the given
155
+ // JSONWP caps to W3C caps
156
+ const w3cSettings = {
157
+ ...jwpSettings,
158
+ ...(0, utils_1.pullSettings)((w3cCapabilities ?? {}).alwaysMatch ?? {}),
159
+ };
160
+ for (const firstMatchEntry of (w3cCapabilities ?? {}).firstMatch ?? []) {
161
+ Object.assign(w3cSettings, (0, utils_1.pullSettings)(firstMatchEntry));
162
+ }
163
+ /** @type {string|undefined} */
164
+ let protocol;
165
+ let innerSessionId, dCaps;
166
+ try {
167
+ // Parse the caps into a format that the InnerDriver will accept
168
+ const parsedCaps = (0, utils_1.parseCapsForInnerDriver)(jsonwpCaps, (0, base_driver_1.promoteAppiumOptions)(/** @type {W3CAppiumDriverCaps} */ (w3cCapabilities)), this.desiredCapConstraints, defaultCapabilities ? (0, base_driver_1.promoteAppiumOptionsForObject)(defaultCapabilities) : undefined);
169
+ const { desiredCaps, processedJsonwpCapabilities, processedW3CCapabilities } =
170
+ /** @type {import('./utils').ParsedDriverCaps<AppiumDriverConstraints>} */ (parsedCaps);
171
+ protocol = parsedCaps.protocol;
172
+ const error = /** @type {import('./utils').InvalidCaps<AppiumDriverConstraints>} */ (parsedCaps).error;
173
+ // If the parsing of the caps produced an error, throw it in here
174
+ if (error) {
175
+ throw error;
176
+ }
177
+ const { driver: InnerDriver, version: driverVersion, driverName, } = this.driverConfig.findMatchingDriver(desiredCaps);
178
+ this.printNewSessionAnnouncement(InnerDriver.name, driverVersion, InnerDriver.baseVersion);
179
+ if (this.args.sessionOverride) {
180
+ await this.deleteAllSessions();
181
+ }
182
+ /**
183
+ * @type {DriverData[]}
184
+ */
185
+ let runningDriversData = [];
186
+ /**
187
+ * @type {DriverData[]}
188
+ */
189
+ let otherPendingDriversData = [];
190
+ const driverInstance = /** @type {ExternalDriver} */ (new InnerDriver(this.args, true));
191
+ // We want to assign security values directly on the driver. The driver
192
+ // should not read security values from `this.opts` because those values
193
+ // could have been set by a malicious user via capabilities, whereas we
194
+ // want a guarantee the values were set by the appium server admin
195
+ if (this.args.relaxedSecurityEnabled) {
196
+ this.log.info(`Applying relaxed security to '${InnerDriver.name}' as per ` +
197
+ `server command line argument. All insecure features will be ` +
198
+ `enabled unless explicitly disabled by --deny-insecure`);
199
+ driverInstance.relaxedSecurityEnabled = true;
200
+ }
201
+ if (!lodash_1.default.isEmpty(this.args.denyInsecure)) {
202
+ this.log.info('Explicitly preventing use of insecure features:');
203
+ this.args.denyInsecure.map((a) => this.log.info(` ${a}`));
204
+ driverInstance.denyInsecure = this.args.denyInsecure;
205
+ }
206
+ if (!lodash_1.default.isEmpty(this.args.allowInsecure)) {
207
+ this.log.info('Explicitly enabling use of insecure features:');
208
+ this.args.allowInsecure.map((a) => this.log.info(` ${a}`));
209
+ driverInstance.allowInsecure = this.args.allowInsecure;
210
+ }
211
+ // Likewise, any driver-specific CLI args that were passed in should be assigned directly to
212
+ // the driver so that they cannot be mimicked by a malicious user sending in capabilities
213
+ const cliArgs = this.getCliArgsForDriver(driverName);
214
+ if (!lodash_1.default.isUndefined(cliArgs)) {
215
+ driverInstance.cliArgs = cliArgs;
216
+ }
217
+ // This assignment is required for correct web sockets functionality inside the driver
218
+ // Drivers/plugins might also want to know where they are hosted
219
+ // XXX: temporary hack to work around #16747
220
+ driverInstance.server = this.server;
221
+ driverInstance.serverHost = this.args.address;
222
+ driverInstance.serverPort = this.args.port;
223
+ driverInstance.serverPath = this.args.basePath;
224
+ try {
225
+ runningDriversData = (await this.curSessionDataForDriver(InnerDriver)) ?? [];
226
+ }
227
+ catch (e) {
228
+ throw new base_driver_1.errors.SessionNotCreatedError(e.message);
229
+ }
230
+ await pendingDriversGuard.acquire(AppiumDriver.name, () => {
231
+ this.pendingDrivers[InnerDriver.name] = this.pendingDrivers[InnerDriver.name] || [];
232
+ otherPendingDriversData = lodash_1.default.compact(this.pendingDrivers[InnerDriver.name].map((drv) => drv.driverData));
233
+ this.pendingDrivers[InnerDriver.name].push(driverInstance);
234
+ });
235
+ try {
236
+ [innerSessionId, dCaps] = await driverInstance.createSession(processedJsonwpCapabilities, reqCaps, processedW3CCapabilities, [...runningDriversData, ...otherPendingDriversData]);
237
+ protocol = driverInstance.protocol;
238
+ this.sessions[innerSessionId] = driverInstance;
239
+ }
240
+ finally {
241
+ await pendingDriversGuard.acquire(AppiumDriver.name, () => {
242
+ lodash_1.default.pull(this.pendingDrivers[InnerDriver.name], driverInstance);
243
+ });
244
+ }
245
+ this.attachUnexpectedShutdownHandler(driverInstance, innerSessionId);
246
+ this.log.info(`New ${InnerDriver.name} session created successfully, session ` +
247
+ `${innerSessionId} added to master session list`);
248
+ // set the New Command Timeout for the inner driver
249
+ driverInstance.startNewCommandTimeout();
250
+ // apply initial values to Appium settings (if provided)
251
+ if (driverInstance.isW3CProtocol() && !lodash_1.default.isEmpty(w3cSettings)) {
252
+ this.log.info(`Applying the initial values to Appium settings parsed from W3C caps: ` +
253
+ JSON.stringify(w3cSettings));
254
+ await driverInstance.updateSettings(w3cSettings);
255
+ }
256
+ else if (driverInstance.isMjsonwpProtocol() && !lodash_1.default.isEmpty(jwpSettings)) {
257
+ this.log.info(`Applying the initial values to Appium settings parsed from MJSONWP caps: ` +
258
+ JSON.stringify(jwpSettings));
259
+ await driverInstance.updateSettings(jwpSettings);
260
+ }
261
+ }
262
+ catch (error) {
263
+ return {
264
+ protocol,
265
+ error,
266
+ };
267
+ }
268
+ return {
269
+ protocol,
270
+ value: [innerSessionId, dCaps, protocol],
271
+ };
272
+ }
273
+ /**
274
+ *
275
+ * @param {ExternalDriver} driver
276
+ * @param {string} innerSessionId
277
+ */
278
+ attachUnexpectedShutdownHandler(driver, innerSessionId) {
279
+ const onShutdown = (cause = new Error('Unknown error')) => {
280
+ this.log.warn(`Ending session, cause was '${cause.message}'`);
281
+ if (this.sessionPlugins[innerSessionId]) {
282
+ for (const plugin of this.sessionPlugins[innerSessionId]) {
283
+ if (lodash_1.default.isFunction(plugin.onUnexpectedShutdown)) {
284
+ this.log.debug(`Plugin ${plugin.name} defines an unexpected shutdown handler; calling it now`);
285
+ try {
286
+ plugin.onUnexpectedShutdown(driver, cause);
287
+ }
288
+ catch (e) {
289
+ this.log.warn(`Got an error when running plugin ${plugin.name} shutdown handler: ${e}`);
290
+ }
291
+ }
292
+ else {
293
+ this.log.debug(`Plugin ${plugin.name} does not define an unexpected shutdown handler`);
294
+ }
295
+ }
296
+ }
297
+ this.log.info(`Removing session '${innerSessionId}' from our master session list`);
298
+ delete this.sessions[innerSessionId];
299
+ delete this.sessionPlugins[innerSessionId];
300
+ };
301
+ if (lodash_1.default.isFunction(driver.onUnexpectedShutdown)) {
302
+ driver.onUnexpectedShutdown(onShutdown);
303
+ }
304
+ else {
305
+ this.log.warn(`Failed to attach the unexpected shutdown listener. ` +
306
+ `Is 'onUnexpectedShutdown' method available for '${driver.constructor.name}'?`);
307
+ }
308
+ }
309
+ /**
310
+ *
311
+ * @param {((...args: any[]) => any)|(new(...args: any[]) => any)} InnerDriver
312
+ * @returns {Promise<DriverData[]>}}
313
+ * @privateRemarks The _intent_ is that `InnerDriver` is the class of a driver, but it only really
314
+ * needs to be a function or constructor.
315
+ */
316
+ async curSessionDataForDriver(InnerDriver) {
317
+ const data = lodash_1.default.compact(lodash_1.default.values(this.sessions)
318
+ .filter((s) => s.constructor.name === InnerDriver.name)
319
+ .map((s) => s.driverData));
320
+ for (const datum of data) {
321
+ if (!datum) {
322
+ throw new Error(`Problem getting session data for driver type ` +
323
+ `${InnerDriver.name}; does it implement 'get driverData'?`);
324
+ }
325
+ }
326
+ return data;
207
327
  }
208
-
209
- return {
210
- protocol,
211
- value: [innerSessionId, dCaps, protocol]
212
- };
213
- }
214
-
215
- attachUnexpectedShutdownHandler(driver, innerSessionId) {
216
- const removeSessionFromMasterList = (cause = new Error('Unknown error')) => {
217
- _logger.default.warn(`Closing session, cause was '${cause.message}'`);
218
-
219
- _logger.default.info(`Removing session '${innerSessionId}' from our master session list`);
220
-
221
- delete this.sessions[innerSessionId];
222
- };
223
-
224
- if (_lodash.default.isFunction((driver.onUnexpectedShutdown || {}).then)) {
225
- driver.onUnexpectedShutdown.then(() => {
226
- throw new Error('Unexpected shutdown');
227
- }).catch(e => {
228
- if (!(e instanceof _bluebird.default.CancellationError)) {
229
- removeSessionFromMasterList(e);
328
+ /**
329
+ * @param {string} sessionId
330
+ */
331
+ async deleteSession(sessionId) {
332
+ let protocol;
333
+ try {
334
+ let otherSessionsData;
335
+ const dstSession = await sessionsListGuard.acquire(AppiumDriver.name, () => {
336
+ if (!this.sessions[sessionId]) {
337
+ return;
338
+ }
339
+ const curConstructorName = this.sessions[sessionId].constructor.name;
340
+ otherSessionsData = lodash_1.default.toPairs(this.sessions)
341
+ .filter(([key, value]) => value.constructor.name === curConstructorName && key !== sessionId)
342
+ .map(([, value]) => value.driverData);
343
+ const dstSession = this.sessions[sessionId];
344
+ protocol = dstSession.protocol;
345
+ this.log.info(`Removing session ${sessionId} from our master session list`);
346
+ // regardless of whether the deleteSession completes successfully or not
347
+ // make the session unavailable, because who knows what state it might
348
+ // be in otherwise
349
+ delete this.sessions[sessionId];
350
+ delete this.sessionPlugins[sessionId];
351
+ return dstSession;
352
+ });
353
+ // this may not be correct, but if `dstSession` was falsy, the call to `deleteSession()` would
354
+ // throw anyway.
355
+ if (!dstSession) {
356
+ throw new Error('Session not found');
357
+ }
358
+ return {
359
+ protocol,
360
+ value: await dstSession.deleteSession(sessionId, otherSessionsData),
361
+ };
362
+ }
363
+ catch (e) {
364
+ this.log.error(`Had trouble ending session ${sessionId}: ${e.message}`);
365
+ return {
366
+ protocol,
367
+ error: e,
368
+ };
230
369
  }
231
- });
232
- } else if (_lodash.default.isFunction(driver.onUnexpectedShutdown)) {
233
- driver.onUnexpectedShutdown(removeSessionFromMasterList);
234
- } else {
235
- _logger.default.warn(`Failed to attach the unexpected shutdown listener. ` + `Is 'onUnexpectedShutdown' method available for '${driver.constructor.name}'?`);
236
370
  }
237
- }
238
-
239
- async curSessionDataForDriver(InnerDriver) {
240
- const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);
241
-
242
- const data = _lodash.default.values(sessions).filter(s => s.constructor.name === InnerDriver.name).map(s => s.driverData);
243
-
244
- for (let datum of data) {
245
- if (!datum) {
246
- throw new Error(`Problem getting session data for driver type ` + `${InnerDriver.name}; does it implement 'get ` + `driverData'?`);
247
- }
371
+ async deleteAllSessions(opts = {}) {
372
+ const sessionsCount = lodash_1.default.size(this.sessions);
373
+ if (0 === sessionsCount) {
374
+ this.log.debug('There are no active sessions for cleanup');
375
+ return;
376
+ }
377
+ const { force = false, reason } = opts;
378
+ this.log.debug(`Cleaning up ${support_1.util.pluralize('active session', sessionsCount, true)}`);
379
+ const cleanupPromises = force
380
+ ? lodash_1.default.values(this.sessions).map((drv) => drv.startUnexpectedShutdown(reason && new Error(reason)))
381
+ : lodash_1.default.keys(this.sessions).map((id) => this.deleteSession(id));
382
+ for (const cleanupPromise of cleanupPromises) {
383
+ try {
384
+ await cleanupPromise;
385
+ }
386
+ catch (e) {
387
+ this.log.debug(e);
388
+ }
389
+ }
248
390
  }
249
-
250
- return data;
251
- }
252
-
253
- async deleteSession(sessionId) {
254
- let protocol;
255
-
256
- try {
257
- let otherSessionsData = null;
258
- let dstSession = null;
259
- await sessionsListGuard.acquire(AppiumDriver.name, () => {
260
- if (!this.sessions[sessionId]) {
261
- return;
391
+ /**
392
+ * Get the appropriate plugins for a session (or sessionless plugins)
393
+ *
394
+ * @param {?string} sessionId - the sessionId (or null) to use to find plugins
395
+ * @returns {Array} - array of plugin instances
396
+ */
397
+ pluginsForSession(sessionId = null) {
398
+ if (sessionId) {
399
+ if (!this.sessionPlugins[sessionId]) {
400
+ this.sessionPlugins[sessionId] = this.createPluginInstances();
401
+ }
402
+ return this.sessionPlugins[sessionId];
403
+ }
404
+ if (lodash_1.default.isEmpty(this.sessionlessPlugins)) {
405
+ this.sessionlessPlugins = this.createPluginInstances();
262
406
  }
263
-
264
- const curConstructorName = this.sessions[sessionId].constructor.name;
265
- otherSessionsData = _lodash.default.toPairs(this.sessions).filter(([key, value]) => value.constructor.name === curConstructorName && key !== sessionId).map(([, value]) => value.driverData);
266
- dstSession = this.sessions[sessionId];
267
- protocol = dstSession.protocol;
268
-
269
- _logger.default.info(`Removing session ${sessionId} from our master session list`);
270
-
271
- delete this.sessions[sessionId];
272
- });
273
- return {
274
- protocol,
275
- value: await dstSession.deleteSession(sessionId, otherSessionsData)
276
- };
277
- } catch (e) {
278
- _logger.default.error(`Had trouble ending session ${sessionId}: ${e.message}`);
279
-
280
- return {
281
- protocol,
282
- error: e
283
- };
407
+ return this.sessionlessPlugins;
408
+ }
409
+ /**
410
+ * To get plugins for a command, we either get the plugin instances associated with the
411
+ * particular command's session, or in the case of sessionless plugins, pull from the set of
412
+ * plugin instances reserved for sessionless commands (and we lazily create plugin instances on
413
+ * first use)
414
+ *
415
+ * @param {string} cmd - the name of the command to find a plugin to handle
416
+ * @param {?string} sessionId - the particular session for which to find a plugin, or null if
417
+ * sessionless
418
+ */
419
+ pluginsToHandleCmd(cmd, sessionId = null) {
420
+ // to handle a given command, a plugin should either implement that command as a plugin
421
+ // instance method or it should implement a generic 'handle' method
422
+ return this.pluginsForSession(sessionId).filter((p) => lodash_1.default.isFunction(p[cmd]) || lodash_1.default.isFunction(p.handle));
284
423
  }
285
- }
286
-
287
- async deleteAllSessions(opts = {}) {
288
- const sessionsCount = _lodash.default.size(this.sessions);
289
-
290
- if (0 === sessionsCount) {
291
- _logger.default.debug('There are no active sessions for cleanup');
292
-
293
- return;
424
+ /**
425
+ * Creates instances of all of the enabled Plugin classes
426
+ * @returns {Plugin[]}
427
+ */
428
+ createPluginInstances() {
429
+ /** @type {Plugin[]} */
430
+ const pluginInstances = [];
431
+ for (const [PluginClass, name] of this.pluginClasses.entries()) {
432
+ const cliArgs = this.getCliArgsForPlugin(name);
433
+ const plugin = new PluginClass(name, cliArgs);
434
+ pluginInstances.push(plugin);
435
+ }
436
+ return pluginInstances;
437
+ }
438
+ /**
439
+ *
440
+ * @param {string} cmd
441
+ * @param {...any} args
442
+ * @returns {Promise<{value: any, error?: Error, protocol: string} | import('type-fest').AsyncReturnType<ExternalDriver['executeCommand']>>}
443
+ */
444
+ async executeCommand(cmd, ...args) {
445
+ // We have basically three cases for how to handle commands:
446
+ // 1. handle getStatus (we do this as a special out of band case so it doesn't get added to an
447
+ // execution queue, and can be called while e.g. createSession is in progress)
448
+ // 2. handle commands that this umbrella driver should handle, rather than the actual session
449
+ // driver (for example, deleteSession, or other non-session commands)
450
+ // 3. handle session driver commands.
451
+ // The tricky part is that because we support command plugins, we need to wrap any of these
452
+ // cases with plugin handling.
453
+ const isGetStatus = cmd === base_driver_1.GET_STATUS_COMMAND;
454
+ const isUmbrellaCmd = isAppiumDriverCommand(cmd);
455
+ const isSessionCmd = (0, base_driver_1.isSessionCommand)(cmd);
456
+ // if a plugin override proxying for this command and that is why we are here instead of just
457
+ // letting the protocol proxy the command entirely, determine that, get the request object for
458
+ // use later on, then clean up the args
459
+ const reqForProxy = lodash_1.default.last(args)?.reqForProxy;
460
+ if (reqForProxy) {
461
+ args.pop();
462
+ }
463
+ // first do some error checking. If we're requesting a session command execution, then make
464
+ // sure that session actually exists on the session driver, and set the session driver itself
465
+ let sessionId = null;
466
+ let dstSession = null;
467
+ let protocol = null;
468
+ /** @type {this | ExternalDriver} */
469
+ // eslint-disable-next-line @typescript-eslint/no-this-alias
470
+ let driver = this;
471
+ if (isSessionCmd) {
472
+ sessionId = lodash_1.default.last(args);
473
+ dstSession = this.sessions[sessionId];
474
+ if (!dstSession) {
475
+ throw new Error(`The session with id '${sessionId}' does not exist`);
476
+ }
477
+ // now save the response protocol given that the session driver's protocol might differ
478
+ protocol = dstSession.protocol;
479
+ if (!isUmbrellaCmd) {
480
+ driver = dstSession;
481
+ }
482
+ }
483
+ // get any plugins which are registered as handling this command
484
+ const plugins = this.pluginsToHandleCmd(cmd, sessionId);
485
+ // now we define a 'cmdHandledBy' object which will keep track of which plugins have handled this
486
+ // command. we care about this because (a) multiple plugins can handle the same command, and
487
+ // (b) there's no guarantee that a plugin will actually call the next() method which runs the
488
+ // original command execution. This results in a situation where the command might be handled
489
+ // by some but not all plugins, or by plugin(s) but not by the default behavior. So start out
490
+ // this object declaring that the default handler has not been executed.
491
+ const cmdHandledBy = { default: false };
492
+ // now we define an async function which will be passed to plugins, and successively wrapped
493
+ // if there is more than one plugin that can handle the command. To start off with, the async
494
+ // function is defined as calling the default behavior, i.e., whichever of the 3 cases above is
495
+ // the appropriate one
496
+ const defaultBehavior = async () => {
497
+ // if we're running with plugins, make sure we log that the default behavior is actually
498
+ // happening so we can tell when the plugin call chain is unwrapping to the default behavior
499
+ // if that's what happens
500
+ plugins.length && this.log.info(`Executing default handling behavior for command '${cmd}'`);
501
+ // if we make it here, we know that the default behavior is handled
502
+ cmdHandledBy.default = true;
503
+ if (reqForProxy) {
504
+ // we would have proxied this command had a plugin not handled it, so the default behavior
505
+ // is to do the proxy and retrieve the result internally so it can be passed to the plugin
506
+ // in case it calls 'await next()'. This requires that the driver have defined
507
+ // 'proxyCommand' and not just 'proxyReqRes'.
508
+ if (!dstSession.proxyCommand) {
509
+ throw new NoDriverProxyCommandError();
510
+ }
511
+ return await dstSession.proxyCommand(reqForProxy.originalUrl, reqForProxy.method, reqForProxy.body);
512
+ }
513
+ if (isGetStatus) {
514
+ return await this.getStatus();
515
+ }
516
+ if (isUmbrellaCmd) {
517
+ // some commands, like deleteSession, we want to make sure to handle on *this* driver,
518
+ // not the platform driver
519
+ return await base_driver_1.BaseDriver.prototype.executeCommand.call(this, cmd, ...args);
520
+ }
521
+ // here we know that we are executing a session command, and have a valid session driver
522
+ return await dstSession.executeCommand(cmd, ...args);
523
+ };
524
+ // now take our default behavior, wrap it with any number of plugin behaviors, and run it
525
+ const wrappedCmd = this.wrapCommandWithPlugins({
526
+ driver,
527
+ cmd,
528
+ args,
529
+ plugins,
530
+ cmdHandledBy,
531
+ next: defaultBehavior,
532
+ });
533
+ const res = await this.executeWrappedCommand({ wrappedCmd, protocol });
534
+ // if we had plugins, make sure to log out the helpful report about which plugins ended up
535
+ // handling the command and which didn't
536
+ this.logPluginHandlerReport(plugins, { cmd, cmdHandledBy });
537
+ // And finally, if the command was createSession, we want to migrate any plugins which were
538
+ // previously sessionless to use the new sessionId, so that plugins can share state between
539
+ // their createSession method and other instance methods
540
+ if (cmd === base_driver_1.CREATE_SESSION_COMMAND && this.sessionlessPlugins.length && !res.error) {
541
+ const sessionId = lodash_1.default.first(res.value);
542
+ this.log.info(`Promoting ${this.sessionlessPlugins.length} sessionless plugins to be attached ` +
543
+ `to session ID ${sessionId}`);
544
+ this.sessionPlugins[sessionId] = this.sessionlessPlugins;
545
+ this.sessionlessPlugins = [];
546
+ }
547
+ return res;
294
548
  }
295
-
296
- const {
297
- force = false,
298
- reason
299
- } = opts;
300
-
301
- _logger.default.debug(`Cleaning up ${_appiumSupport.util.pluralize('active session', sessionsCount, true)}`);
302
-
303
- 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));
304
-
305
- for (const cleanupPromise of cleanupPromises) {
306
- try {
307
- await cleanupPromise;
308
- } catch (e) {
309
- _logger.default.debug(e);
310
- }
549
+ wrapCommandWithPlugins({ driver, cmd, args, next, cmdHandledBy, plugins }) {
550
+ plugins.length &&
551
+ this.log.info(`Plugins which can handle cmd '${cmd}': ${plugins.map((p) => p.name)}`);
552
+ // now we can go through each plugin and wrap `next` around its own handler, passing the *old*
553
+ // next in so that it can call it if it wants to
554
+ for (const plugin of plugins) {
555
+ // need an IIFE here because we want the value of next that's passed to plugin.handle to be
556
+ // exactly the value of next here before reassignment; we don't want it to be lazily
557
+ // evaluated, otherwise we end up with infinite recursion of the last `next` to be defined.
558
+ cmdHandledBy[plugin.name] = false; // we see a new plugin, so add it to the 'cmdHandledBy' object
559
+ next = ((_next) => async () => {
560
+ this.log.info(`Plugin ${plugin.name} is now handling cmd '${cmd}'`);
561
+ cmdHandledBy[plugin.name] = true; // if we make it here, this plugin has attempted to handle cmd
562
+ // first attempt to handle the command via a command-specific handler on the plugin
563
+ if (plugin[cmd]) {
564
+ return await plugin[cmd](_next, driver, ...args);
565
+ }
566
+ // otherwise, call the generic 'handle' method
567
+ return await plugin.handle(_next, driver, cmd, ...args);
568
+ })(next);
569
+ }
570
+ return next;
311
571
  }
312
- }
313
-
314
- pluginsToHandleCmd(cmd) {
315
- return this.plugins.filter(p => p.commands === true || _lodash.default.isArray(p.commands) && _lodash.default.includes(p.commands, cmd));
316
- }
317
-
318
- async executeCommand(cmd, ...args) {
319
- const isGetStatus = cmd === 'getStatus';
320
- const isUmbrellaCmd = !isGetStatus && isAppiumDriverCommand(cmd);
321
- const isSessionCmd = !isGetStatus && !isUmbrellaCmd;
322
- const plugins = this.pluginsToHandleCmd(cmd);
323
- let sessionId = null;
324
- let dstSession = null;
325
- let protocol = null;
326
-
327
- if (isSessionCmd) {
328
- sessionId = _lodash.default.last(args);
329
- dstSession = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions[sessionId]);
330
-
331
- if (!dstSession) {
332
- throw new Error(`The session with id '${sessionId}' does not exist`);
333
- }
334
-
335
- protocol = dstSession.protocol;
572
+ logPluginHandlerReport(plugins, { cmd, cmdHandledBy }) {
573
+ if (!plugins.length) {
574
+ return;
575
+ }
576
+ // at the end of the day, we have an object representing which plugins ended up getting
577
+ // their code run as part of handling this command. Because plugins can choose *not* to
578
+ // pass control to other plugins or to the default driver behavior, this is information
579
+ // which is probably useful to the user (especially in situations where plugins might not
580
+ // interact well together, and it would be hard to debug otherwise without this kind of
581
+ // message).
582
+ const didHandle = Object.keys(cmdHandledBy).filter((k) => cmdHandledBy[k]);
583
+ const didntHandle = Object.keys(cmdHandledBy).filter((k) => !cmdHandledBy[k]);
584
+ if (didntHandle.length > 0) {
585
+ this.log.info(`Command '${cmd}' was *not* handled by the following behaviours or plugins, even ` +
586
+ `though they were registered to handle it: ${JSON.stringify(didntHandle)}. The ` +
587
+ `command *was* handled by these: ${JSON.stringify(didHandle)}.`);
588
+ }
336
589
  }
337
-
338
- const cmdHandledBy = {
339
- default: false
340
- };
341
-
342
- const defaultBehavior = async () => {
343
- plugins.length && _logger.default.info(`Executing default handling behavior for command '${cmd}'`);
344
- cmdHandledBy.default = true;
345
-
346
- if (isGetStatus) {
347
- return await this.getStatus();
348
- }
349
-
350
- if (isUmbrellaCmd) {
351
- return await super.executeCommand(cmd, ...args);
352
- }
353
-
354
- return await dstSession.executeCommand(cmd, ...args);
355
- };
356
-
357
- const wrappedCmd = this.wrapCommandWithPlugins({
358
- cmd,
359
- args,
360
- plugins,
361
- cmdHandledBy,
362
- next: defaultBehavior
363
- });
364
- const res = await this.executeWrappedCommand({
365
- wrappedCmd,
366
- protocol
367
- });
368
- plugins.length && this.logPluginHandlerReport({
369
- cmd,
370
- cmdHandledBy
371
- });
372
- return res;
373
- }
374
-
375
- wrapCommandWithPlugins({
376
- cmd,
377
- args,
378
- next,
379
- cmdHandledBy,
380
- plugins
381
- }) {
382
- plugins.length && _logger.default.info(`Plugins which can handle cmd '${cmd}': ${plugins.map(p => p.name)}`);
383
-
384
- for (const plugin of plugins) {
385
- cmdHandledBy[plugin.name] = false;
386
-
387
- next = (_next => async () => {
388
- _logger.default.info(`Plugin ${plugin.name} is now handling cmd '${cmd}'`);
389
-
390
- cmdHandledBy[plugin.name] = true;
391
- return await plugin.handle(_next, this, cmd, ...args);
392
- })(next);
590
+ async executeWrappedCommand({ wrappedCmd, protocol }) {
591
+ let cmdRes, cmdErr, res = {};
592
+ try {
593
+ // At this point, `wrappedCmd` defines a whole sequence of plugin handlers, culminating in
594
+ // our default handler. Whatever it returns is what we're going to want to send back to the
595
+ // user.
596
+ cmdRes = await wrappedCmd();
597
+ }
598
+ catch (e) {
599
+ cmdErr = e;
600
+ }
601
+ // Sadly, we don't know exactly what kind of object will be returned. It will either be a bare
602
+ // object, or a protocol-aware object with protocol and error/value keys. So we need to sniff
603
+ // it and make sure we don't double-wrap it if it's the latter kind.
604
+ if (lodash_1.default.isPlainObject(cmdRes) && lodash_1.default.has(cmdRes, 'protocol')) {
605
+ res = cmdRes;
606
+ }
607
+ else {
608
+ res.value = cmdRes;
609
+ res.error = cmdErr;
610
+ res.protocol = protocol;
611
+ }
612
+ return res;
393
613
  }
394
-
395
- return next;
396
- }
397
-
398
- logPluginHandlerReport({
399
- cmd,
400
- cmdHandledBy
401
- }) {
402
- const didHandle = Object.keys(cmdHandledBy).filter(k => cmdHandledBy[k]);
403
- const didntHandle = Object.keys(cmdHandledBy).filter(k => !cmdHandledBy[k]);
404
-
405
- if (didntHandle.length > 0) {
406
- _logger.default.info(`Command '${cmd}' was not handled by the following beahviors or plugins, even ` + `though they were registered to handle it: ${JSON.stringify(didntHandle)}. The ` + `command *was* handled by these: ${JSON.stringify(didHandle)}.`);
614
+ proxyActive(sessionId) {
615
+ const dstSession = this.sessions[sessionId];
616
+ return dstSession && lodash_1.default.isFunction(dstSession.proxyActive) && dstSession.proxyActive(sessionId);
407
617
  }
408
- }
409
-
410
- async executeWrappedCommand({
411
- wrappedCmd,
412
- protocol
413
- }) {
414
- let cmdRes,
415
- cmdErr,
416
- res = {};
417
-
418
- try {
419
- cmdRes = await wrappedCmd();
420
- } catch (e) {
421
- cmdErr = e;
618
+ /**
619
+ *
620
+ * @param {string} sessionId
621
+ * @returns {import('@appium/types').RouteMatcher[]}
622
+ */
623
+ getProxyAvoidList(sessionId) {
624
+ const dstSession = this.sessions[sessionId];
625
+ return dstSession ? dstSession.getProxyAvoidList() : [];
422
626
  }
423
-
424
- if (_lodash.default.isPlainObject(cmdRes) && _lodash.default.has(cmdRes, 'protocol')) {
425
- res = cmdRes;
426
- } else {
427
- res.value = cmdRes;
428
- res.error = cmdErr;
429
- res.protocol = protocol;
627
+ canProxy(sessionId) {
628
+ const dstSession = this.sessions[sessionId];
629
+ return dstSession && dstSession.canProxy(sessionId);
430
630
  }
431
-
432
- return res;
433
- }
434
-
435
- proxyActive(sessionId) {
436
- const dstSession = this.sessions[sessionId];
437
- return dstSession && _lodash.default.isFunction(dstSession.proxyActive) && dstSession.proxyActive(sessionId);
438
- }
439
-
440
- getProxyAvoidList(sessionId) {
441
- const dstSession = this.sessions[sessionId];
442
- return dstSession ? dstSession.getProxyAvoidList() : [];
443
- }
444
-
445
- canProxy(sessionId) {
446
- const dstSession = this.sessions[sessionId];
447
- return dstSession && dstSession.canProxy(sessionId);
448
- }
449
-
450
631
  }
451
-
452
632
  exports.AppiumDriver = AppiumDriver;
453
-
633
+ // help decide which commands should be proxied to sub-drivers and which
634
+ // should be handled by this, our umbrella driver
454
635
  function isAppiumDriverCommand(cmd) {
455
- return !(0, _appiumBaseDriver.isSessionCommand)(cmd) || cmd === 'deleteSession';
456
- }require('source-map-support').install();
457
-
458
-
459
- //# 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","Object","assign","sessions","pendingDrivers","plugins","isCommandsQueueEnabled","sessionExists","sessionId","dstSession","driverForSession","getStatus","build","_","clone","getSessions","acquire","name","toPairs","map","id","driver","capabilities","caps","printNewSessionAnnouncement","driverName","driverVersion","introString","APPIUM_VER","log","info","_findMatchingDriver","createSession","jsonwpCaps","reqCaps","w3cCapabilities","defaultCapabilities","cloneDeep","defaultSettings","jwpSettings","w3cSettings","alwaysMatch","firstMatchEntry","firstMatch","protocol","innerSessionId","dCaps","parsedCaps","desiredCaps","processedJsonwpCapabilities","processedW3CCapabilities","error","InnerDriver","version","driverConfig","sessionOverride","deleteAllSessions","runningDriversData","otherPendingDriversData","d","relaxedSecurityEnabled","isEmpty","denyInsecure","a","allowInsecure","server","curSessionDataForDriver","e","errors","SessionNotCreatedError","message","drv","driverData","push","pull","attachUnexpectedShutdownHandler","startNewCommandTimeout","isW3CProtocol","JSON","stringify","updateSettings","isMjsonwpProtocol","value","removeSessionFromMasterList","cause","Error","warn","isFunction","onUnexpectedShutdown","then","catch","B","CancellationError","data","values","filter","s","datum","deleteSession","otherSessionsData","curConstructorName","key","opts","sessionsCount","size","debug","force","reason","util","pluralize","cleanupPromises","startUnexpectedShutdown","keys","cleanupPromise","pluginsToHandleCmd","cmd","p","commands","isArray","includes","executeCommand","isGetStatus","isUmbrellaCmd","isAppiumDriverCommand","isSessionCmd","last","cmdHandledBy","default","defaultBehavior","length","wrappedCmd","wrapCommandWithPlugins","next","res","executeWrappedCommand","logPluginHandlerReport","plugin","_next","handle","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 } from 'appium-base-driver';\nimport B from 'bluebird';\nimport AsyncLock from 'async-lock';\nimport { parseCapsForInnerDriver, pullSettings } from './utils';\nimport { util } from 'appium-support';\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 = Object.assign({}, 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    this.plugins = [];\n\n    // allow this to happen in the background, so no `await`\n    updateBuildInfo();\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) {\n    const introString = driverVersion\n      ? `Appium v${APPIUM_VER} creating new ${driverName} (v${driverVersion}) session`\n      : `Appium v${APPIUM_VER} creating new ${driverName} session`;\n    log.info(introString);\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   * 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      } = this._findMatchingDriver(this.driverConfig, desiredCaps);\n      this.printNewSessionAnnouncement(InnerDriver.name, driverVersion);\n\n      if (this.args.sessionOverride) {\n        await this.deleteAllSessions();\n      }\n\n      let runningDriversData, otherPendingDriversData;\n      const d = new InnerDriver(this.args);\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        d.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        d.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        d.allowInsecure = this.args.allowInsecure;\n      }\n\n      // This assignment is required for correct web sockets functionality inside the driver\n      d.server = this.server;\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(d);\n      });\n\n      try {\n        [innerSessionId, dCaps] = await d.createSession(\n          processedJsonwpCapabilities,\n          reqCaps,\n          processedW3CCapabilities,\n          [...runningDriversData, ...otherPendingDriversData]\n        );\n        protocol = d.protocol;\n        await sessionsListGuard.acquire(AppiumDriver.name, () => {\n          this.sessions[innerSessionId] = d;\n        });\n      } finally {\n        await pendingDriversGuard.acquire(AppiumDriver.name, () => {\n          _.pull(this.pendingDrivers[InnerDriver.name], d);\n        });\n      }\n\n      this.attachUnexpectedShutdownHandler(d, 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      d.startNewCommandTimeout();\n\n      // apply initial values to Appium settings (if provided)\n      if (d.isW3CProtocol() && !_.isEmpty(w3cSettings)) {\n        log.info(`Applying the initial values to Appium settings parsed from W3C caps: ` +\n          JSON.stringify(w3cSettings));\n        await d.updateSettings(w3cSettings);\n      } else if (d.isMjsonwpProtocol() && !_.isEmpty(jwpSettings)) {\n        log.info(`Applying the initial values to Appium settings parsed from MJSONWP caps: ` +\n          JSON.stringify(jwpSettings));\n        await d.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 removeSessionFromMasterList = (cause = new Error('Unknown error')) => {\n      log.warn(`Closing session, cause was '${cause.message}'`);\n      log.info(`Removing session '${innerSessionId}' from our master session list`);\n      delete this.sessions[innerSessionId];\n    };\n\n    // eslint-disable-next-line promise/prefer-await-to-then\n    if (_.isFunction((driver.onUnexpectedShutdown || {}).then)) {\n      // TODO: Remove this block after all the drivers use base driver above v 5.0.0\n      // Remove the session on unexpected shutdown, so that we are in a position\n      // to open another session later on.\n      driver.onUnexpectedShutdown\n        // eslint-disable-next-line promise/prefer-await-to-then\n        .then(() => {\n          // if we get here, we've had an unexpected shutdown, so error\n          throw new Error('Unexpected shutdown');\n        })\n        .catch((e) => {\n          // if we cancelled the unexpected shutdown promise, that means we\n          // no longer care about it, and can safely ignore it\n          if (!(e instanceof B.CancellationError)) {\n            removeSessionFromMasterList(e);\n          }\n        }); // this is a cancellable promise\n    } else if (_.isFunction(driver.onUnexpectedShutdown)) {\n      // since base driver v 5.0.0\n      driver.onUnexpectedShutdown(removeSessionFromMasterList);\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      });\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  pluginsToHandleCmd (cmd) {\n    return this.plugins.filter((p) =>\n      p.commands === true ||\n      (_.isArray(p.commands) && _.includes(p.commands, cmd))\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 === 'getStatus';\n    const isUmbrellaCmd = !isGetStatus && isAppiumDriverCommand(cmd);\n    const isSessionCmd = !isGetStatus && !isUmbrellaCmd;\n\n    // get any plugins which are registered as handling this command\n    const plugins = this.pluginsToHandleCmd(cmd);\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    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    }\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 (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({cmd, args, plugins, cmdHandledBy, next: defaultBehavior});\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    plugins.length && this.logPluginHandlerReport({cmd, cmdHandledBy});\n\n    return res;\n  }\n\n  wrapCommandWithPlugins ({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        return await plugin.handle(_next, this, cmd, ...args);\n      })(next);\n    }\n\n    return next;\n  }\n\n  logPluginHandlerReport ({cmd, cmdHandledBy}) {\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 beahviors 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\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\nexport { AppiumDriver };\n"],"file":"lib/appium.js","sourceRoot":"../.."}
636
+ return !(0, base_driver_1.isSessionCommand)(cmd) || cmd === base_driver_1.DELETE_SESSION_COMMAND;
637
+ }
638
+ /**
639
+ * Thrown when Appium tried to proxy a command using a driver's `proxyCommand` method but the
640
+ * method did not exist
641
+ */
642
+ class NoDriverProxyCommandError extends Error {
643
+ constructor() {
644
+ super(`The default behavior for this command was to proxy, but the driver ` +
645
+ `did not have the 'proxyCommand' method defined. To fully support ` +
646
+ `plugins, drivers should have 'proxyCommand' set to a jwpProxy object's ` +
647
+ `'command()' method, in addition to the normal 'proxyReqRes'`);
648
+ /**
649
+ * @type {Readonly<string>}
650
+ */
651
+ this.code = 'APPIUMERR_NO_DRIVER_PROXYCOMMAND';
652
+ }
653
+ }
654
+ exports.NoDriverProxyCommandError = NoDriverProxyCommandError;
655
+ /**
656
+ * @typedef {import('@appium/types').DriverData} DriverData
657
+ * @typedef {import('@appium/types').ServerArgs} DriverOpts
658
+ * @typedef {import('@appium/types').Constraints} Constraints
659
+ * @typedef {import('@appium/types').AppiumServer} AppiumServer
660
+ * @typedef {import('@appium/types').ExtensionType} ExtensionType
661
+ * @typedef {import('./extension/driver-config').DriverConfig} DriverConfig
662
+ * @typedef {import('@appium/types').PluginType} PluginType
663
+ * @typedef {import('@appium/types').DriverType} DriverType
664
+ * @typedef {import('@appium/types').StringRecord} StringRecord
665
+ * @typedef {import('@appium/types').ExternalDriver} ExternalDriver
666
+ * @typedef {import('@appium/types').PluginClass} PluginClass
667
+ * @typedef {import('@appium/types').Plugin} Plugin
668
+ * @typedef {import('@appium/types').DriverClass<import('@appium/types').Driver>} DriverClass
669
+ */
670
+ /**
671
+ * @typedef {import('@appium/types').ISessionHandler<AppiumDriverConstraints,
672
+ * SessionHandlerCreateResult, SessionHandlerDeleteResult>} AppiumSessionHandler
673
+ */
674
+ /**
675
+ * @typedef {SessionHandlerResult<[innerSessionId: string, caps:
676
+ * import('@appium/types').DriverCaps<Constraints>, protocol: string|undefined]>} SessionHandlerCreateResult
677
+ */
678
+ /**
679
+ * @template {Constraints} C
680
+ * @typedef {import('@appium/types').Core<C>} Core
681
+ */
682
+ /**
683
+ * @typedef {SessionHandlerResult<void>} SessionHandlerDeleteResult
684
+ */
685
+ /**
686
+ * Used by {@linkcode AppiumDriver.createSession} and {@linkcode AppiumDriver.deleteSession} to describe
687
+ * result.
688
+ * @template V
689
+ * @typedef SessionHandlerResult
690
+ * @property {V} [value]
691
+ * @property {Error} [error]
692
+ * @property {string} [protocol]
693
+ */
694
+ /**
695
+ * @typedef {typeof desiredCapabilityConstraints} AppiumDriverConstraints
696
+ * @typedef {import('@appium/types').W3CDriverCaps<AppiumDriverConstraints>} W3CAppiumDriverCaps
697
+ */
698
+ //# sourceMappingURL=appium.js.map