appium 2.0.0-beta.4 → 2.0.0-beta.40

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 (151) hide show
  1. package/README.md +10 -11
  2. package/build/lib/appium.d.ts +204 -0
  3. package/build/lib/appium.d.ts.map +1 -0
  4. package/build/lib/appium.js +257 -131
  5. package/build/lib/cli/args.d.ts +20 -0
  6. package/build/lib/cli/args.d.ts.map +1 -0
  7. package/build/lib/cli/args.js +96 -282
  8. package/build/lib/cli/driver-command.d.ts +36 -0
  9. package/build/lib/cli/driver-command.d.ts.map +1 -0
  10. package/build/lib/cli/driver-command.js +25 -18
  11. package/build/lib/cli/extension-command.d.ts +372 -0
  12. package/build/lib/cli/extension-command.d.ts.map +1 -0
  13. package/build/lib/cli/extension-command.js +286 -156
  14. package/build/lib/cli/extension.d.ts +18 -0
  15. package/build/lib/cli/extension.d.ts.map +1 -0
  16. package/build/lib/cli/extension.js +30 -17
  17. package/build/lib/cli/parser.d.ts +80 -0
  18. package/build/lib/cli/parser.d.ts.map +1 -0
  19. package/build/lib/cli/parser.js +152 -95
  20. package/build/lib/cli/plugin-command.d.ts +33 -0
  21. package/build/lib/cli/plugin-command.d.ts.map +1 -0
  22. package/build/lib/cli/plugin-command.js +24 -19
  23. package/build/lib/cli/utils.d.ts +29 -0
  24. package/build/lib/cli/utils.d.ts.map +1 -0
  25. package/build/lib/cli/utils.js +27 -3
  26. package/build/lib/config-file.d.ts +100 -0
  27. package/build/lib/config-file.d.ts.map +1 -0
  28. package/build/lib/config-file.js +136 -0
  29. package/build/lib/config.d.ts +41 -0
  30. package/build/lib/config.d.ts.map +1 -0
  31. package/build/lib/config.js +92 -67
  32. package/build/lib/constants.d.ts +48 -0
  33. package/build/lib/constants.d.ts.map +1 -0
  34. package/build/lib/constants.js +60 -0
  35. package/build/lib/extension/driver-config.d.ts +81 -0
  36. package/build/lib/extension/driver-config.d.ts.map +1 -0
  37. package/build/lib/extension/driver-config.js +177 -0
  38. package/build/lib/extension/extension-config.d.ts +242 -0
  39. package/build/lib/extension/extension-config.d.ts.map +1 -0
  40. package/build/lib/extension/extension-config.js +436 -0
  41. package/build/lib/extension/index.d.ts +48 -0
  42. package/build/lib/extension/index.d.ts.map +1 -0
  43. package/build/lib/extension/index.js +74 -0
  44. package/build/lib/extension/manifest.d.ts +174 -0
  45. package/build/lib/extension/manifest.d.ts.map +1 -0
  46. package/build/lib/extension/manifest.js +256 -0
  47. package/build/lib/extension/package-changed.d.ts +11 -0
  48. package/build/lib/extension/package-changed.d.ts.map +1 -0
  49. package/build/lib/extension/package-changed.js +68 -0
  50. package/build/lib/extension/plugin-config.d.ts +57 -0
  51. package/build/lib/extension/plugin-config.d.ts.map +1 -0
  52. package/build/lib/extension/plugin-config.js +78 -0
  53. package/build/lib/grid-register.d.ts +10 -0
  54. package/build/lib/grid-register.d.ts.map +1 -0
  55. package/build/lib/grid-register.js +21 -25
  56. package/build/lib/logger.d.ts +3 -0
  57. package/build/lib/logger.d.ts.map +1 -0
  58. package/build/lib/logger.js +4 -6
  59. package/build/lib/logsink.d.ts +4 -0
  60. package/build/lib/logsink.d.ts.map +1 -0
  61. package/build/lib/logsink.js +14 -17
  62. package/build/lib/main.d.ts +55 -0
  63. package/build/lib/main.d.ts.map +1 -0
  64. package/build/lib/main.js +189 -90
  65. package/build/lib/schema/arg-spec.d.ts +143 -0
  66. package/build/lib/schema/arg-spec.d.ts.map +1 -0
  67. package/build/lib/schema/arg-spec.js +119 -0
  68. package/build/lib/schema/cli-args.d.ts +19 -0
  69. package/build/lib/schema/cli-args.d.ts.map +1 -0
  70. package/build/lib/schema/cli-args.js +180 -0
  71. package/build/lib/schema/cli-transformers.d.ts +5 -0
  72. package/build/lib/schema/cli-transformers.d.ts.map +1 -0
  73. package/build/lib/schema/cli-transformers.js +74 -0
  74. package/build/lib/schema/index.d.ts +3 -0
  75. package/build/lib/schema/index.d.ts.map +1 -0
  76. package/build/lib/schema/index.js +34 -0
  77. package/build/lib/schema/keywords.d.ts +24 -0
  78. package/build/lib/schema/keywords.d.ts.map +1 -0
  79. package/build/lib/schema/keywords.js +70 -0
  80. package/build/lib/schema/schema.d.ts +259 -0
  81. package/build/lib/schema/schema.d.ts.map +1 -0
  82. package/build/lib/schema/schema.js +452 -0
  83. package/build/lib/utils.d.ts +66 -0
  84. package/build/lib/utils.d.ts.map +1 -0
  85. package/build/lib/utils.js +35 -139
  86. package/build/tsconfig.tsbuildinfo +1 -0
  87. package/build/types/appium-manifest.d.ts +59 -0
  88. package/build/types/appium-manifest.d.ts.map +1 -0
  89. package/build/types/cli.d.ts +112 -0
  90. package/build/types/cli.d.ts.map +1 -0
  91. package/build/types/extension-manifest.d.ts +55 -0
  92. package/build/types/extension-manifest.d.ts.map +1 -0
  93. package/build/types/index.d.ts +16 -0
  94. package/build/types/index.d.ts.map +1 -0
  95. package/driver.d.ts +1 -0
  96. package/driver.js +14 -0
  97. package/index.js +11 -0
  98. package/lib/appium.js +517 -186
  99. package/lib/cli/args.js +269 -422
  100. package/lib/cli/driver-command.js +58 -23
  101. package/lib/cli/extension-command.js +612 -260
  102. package/lib/cli/extension.js +34 -16
  103. package/lib/cli/parser.js +241 -83
  104. package/lib/cli/plugin-command.js +48 -20
  105. package/lib/cli/utils.js +24 -10
  106. package/lib/config-file.js +219 -0
  107. package/lib/config.js +210 -91
  108. package/lib/constants.js +69 -0
  109. package/lib/extension/driver-config.js +249 -0
  110. package/lib/extension/extension-config.js +679 -0
  111. package/lib/extension/index.js +116 -0
  112. package/lib/extension/manifest.js +475 -0
  113. package/lib/extension/package-changed.js +64 -0
  114. package/lib/extension/plugin-config.js +113 -0
  115. package/lib/grid-register.js +49 -35
  116. package/lib/logger.js +1 -2
  117. package/lib/logsink.js +38 -33
  118. package/lib/main.js +303 -100
  119. package/lib/schema/arg-spec.js +229 -0
  120. package/lib/schema/cli-args.js +238 -0
  121. package/lib/schema/cli-transformers.js +115 -0
  122. package/lib/schema/index.js +2 -0
  123. package/lib/schema/keywords.js +136 -0
  124. package/lib/schema/schema.js +717 -0
  125. package/lib/utils.js +121 -140
  126. package/package.json +75 -85
  127. package/plugin.d.ts +1 -0
  128. package/plugin.js +13 -0
  129. package/scripts/autoinstall-extensions.js +177 -0
  130. package/support.d.ts +1 -0
  131. package/support.js +13 -0
  132. package/types/appium-manifest.ts +73 -0
  133. package/types/cli.ts +146 -0
  134. package/types/extension-manifest.ts +64 -0
  135. package/types/index.ts +21 -0
  136. package/CHANGELOG.md +0 -3515
  137. package/bin/ios-webkit-debug-proxy-launcher.js +0 -71
  138. package/build/lib/cli/npm.js +0 -206
  139. package/build/lib/cli/parser-helpers.js +0 -82
  140. package/build/lib/driver-config.js +0 -77
  141. package/build/lib/drivers.js +0 -96
  142. package/build/lib/extension-config.js +0 -253
  143. package/build/lib/plugin-config.js +0 -59
  144. package/build/lib/plugins.js +0 -14
  145. package/lib/cli/npm.js +0 -183
  146. package/lib/cli/parser-helpers.js +0 -79
  147. package/lib/driver-config.js +0 -46
  148. package/lib/drivers.js +0 -81
  149. package/lib/extension-config.js +0 -209
  150. package/lib/plugin-config.js +0 -34
  151. package/lib/plugins.js +0 -10
@@ -5,21 +5,25 @@ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefau
5
5
  Object.defineProperty(exports, "__esModule", {
6
6
  value: true
7
7
  });
8
- exports.default = void 0;
8
+ exports.default = exports.ExtensionCommand = void 0;
9
9
 
10
10
  require("source-map-support/register");
11
11
 
12
- var _lodash = _interopRequireDefault(require("lodash"));
12
+ var _bluebird = _interopRequireDefault(require("bluebird"));
13
13
 
14
- var _npm = _interopRequireDefault(require("./npm"));
14
+ var _lodash = _interopRequireDefault(require("lodash"));
15
15
 
16
16
  var _path = _interopRequireDefault(require("path"));
17
17
 
18
- var _appiumSupport = require("appium-support");
18
+ var _support = require("@appium/support");
19
19
 
20
20
  var _utils = require("./utils");
21
21
 
22
- var _extensionConfig = require("../extension-config");
22
+ var _teen_process = require("teen_process");
23
+
24
+ var _extensionConfig = require("../extension/extension-config");
25
+
26
+ var _packageChanged = require("../extension/package-changed");
23
27
 
24
28
  const UPDATE_ALL = 'installed';
25
29
 
@@ -28,23 +32,34 @@ class NotUpdatableError extends Error {}
28
32
  class NoUpdatesAvailableError extends Error {}
29
33
 
30
34
  class ExtensionCommand {
35
+ config;
36
+ knownExtensions;
37
+ isJsonOutput;
38
+
31
39
  constructor({
32
40
  config,
33
- json,
34
- type
41
+ json
35
42
  }) {
36
43
  this.config = config;
37
- this.type = type;
38
- this.isJsonOutput = json;
39
- this.npm = new _npm.default(this.config.appiumHome);
40
- this.knownExtensions = {};
44
+ this.log = new _support.console.CliConsole({
45
+ jsonMode: json
46
+ });
47
+ this.isJsonOutput = Boolean(json);
48
+ }
49
+
50
+ get type() {
51
+ return this.config.extensionType;
52
+ }
53
+
54
+ _createFatalError(message) {
55
+ return new Error(this.log.decorate(message, 'error'));
41
56
  }
42
57
 
43
58
  async execute(args) {
44
59
  const cmd = args[`${this.type}Command`];
45
60
 
46
- if (!_lodash.default.isFunction(ExtensionCommand.prototype[cmd])) {
47
- throw new Error(`Cannot handle ${this.type} command ${cmd}`);
61
+ if (!_lodash.default.isFunction(this[cmd])) {
62
+ throw this._createFatalError(`Cannot handle ${this.type} command ${cmd}`);
48
63
  }
49
64
 
50
65
  const executeCmd = this[cmd].bind(this);
@@ -80,12 +95,7 @@ class ExtensionCommand {
80
95
  }
81
96
 
82
97
  for (const [ext, data] of _lodash.default.toPairs(exts)) {
83
- const {
84
- installed,
85
- installType
86
- } = data;
87
-
88
- if (!installed || installType !== _extensionConfig.INSTALL_TYPE_NPM) {
98
+ if (!data.installed || data.installType !== _extensionConfig.INSTALL_TYPE_NPM) {
89
99
  continue;
90
100
  }
91
101
 
@@ -95,127 +105,178 @@ class ExtensionCommand {
95
105
  data.upToDate = updates.safeUpdate === null && updates.unsafeUpdate === null;
96
106
  }
97
107
  });
108
+ const listData = exts;
98
109
 
99
110
  if (this.isJsonOutput) {
100
- return exts;
111
+ return listData;
101
112
  }
102
113
 
103
- for (const [name, {
104
- installType,
105
- installSpec,
106
- installed,
107
- updateVersion,
108
- unsafeUpdateVersion,
109
- version,
110
- upToDate
111
- }] of _lodash.default.toPairs(exts)) {
112
- let typeTxt;
113
-
114
- switch (installType) {
115
- case _extensionConfig.INSTALL_TYPE_GIT:
116
- case _extensionConfig.INSTALL_TYPE_GITHUB:
117
- typeTxt = `(cloned from ${installSpec})`.yellow;
118
- break;
119
-
120
- case _extensionConfig.INSTALL_TYPE_LOCAL:
121
- typeTxt = `(linked from ${installSpec})`.magenta;
122
- break;
123
-
124
- default:
125
- typeTxt = '(NPM)';
114
+ for (const [name, data] of _lodash.default.toPairs(listData)) {
115
+ let installTxt = ' [not installed]'.grey;
116
+ let updateTxt = '';
117
+ let upToDateTxt = '';
118
+ let unsafeUpdateTxt = '';
119
+
120
+ if (data.installed) {
121
+ const {
122
+ installType,
123
+ installSpec,
124
+ updateVersion,
125
+ unsafeUpdateVersion,
126
+ version,
127
+ upToDate
128
+ } = data;
129
+ let typeTxt;
130
+
131
+ switch (installType) {
132
+ case _extensionConfig.INSTALL_TYPE_GIT:
133
+ case _extensionConfig.INSTALL_TYPE_GITHUB:
134
+ typeTxt = `(cloned from ${installSpec})`.yellow;
135
+ break;
136
+
137
+ case _extensionConfig.INSTALL_TYPE_LOCAL:
138
+ typeTxt = `(linked from ${installSpec})`.magenta;
139
+ break;
140
+
141
+ default:
142
+ typeTxt = '(NPM)';
143
+ }
144
+
145
+ installTxt = `@${version.yellow} ${('[installed ' + typeTxt + ']').green}`;
146
+
147
+ if (showUpdates) {
148
+ if (updateVersion) {
149
+ updateTxt = ` [${updateVersion} available]`.magenta;
150
+ }
151
+
152
+ if (upToDate) {
153
+ upToDateTxt = ` [Up to date]`.green;
154
+ }
155
+
156
+ if (unsafeUpdateVersion) {
157
+ unsafeUpdateTxt = ` [${unsafeUpdateVersion} available (potentially unsafe)]`.cyan;
158
+ }
159
+ }
126
160
  }
127
161
 
128
- const installTxt = installed ? `@${version.yellow} ${('[installed ' + typeTxt + ']').green}` : ' [not installed]'.grey;
129
- const updateTxt = showUpdates && updateVersion ? ` [${updateVersion} available]`.magenta : '';
130
- const upToDateTxt = showUpdates && upToDate ? ` [Up to date]`.green : '';
131
- const unsafeUpdateTxt = showUpdates && unsafeUpdateVersion ? ` [${unsafeUpdateVersion} available (potentially unsafe)]`.cyan : '';
132
- console.log(`- ${name.yellow}${installTxt}${updateTxt}${upToDateTxt}${unsafeUpdateTxt}`);
162
+ this.log.log(`- ${name.yellow}${installTxt}${updateTxt}${upToDateTxt}${unsafeUpdateTxt}`);
133
163
  }
134
164
 
135
- return exts;
165
+ return listData;
136
166
  }
137
167
 
138
- async install({
139
- ext,
168
+ async _install({
169
+ installSpec,
140
170
  installType,
141
171
  packageName
142
172
  }) {
143
- (0, _utils.log)(this.isJsonOutput, `Attempting to find and install ${this.type} '${ext}'`);
144
173
  let extData;
145
- let installSpec = ext;
146
174
 
147
175
  if (packageName && [_extensionConfig.INSTALL_TYPE_LOCAL, _extensionConfig.INSTALL_TYPE_NPM].includes(installType)) {
148
- throw new Error(`When using --source=${installType}, cannot also use --package`);
176
+ throw this._createFatalError(`When using --source=${installType}, cannot also use --package`);
149
177
  }
150
178
 
151
179
  if (!packageName && [_extensionConfig.INSTALL_TYPE_GIT, _extensionConfig.INSTALL_TYPE_GITHUB].includes(installType)) {
152
- throw new Error(`When using --source=${installType}, must also use --package`);
180
+ throw this._createFatalError(`When using --source=${installType}, must also use --package`);
153
181
  }
154
182
 
155
- if (installType === _extensionConfig.INSTALL_TYPE_LOCAL) {
156
- const msg = `Linking ${this.type} from local path`;
157
- const pkgJsonData = await (0, _utils.spinWith)(this.isJsonOutput, msg, async () => await this.npm.linkPackage(installSpec));
158
- extData = this.getExtensionFields(pkgJsonData);
159
- extData.installPath = extData.pkgName;
160
- } else if (installType === _extensionConfig.INSTALL_TYPE_GITHUB) {
183
+ let installOpts;
184
+ let probableExtName = '';
185
+
186
+ if (installType === _extensionConfig.INSTALL_TYPE_GITHUB) {
161
187
  if (installSpec.split('/').length !== 2) {
162
- throw new Error(`Github ${this.type} spec ${installSpec} appeared to be invalid; ` + 'it should be of the form <org>/<repo>');
188
+ throw this._createFatalError(`Github ${this.type} spec ${installSpec} appeared to be invalid; ` + 'it should be of the form <org>/<repo>');
163
189
  }
164
190
 
165
- extData = await this.installViaNpm({
166
- ext: installSpec,
191
+ installOpts = {
192
+ installSpec,
167
193
  pkgName: packageName
168
- });
194
+ };
195
+ probableExtName = installSpec;
169
196
  } else if (installType === _extensionConfig.INSTALL_TYPE_GIT) {
170
197
  installSpec = installSpec.replace(/\.git$/, '');
171
- extData = await this.installViaNpm({
172
- ext: installSpec,
198
+ installOpts = {
199
+ installSpec,
173
200
  pkgName: packageName
174
- });
201
+ };
202
+ probableExtName = installSpec;
175
203
  } else {
176
- let name, pkgVer;
177
- const splits = installSpec.split('@');
204
+ let pkgName, pkgVer;
178
205
 
179
- if (installSpec[0] === '@') {
180
- [name, pkgVer] = [`@${splits[1]}`, splits[2]];
206
+ if (installType === _extensionConfig.INSTALL_TYPE_LOCAL) {
207
+ pkgName = _path.default.isAbsolute(installSpec) ? installSpec : _path.default.resolve(installSpec);
181
208
  } else {
182
- [name, pkgVer] = splits;
183
- }
209
+ let name;
210
+ const splits = installSpec.split('@');
184
211
 
185
- let pkgName;
212
+ if (installSpec[0] === '@') {
213
+ [name, pkgVer] = [`@${splits[1]}`, splits[2]];
214
+ } else {
215
+ [name, pkgVer] = splits;
216
+ }
186
217
 
187
- if (installType === _extensionConfig.INSTALL_TYPE_NPM) {
188
- pkgName = name;
189
- } else {
190
- const knownNames = Object.keys(this.knownExtensions);
218
+ if (installType === _extensionConfig.INSTALL_TYPE_NPM) {
219
+ pkgName = name;
220
+ } else {
221
+ const knownNames = Object.keys(this.knownExtensions);
191
222
 
192
- if (!_lodash.default.includes(knownNames, name)) {
193
- const msg = `Could not resolve ${this.type}; are you sure it's in the list ` + `of supported ${this.type}s? ${JSON.stringify(knownNames)}`;
194
- throw new Error(msg);
195
- }
223
+ if (!_lodash.default.includes(knownNames, name)) {
224
+ const msg = `Could not resolve ${this.type}; are you sure it's in the list ` + `of supported ${this.type}s? ${JSON.stringify(knownNames)}`;
225
+ throw this._createFatalError(msg);
226
+ }
196
227
 
197
- pkgName = this.knownExtensions[name];
198
- installType = _extensionConfig.INSTALL_TYPE_NPM;
228
+ probableExtName = name;
229
+ pkgName = this.knownExtensions[name];
230
+ installType = _extensionConfig.INSTALL_TYPE_NPM;
231
+ }
199
232
  }
200
233
 
201
- extData = await this.installViaNpm({
202
- ext,
234
+ installOpts = {
235
+ installSpec,
203
236
  pkgName,
204
237
  pkgVer
205
- });
238
+ };
239
+ }
240
+
241
+ if (probableExtName && this.config.isInstalled(probableExtName)) {
242
+ throw this._createFatalError(`A ${this.type} named "${probableExtName}" is already installed. ` + `Did you mean to update? Run "appium ${this.type} update". See ` + `installed ${this.type}s with "appium ${this.type} list --installed".`);
206
243
  }
207
244
 
245
+ extData = await this.installViaNpm(installOpts);
208
246
  const extName = extData[`${this.type}Name`];
209
- delete extData[`${this.type}Name`];
210
247
 
211
248
  if (this.config.isInstalled(extName)) {
212
- throw new Error(`A ${this.type} named '${extName}' is already installed. ` + `Did you mean to update? 'appium ${this.type} update'. See ` + `installed ${this.type}s with 'appium ${this.type} list --installed'.`);
249
+ throw this._createFatalError(`A ${this.type} named "${extName}" is already installed. ` + `Did you mean to update? Run "appium ${this.type} update". See ` + `installed ${this.type}s with "appium ${this.type} list --installed".`);
250
+ }
251
+
252
+ delete extData[`${this.type}Name`];
253
+ const extManifest = { ...extData,
254
+ installType,
255
+ installSpec
256
+ };
257
+ const [errors, warnings] = await _bluebird.default.all([this.config.getProblems(extName, extManifest), this.config.getWarnings(extName, extManifest)]);
258
+ const errorMap = new Map([[extName, errors]]);
259
+ const warningMap = new Map([[extName, warnings]]);
260
+ const {
261
+ errorSummaries,
262
+ warningSummaries
263
+ } = this.config.getValidationResultSummaries(errorMap, warningMap);
264
+
265
+ if (!_lodash.default.isEmpty(errorSummaries)) {
266
+ throw this._createFatalError(errorSummaries.join('\n'));
267
+ }
268
+
269
+ if (!_lodash.default.isEmpty(warningSummaries)) {
270
+ this.log.warn(warningSummaries.join('\n'));
213
271
  }
214
272
 
215
- extData.installType = installType;
216
- extData.installSpec = installSpec;
217
- await this.config.addExtension(extName, extData);
218
- (0, _utils.log)(this.isJsonOutput, this.getPostInstallText({
273
+ await this.config.addExtension(extName, extManifest);
274
+
275
+ if (await _support.env.hasAppiumDependency(this.config.appiumHome)) {
276
+ await (0, _packageChanged.packageDidChange)(this.config.appiumHome);
277
+ }
278
+
279
+ this.log.info(this.getPostInstallText({
219
280
  extName,
220
281
  extData
221
282
  }));
@@ -223,81 +284,101 @@ class ExtensionCommand {
223
284
  }
224
285
 
225
286
  async installViaNpm({
226
- ext,
287
+ installSpec,
227
288
  pkgName,
228
289
  pkgVer
229
290
  }) {
230
291
  const npmSpec = `${pkgName}${pkgVer ? '@' + pkgVer : ''}`;
231
- const specMsg = npmSpec === ext ? '' : ` using NPM install spec '${npmSpec}'`;
232
- const msg = `Installing '${ext}'${specMsg}`;
292
+ const specMsg = npmSpec === installSpec ? '' : ` using NPM install spec '${npmSpec}'`;
293
+ const msg = `Installing '${installSpec}'${specMsg}`;
233
294
 
234
295
  try {
235
- const pkgJsonData = await (0, _utils.spinWith)(this.isJsonOutput, msg, async () => await this.npm.installPackage({
236
- pkgDir: _path.default.resolve(this.config.appiumHome, pkgName),
237
- pkgName,
238
- pkgVer
239
- }));
240
- const extData = this.getExtensionFields(pkgJsonData);
241
- extData.installPath = pkgName;
242
- return extData;
296
+ const pkgJsonData = await (0, _utils.spinWith)(this.isJsonOutput, msg, async () => {
297
+ const pkgJsonData = await _support.npm.installPackage(this.config.appiumHome, pkgName, {
298
+ pkgVer
299
+ });
300
+ this.validatePackageJson(pkgJsonData, installSpec);
301
+ return pkgJsonData;
302
+ });
303
+ return this.getExtensionFields(pkgJsonData);
243
304
  } catch (err) {
244
- throw new Error(`Encountered an error when installing package: ${err.message}`);
305
+ throw this._createFatalError(`Encountered an error when installing package: ${err.message}`);
245
306
  }
246
307
  }
247
308
 
248
- getPostInstallText() {
249
- throw new Error('Must be implemented in final class');
309
+ getPostInstallText(args) {
310
+ throw this._createFatalError('Must be implemented in final class');
250
311
  }
251
312
 
252
- getExtensionFields(pkgJsonData) {
253
- if (!pkgJsonData.appium) {
254
- throw new Error(`Installed driver did not have an 'appium' section in its ` + `package.json file as expected`);
255
- }
256
-
313
+ getExtensionFields(pkgJson) {
257
314
  const {
258
315
  appium,
259
316
  name,
260
- version
261
- } = pkgJsonData;
262
- this.validateExtensionFields(appium);
263
- return { ...appium,
317
+ version,
318
+ peerDependencies
319
+ } = pkgJson;
320
+ const result = { ...appium,
264
321
  pkgName: name,
265
- version
322
+ version,
323
+ appiumVersion: peerDependencies === null || peerDependencies === void 0 ? void 0 : peerDependencies.appium
266
324
  };
325
+ return result;
267
326
  }
268
327
 
269
- validateExtensionFields() {
270
- throw new Error('Must be implemented in final class');
271
- }
328
+ validatePackageJson(pkgJson, installSpec) {
329
+ const {
330
+ appium,
331
+ name,
332
+ version
333
+ } = pkgJson;
272
334
 
273
- async uninstall({
274
- ext
275
- }) {
276
- if (!this.config.isInstalled(ext)) {
277
- throw new Error(`Can't uninstall ${this.type} '${ext}'; it is not installed`);
335
+ const createMissingFieldError = field => new ReferenceError(`${this.type} "${installSpec}" invalid; missing a \`${field}\` field of its \`package.json\``);
336
+
337
+ if (!name) {
338
+ throw createMissingFieldError('name');
278
339
  }
279
340
 
280
- try {
281
- await _appiumSupport.fs.rimraf(this.config.getInstallPath(ext));
282
- } finally {
283
- await this.config.removeExtension(ext);
341
+ if (!version) {
342
+ throw createMissingFieldError('version');
343
+ }
344
+
345
+ if (!appium) {
346
+ throw createMissingFieldError('appium');
347
+ }
348
+
349
+ this.validateExtensionFields(appium, installSpec);
350
+ return true;
351
+ }
352
+
353
+ validateExtensionFields(extMetadata, installSpec) {
354
+ throw this._createFatalError('Must be implemented in final class');
355
+ }
356
+
357
+ async _uninstall({
358
+ installSpec
359
+ }) {
360
+ if (!this.config.isInstalled(installSpec)) {
361
+ throw this._createFatalError(`Can't uninstall ${this.type} '${installSpec}'; it is not installed`);
284
362
  }
285
363
 
286
- (0, _utils.log)(this.isJsonOutput, `Successfully uninstalled ${this.type} '${ext}'`.green);
364
+ const pkgName = this.config.installedExtensions[installSpec].pkgName;
365
+ await _support.npm.uninstallPackage(this.config.appiumHome, pkgName);
366
+ await this.config.removeExtension(installSpec);
367
+ this.log.ok(`Successfully uninstalled ${this.type} '${installSpec}'`.green);
287
368
  return this.config.installedExtensions;
288
369
  }
289
370
 
290
- async update({
291
- ext,
371
+ async _update({
372
+ installSpec,
292
373
  unsafe
293
374
  }) {
294
- const shouldUpdateAll = ext === UPDATE_ALL;
375
+ const shouldUpdateAll = installSpec === UPDATE_ALL;
295
376
 
296
- if (!shouldUpdateAll && !this.config.isInstalled(ext)) {
297
- throw new Error(`The ${this.type} '${ext}' was not installed, so can't be updated`);
377
+ if (!shouldUpdateAll && !this.config.isInstalled(installSpec)) {
378
+ throw this._createFatalError(`The ${this.type} "${installSpec}" was not installed, so can't be updated`);
298
379
  }
299
380
 
300
- const extsToUpdate = shouldUpdateAll ? Object.keys(this.config.installedExtensions) : [ext];
381
+ const extsToUpdate = shouldUpdateAll ? Object.keys(this.config.installedExtensions) : [installSpec];
301
382
  const errors = {};
302
383
  const updates = {};
303
384
 
@@ -319,7 +400,7 @@ class ExtensionCommand {
319
400
  });
320
401
 
321
402
  if (!unsafe && !update.safeUpdate) {
322
- throw new Error(`The ${this.type} '${e}' has a major revision update ` + `(${update.current} => ${update.unsafeUpdate}), which could include ` + `breaking changes. If you want to apply this update, re-run with --unsafe`);
403
+ throw this._createFatalError(`The ${this.type} '${e}' has a major revision update ` + `(${update.current} => ${update.unsafeUpdate}), which could include ` + `breaking changes. If you want to apply this update, re-run with --unsafe`);
323
404
  }
324
405
 
325
406
  const updateVer = unsafe && update.unsafeUpdate ? update.unsafeUpdate : update.safeUpdate;
@@ -333,19 +414,19 @@ class ExtensionCommand {
333
414
  }
334
415
  }
335
416
 
336
- (0, _utils.log)(this.isJsonOutput, 'Update report:');
417
+ this.log.info('Update report:');
337
418
 
338
419
  for (const [e, update] of _lodash.default.toPairs(updates)) {
339
- (0, _utils.log)(this.isJsonOutput, `- ${this.type} ${e} updated: ${update.from} => ${update.to}`.green);
420
+ this.log.ok(` - ${this.type} ${e} updated: ${update.from} => ${update.to}`.green);
340
421
  }
341
422
 
342
423
  for (const [e, err] of _lodash.default.toPairs(errors)) {
343
424
  if (err instanceof NotUpdatableError) {
344
- (0, _utils.log)(this.isJsonOutput, `- '${e}' was not installed via npm, so we could not check ` + `for updates`.yellow);
425
+ this.log.warn(` - '${e}' was not installed via npm, so we could not check ` + `for updates`.yellow);
345
426
  } else if (err instanceof NoUpdatesAvailableError) {
346
- (0, _utils.log)(this.isJsonOutput, `- '${e}' had no updates available`.yellow);
427
+ this.log.info(` - '${e}' had no updates available`.yellow);
347
428
  } else {
348
- (0, _utils.log)(this.isJsonOutput, `- '${e}' failed to update: ${err}`.red);
429
+ this.log.error(` - '${e}' failed to update: ${err}`.red);
349
430
  }
350
431
  }
351
432
 
@@ -360,10 +441,10 @@ class ExtensionCommand {
360
441
  version,
361
442
  pkgName
362
443
  } = this.config.installedExtensions[ext];
363
- let unsafeUpdate = await this.npm.getLatestVersion(pkgName);
364
- let safeUpdate = await this.npm.getLatestSafeUpgradeVersion(pkgName, version);
444
+ let unsafeUpdate = await _support.npm.getLatestVersion(this.config.appiumHome, pkgName);
445
+ let safeUpdate = await _support.npm.getLatestSafeUpgradeVersion(this.config.appiumHome, pkgName, version);
365
446
 
366
- if (!_appiumSupport.util.compareVersions(unsafeUpdate, '>', version)) {
447
+ if (!_support.util.compareVersions(unsafeUpdate, '>', version)) {
367
448
  unsafeUpdate = null;
368
449
  safeUpdate = null;
369
450
  }
@@ -372,7 +453,7 @@ class ExtensionCommand {
372
453
  unsafeUpdate = null;
373
454
  }
374
455
 
375
- if (safeUpdate && !_appiumSupport.util.compareVersions(safeUpdate, '>', version)) {
456
+ if (safeUpdate && !_support.util.compareVersions(safeUpdate, '>', version)) {
376
457
  safeUpdate = null;
377
458
  }
378
459
 
@@ -383,22 +464,71 @@ class ExtensionCommand {
383
464
  };
384
465
  }
385
466
 
386
- async updateExtension(ext, version) {
467
+ async updateExtension(installSpec, version) {
387
468
  const {
388
469
  pkgName
389
- } = this.config.installedExtensions[ext];
390
- await this.installViaNpm({
391
- ext,
470
+ } = this.config.installedExtensions[installSpec];
471
+ const extData = await this.installViaNpm({
472
+ installSpec,
392
473
  pkgName,
393
474
  pkgVer: version
394
475
  });
395
- this.config.installedExtensions[ext].version = version;
396
- await this.config.write();
476
+ delete extData[`${this.type}Name`];
477
+ await this.config.updateExtension(installSpec, extData);
397
478
  }
398
479
 
399
- }
480
+ async _run({
481
+ installSpec,
482
+ scriptName
483
+ }) {
484
+ if (!this.config.isInstalled(installSpec)) {
485
+ throw this._createFatalError(`The ${this.type} "${installSpec}" is not installed`);
486
+ }
487
+
488
+ const extConfig = this.config.installedExtensions[installSpec];
489
+
490
+ if (!extConfig.scripts) {
491
+ throw this._createFatalError(`The ${this.type} named '${installSpec}' does not contain the ` + `"scripts" field underneath the "appium" field in its package.json`);
492
+ }
400
493
 
401
- exports.default = ExtensionCommand;require('source-map-support').install();
494
+ const extScripts = extConfig.scripts;
402
495
 
496
+ if (!_lodash.default.isPlainObject(extScripts)) {
497
+ throw this._createFatalError(`The ${this.type} named '${installSpec}' "scripts" field must be a plain object`);
498
+ }
499
+
500
+ if (!_lodash.default.has(extScripts, scriptName)) {
501
+ throw this._createFatalError(`The ${this.type} named '${installSpec}' does not support the script: '${scriptName}'`);
502
+ }
503
+
504
+ const runner = new _teen_process.SubProcess(process.execPath, [extScripts[scriptName]], {
505
+ cwd: this.config.getInstallPath(installSpec)
506
+ });
507
+ const output = new _utils.RingBuffer(50);
508
+ runner.on('stream-line', line => {
509
+ output.enqueue(line);
510
+ this.log.log(line);
511
+ });
512
+ await runner.start(0);
513
+
514
+ try {
515
+ await runner.join();
516
+ this.log.ok(`${scriptName} successfully ran`.green);
517
+ return {
518
+ output: output.getBuff()
519
+ };
520
+ } catch (err) {
521
+ this.log.error(`Encountered an error when running '${scriptName}': ${err.message}`.red);
522
+ return {
523
+ error: err.message,
524
+ output: output.getBuff()
525
+ };
526
+ }
527
+ }
528
+
529
+ }
403
530
 
404
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/cli/extension-command.js"],"names":["UPDATE_ALL","NotUpdatableError","Error","NoUpdatesAvailableError","ExtensionCommand","constructor","config","json","type","isJsonOutput","npm","NPM","appiumHome","knownExtensions","execute","args","cmd","_","isFunction","prototype","executeCmd","bind","list","showInstalled","showUpdates","lsMsg","installedNames","Object","keys","installedExtensions","knownNames","exts","reduce","acc","name","includes","installed","pkgName","ext","data","toPairs","installType","INSTALL_TYPE_NPM","updates","checkForExtensionUpdate","updateVersion","safeUpdate","unsafeUpdateVersion","unsafeUpdate","upToDate","installSpec","version","typeTxt","INSTALL_TYPE_GIT","INSTALL_TYPE_GITHUB","yellow","INSTALL_TYPE_LOCAL","magenta","installTxt","green","grey","updateTxt","upToDateTxt","unsafeUpdateTxt","cyan","console","log","install","packageName","extData","msg","pkgJsonData","linkPackage","getExtensionFields","installPath","split","length","installViaNpm","replace","pkgVer","splits","JSON","stringify","extName","isInstalled","addExtension","getPostInstallText","npmSpec","specMsg","installPackage","pkgDir","path","resolve","err","message","appium","validateExtensionFields","uninstall","fs","rimraf","getInstallPath","removeExtension","update","unsafe","shouldUpdateAll","extsToUpdate","errors","e","current","updateVer","updateExtension","from","to","red","getLatestVersion","getLatestSafeUpgradeVersion","util","compareVersions","write"],"mappings":";;;;;;;;;;;AAEA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAGA,MAAMA,UAAU,GAAG,WAAnB;;AAEA,MAAMC,iBAAN,SAAgCC,KAAhC,CAAsC;;AACtC,MAAMC,uBAAN,SAAsCD,KAAtC,CAA4C;;AAE7B,MAAME,gBAAN,CAAuB;AAepCC,EAAAA,WAAW,CAAE;AAACC,IAAAA,MAAD;AAASC,IAAAA,IAAT;AAAeC,IAAAA;AAAf,GAAF,EAAwB;AACjC,SAAKF,MAAL,GAAcA,MAAd;AACA,SAAKE,IAAL,GAAYA,IAAZ;AACA,SAAKC,YAAL,GAAoBF,IAApB;AACA,SAAKG,GAAL,GAAW,IAAIC,YAAJ,CAAQ,KAAKL,MAAL,CAAYM,UAApB,CAAX;AACA,SAAKC,eAAL,GAAuB,EAAvB;AACD;;AAQD,QAAMC,OAAN,CAAeC,IAAf,EAAqB;AACnB,UAAMC,GAAG,GAAGD,IAAI,CAAE,GAAE,KAAKP,IAAK,SAAd,CAAhB;;AACA,QAAI,CAACS,gBAAEC,UAAF,CAAad,gBAAgB,CAACe,SAAjB,CAA2BH,GAA3B,CAAb,CAAL,EAAoD;AAClD,YAAM,IAAId,KAAJ,CAAW,iBAAgB,KAAKM,IAAK,YAAWQ,GAAI,EAApD,CAAN;AACD;;AACD,UAAMI,UAAU,GAAG,KAAKJ,GAAL,EAAUK,IAAV,CAAe,IAAf,CAAnB;AACA,WAAO,MAAMD,UAAU,CAACL,IAAD,CAAvB;AACD;;AAcD,QAAMO,IAAN,CAAY;AAACC,IAAAA,aAAD;AAAgBC,IAAAA;AAAhB,GAAZ,EAA0C;AACxC,UAAMC,KAAK,GAAI,WAAUF,aAAa,GAAG,WAAH,GAAiB,WAAY,IAAG,KAAKf,IAAK,GAAhF;AACA,UAAMkB,cAAc,GAAGC,MAAM,CAACC,IAAP,CAAY,KAAKtB,MAAL,CAAYuB,mBAAxB,CAAvB;AACA,UAAMC,UAAU,GAAGH,MAAM,CAACC,IAAP,CAAY,KAAKf,eAAjB,CAAnB;AACA,UAAMkB,IAAI,GAAG,CAAC,GAAGL,cAAJ,EAAoB,GAAGI,UAAvB,EAAmCE,MAAnC,CAA0C,CAACC,GAAD,EAAMC,IAAN,KAAe;AACpE,UAAI,CAACD,GAAG,CAACC,IAAD,CAAR,EAAgB;AACd,YAAIR,cAAc,CAACS,QAAf,CAAwBD,IAAxB,CAAJ,EAAmC;AACjCD,UAAAA,GAAG,CAACC,IAAD,CAAH,GAAY,EAAC,GAAG,KAAK5B,MAAL,CAAYuB,mBAAZ,CAAgCK,IAAhC,CAAJ;AAA2CE,YAAAA,SAAS,EAAE;AAAtD,WAAZ;AACD,SAFD,MAEO,IAAI,CAACb,aAAL,EAAoB;AACzBU,UAAAA,GAAG,CAACC,IAAD,CAAH,GAAY;AAACG,YAAAA,OAAO,EAAE,KAAKxB,eAAL,CAAqBqB,IAArB,CAAV;AAAsCE,YAAAA,SAAS,EAAE;AAAjD,WAAZ;AACD;AACF;;AACD,aAAOH,GAAP;AACD,KATY,EASV,EATU,CAAb;AAYA,UAAM,qBAAS,KAAKxB,YAAd,EAA4BgB,KAA5B,EAAmC,YAAY;AACnD,UAAI,CAACD,WAAL,EAAkB;AAChB;AACD;;AACD,WAAK,MAAM,CAACc,GAAD,EAAMC,IAAN,CAAX,IAA0BtB,gBAAEuB,OAAF,CAAUT,IAAV,CAA1B,EAA2C;AACzC,cAAM;AAACK,UAAAA,SAAD;AAAYK,UAAAA;AAAZ,YAA2BF,IAAjC;;AACA,YAAI,CAACH,SAAD,IAAcK,WAAW,KAAKC,iCAAlC,EAAoD;AAGlD;AACD;;AACD,cAAMC,OAAO,GAAG,MAAM,KAAKC,uBAAL,CAA6BN,GAA7B,CAAtB;AACAC,QAAAA,IAAI,CAACM,aAAL,GAAqBF,OAAO,CAACG,UAA7B;AACAP,QAAAA,IAAI,CAACQ,mBAAL,GAA2BJ,OAAO,CAACK,YAAnC;AACAT,QAAAA,IAAI,CAACU,QAAL,GAAgBN,OAAO,CAACG,UAAR,KAAuB,IAAvB,IAA+BH,OAAO,CAACK,YAAR,KAAyB,IAAxE;AACD;AACF,KAhBK,CAAN;;AAoBA,QAAI,KAAKvC,YAAT,EAAuB;AACrB,aAAOsB,IAAP;AACD;;AAED,SAAK,MAAM,CACTG,IADS,EAET;AAACO,MAAAA,WAAD;AAAcS,MAAAA,WAAd;AAA2Bd,MAAAA,SAA3B;AAAsCS,MAAAA,aAAtC;AAAqDE,MAAAA,mBAArD;AAA0EI,MAAAA,OAA1E;AAAmFF,MAAAA;AAAnF,KAFS,CAAX,IAGKhC,gBAAEuB,OAAF,CAAUT,IAAV,CAHL,EAGsB;AACpB,UAAIqB,OAAJ;;AACA,cAAQX,WAAR;AACE,aAAKY,iCAAL;AACA,aAAKC,oCAAL;AACEF,UAAAA,OAAO,GAAI,gBAAeF,WAAY,GAA5B,CAA+BK,MAAzC;AACA;;AACF,aAAKC,mCAAL;AACEJ,UAAAA,OAAO,GAAI,gBAAeF,WAAY,GAA5B,CAA+BO,OAAzC;AACA;;AACF;AACEL,UAAAA,OAAO,GAAG,OAAV;AATJ;;AAWA,YAAMM,UAAU,GAAGtB,SAAS,GACzB,IAAGe,OAAO,CAACI,MAAO,IAAG,CAAC,gBAAgBH,OAAhB,GAA0B,GAA3B,EAAgCO,KAAM,EADlC,GAE1B,mBAAmBC,IAFrB;AAGA,YAAMC,SAAS,GAAGrC,WAAW,IAAIqB,aAAf,GACf,KAAIA,aAAc,aAAnB,CAAgCY,OADhB,GAEhB,EAFF;AAGA,YAAMK,WAAW,GAAGtC,WAAW,IAAIyB,QAAf,GACjB,eAAD,CAAgBU,KADE,GAElB,EAFF;AAGA,YAAMI,eAAe,GAAGvC,WAAW,IAAIuB,mBAAf,GACrB,KAAIA,mBAAoB,kCAAzB,CAA2DiB,IADrC,GAEtB,EAFF;AAIAC,MAAAA,OAAO,CAACC,GAAR,CAAa,KAAIhC,IAAI,CAACqB,MAAO,GAAEG,UAAW,GAAEG,SAAU,GAAEC,WAAY,GAAEC,eAAgB,EAAtF;AACD;;AAED,WAAOhC,IAAP;AACD;;AAeD,QAAMoC,OAAN,CAAe;AAAC7B,IAAAA,GAAD;AAAMG,IAAAA,WAAN;AAAmB2B,IAAAA;AAAnB,GAAf,EAAgD;AAC9C,oBAAI,KAAK3D,YAAT,EAAwB,kCAAiC,KAAKD,IAAK,KAAI8B,GAAI,GAA3E;AAEA,QAAI+B,OAAJ;AACA,QAAInB,WAAW,GAAGZ,GAAlB;;AAEA,QAAI8B,WAAW,IAAI,CAACZ,mCAAD,EAAqBd,iCAArB,EAAuCP,QAAvC,CAAgDM,WAAhD,CAAnB,EAAiF;AAC/E,YAAM,IAAIvC,KAAJ,CAAW,uBAAsBuC,WAAY,6BAA7C,CAAN;AACD;;AAED,QAAI,CAAC2B,WAAD,IAAgB,CAACf,iCAAD,EAAmBC,oCAAnB,EAAwCnB,QAAxC,CAAiDM,WAAjD,CAApB,EAAmF;AACjF,YAAM,IAAIvC,KAAJ,CAAW,uBAAsBuC,WAAY,2BAA7C,CAAN;AACD;;AAED,QAAIA,WAAW,KAAKe,mCAApB,EAAwC;AACtC,YAAMc,GAAG,GAAI,WAAU,KAAK9D,IAAK,kBAAjC;AACA,YAAM+D,WAAW,GAAG,MAAM,qBAAS,KAAK9D,YAAd,EAA4B6D,GAA5B,EAAiC,YACzD,MAAM,KAAK5D,GAAL,CAAS8D,WAAT,CAAqBtB,WAArB,CADkB,CAA1B;AAGAmB,MAAAA,OAAO,GAAG,KAAKI,kBAAL,CAAwBF,WAAxB,CAAV;AACAF,MAAAA,OAAO,CAACK,WAAR,GAAsBL,OAAO,CAAChC,OAA9B;AACD,KAPD,MAOO,IAAII,WAAW,KAAKa,oCAApB,EAAyC;AAC9C,UAAIJ,WAAW,CAACyB,KAAZ,CAAkB,GAAlB,EAAuBC,MAAvB,KAAkC,CAAtC,EAAyC;AACvC,cAAM,IAAI1E,KAAJ,CAAW,UAAS,KAAKM,IAAK,SAAQ0C,WAAY,2BAAxC,GACA,uCADV,CAAN;AAED;;AACDmB,MAAAA,OAAO,GAAG,MAAM,KAAKQ,aAAL,CAAmB;AAACvC,QAAAA,GAAG,EAAEY,WAAN;AAAmBb,QAAAA,OAAO,EAAE+B;AAA5B,OAAnB,CAAhB;AACD,KANM,MAMA,IAAI3B,WAAW,KAAKY,iCAApB,EAAsC;AAG3CH,MAAAA,WAAW,GAAGA,WAAW,CAAC4B,OAAZ,CAAoB,QAApB,EAA8B,EAA9B,CAAd;AACAT,MAAAA,OAAO,GAAG,MAAM,KAAKQ,aAAL,CAAmB;AAACvC,QAAAA,GAAG,EAAEY,WAAN;AAAmBb,QAAAA,OAAO,EAAE+B;AAA5B,OAAnB,CAAhB;AACD,KALM,MAKA;AAML,UAAIlC,IAAJ,EAAU6C,MAAV;AACA,YAAMC,MAAM,GAAG9B,WAAW,CAACyB,KAAZ,CAAkB,GAAlB,CAAf;;AACA,UAAIzB,WAAW,CAAC,CAAD,CAAX,KAAmB,GAAvB,EAA4B;AAE1B,SAAChB,IAAD,EAAO6C,MAAP,IAAiB,CAAE,IAAGC,MAAM,CAAC,CAAD,CAAI,EAAf,EAAkBA,MAAM,CAAC,CAAD,CAAxB,CAAjB;AACD,OAHD,MAGO;AAEL,SAAC9C,IAAD,EAAO6C,MAAP,IAAiBC,MAAjB;AACD;;AACD,UAAI3C,OAAJ;;AAEA,UAAII,WAAW,KAAKC,iCAApB,EAAsC;AAGpCL,QAAAA,OAAO,GAAGH,IAAV;AACD,OAJD,MAIO;AAIL,cAAMJ,UAAU,GAAGH,MAAM,CAACC,IAAP,CAAY,KAAKf,eAAjB,CAAnB;;AACA,YAAI,CAACI,gBAAEkB,QAAF,CAAWL,UAAX,EAAuBI,IAAvB,CAAL,EAAmC;AACjC,gBAAMoC,GAAG,GAAI,qBAAoB,KAAK9D,IAAK,kCAA/B,GACC,gBAAe,KAAKA,IAAK,MAAKyE,IAAI,CAACC,SAAL,CAAepD,UAAf,CAA2B,EADtE;AAEA,gBAAM,IAAI5B,KAAJ,CAAUoE,GAAV,CAAN;AACD;;AACDjC,QAAAA,OAAO,GAAG,KAAKxB,eAAL,CAAqBqB,IAArB,CAAV;AAGAO,QAAAA,WAAW,GAAGC,iCAAd;AACD;;AAED2B,MAAAA,OAAO,GAAG,MAAM,KAAKQ,aAAL,CAAmB;AAACvC,QAAAA,GAAD;AAAMD,QAAAA,OAAN;AAAe0C,QAAAA;AAAf,OAAnB,CAAhB;AACD;;AAED,UAAMI,OAAO,GAAGd,OAAO,CAAE,GAAE,KAAK7D,IAAK,MAAd,CAAvB;AACA,WAAO6D,OAAO,CAAE,GAAE,KAAK7D,IAAK,MAAd,CAAd;;AAEA,QAAI,KAAKF,MAAL,CAAY8E,WAAZ,CAAwBD,OAAxB,CAAJ,EAAsC;AACpC,YAAM,IAAIjF,KAAJ,CAAW,KAAI,KAAKM,IAAK,WAAU2E,OAAQ,0BAAjC,GACC,mCAAkC,KAAK3E,IAAK,gBAD7C,GAEC,aAAY,KAAKA,IAAK,kBAAiB,KAAKA,IAAK,qBAF5D,CAAN;AAGD;;AAED6D,IAAAA,OAAO,CAAC5B,WAAR,GAAsBA,WAAtB;AACA4B,IAAAA,OAAO,CAACnB,WAAR,GAAsBA,WAAtB;AACA,UAAM,KAAK5C,MAAL,CAAY+E,YAAZ,CAAyBF,OAAzB,EAAkCd,OAAlC,CAAN;AAGA,oBAAI,KAAK5D,YAAT,EAAuB,KAAK6E,kBAAL,CAAwB;AAACH,MAAAA,OAAD;AAAUd,MAAAA;AAAV,KAAxB,CAAvB;AAEA,WAAO,KAAK/D,MAAL,CAAYuB,mBAAnB;AACD;;AAcD,QAAMgD,aAAN,CAAqB;AAACvC,IAAAA,GAAD;AAAMD,IAAAA,OAAN;AAAe0C,IAAAA;AAAf,GAArB,EAA6C;AAC3C,UAAMQ,OAAO,GAAI,GAAElD,OAAQ,GAAE0C,MAAM,GAAG,MAAMA,MAAT,GAAkB,EAAG,EAAxD;AACA,UAAMS,OAAO,GAAGD,OAAO,KAAKjD,GAAZ,GAAkB,EAAlB,GAAwB,4BAA2BiD,OAAQ,GAA3E;AACA,UAAMjB,GAAG,GAAI,eAAchC,GAAI,IAAGkD,OAAQ,EAA1C;;AACA,QAAI;AACF,YAAMjB,WAAW,GAAG,MAAM,qBAAS,KAAK9D,YAAd,EAA4B6D,GAA5B,EAAiC,YACzD,MAAM,KAAK5D,GAAL,CAAS+E,cAAT,CAAwB;AAC5BC,QAAAA,MAAM,EAAEC,cAAKC,OAAL,CAAa,KAAKtF,MAAL,CAAYM,UAAzB,EAAqCyB,OAArC,CADoB;AAE5BA,QAAAA,OAF4B;AAG5B0C,QAAAA;AAH4B,OAAxB,CADkB,CAA1B;AAOA,YAAMV,OAAO,GAAG,KAAKI,kBAAL,CAAwBF,WAAxB,CAAhB;AACAF,MAAAA,OAAO,CAACK,WAAR,GAAsBrC,OAAtB;AACA,aAAOgC,OAAP;AACD,KAXD,CAWE,OAAOwB,GAAP,EAAY;AACZ,YAAM,IAAI3F,KAAJ,CAAW,iDAAgD2F,GAAG,CAACC,OAAQ,EAAvE,CAAN;AACD;AACF;;AAcDR,EAAAA,kBAAkB,GAA0B;AAC1C,UAAM,IAAIpF,KAAJ,CAAU,oCAAV,CAAN;AACD;;AAWDuE,EAAAA,kBAAkB,CAAEF,WAAF,EAAe;AAC/B,QAAI,CAACA,WAAW,CAACwB,MAAjB,EAAyB;AACvB,YAAM,IAAI7F,KAAJ,CAAW,2DAAD,GACC,+BADX,CAAN;AAED;;AACD,UAAM;AAAC6F,MAAAA,MAAD;AAAS7D,MAAAA,IAAT;AAAeiB,MAAAA;AAAf,QAA0BoB,WAAhC;AACA,SAAKyB,uBAAL,CAA6BD,MAA7B;AAEA,WAAO,EAAC,GAAGA,MAAJ;AAAY1D,MAAAA,OAAO,EAAEH,IAArB;AAA2BiB,MAAAA;AAA3B,KAAP;AACD;;AAUD6C,EAAAA,uBAAuB,GAAqB;AAC1C,UAAM,IAAI9F,KAAJ,CAAU,oCAAV,CAAN;AACD;;AAaD,QAAM+F,SAAN,CAAiB;AAAC3D,IAAAA;AAAD,GAAjB,EAAwB;AACtB,QAAI,CAAC,KAAKhC,MAAL,CAAY8E,WAAZ,CAAwB9C,GAAxB,CAAL,EAAmC;AACjC,YAAM,IAAIpC,KAAJ,CAAW,mBAAkB,KAAKM,IAAK,KAAI8B,GAAI,wBAA/C,CAAN;AACD;;AACD,QAAI;AACF,YAAM4D,kBAAGC,MAAH,CAAU,KAAK7F,MAAL,CAAY8F,cAAZ,CAA2B9D,GAA3B,CAAV,CAAN;AACD,KAFD,SAEU;AACR,YAAM,KAAKhC,MAAL,CAAY+F,eAAZ,CAA4B/D,GAA5B,CAAN;AACD;;AACD,oBAAI,KAAK7B,YAAT,EAAwB,4BAA2B,KAAKD,IAAK,KAAI8B,GAAI,GAA9C,CAAiDqB,KAAxE;AACA,WAAO,KAAKrD,MAAL,CAAYuB,mBAAnB;AACD;;AA2BD,QAAMyE,MAAN,CAAc;AAAChE,IAAAA,GAAD;AAAMiE,IAAAA;AAAN,GAAd,EAA6B;AAC3B,UAAMC,eAAe,GAAGlE,GAAG,KAAKtC,UAAhC;;AAEA,QAAI,CAACwG,eAAD,IAAoB,CAAC,KAAKlG,MAAL,CAAY8E,WAAZ,CAAwB9C,GAAxB,CAAzB,EAAuD;AACrD,YAAM,IAAIpC,KAAJ,CAAW,OAAM,KAAKM,IAAK,KAAI8B,GAAI,0CAAnC,CAAN;AACD;;AACD,UAAMmE,YAAY,GAAGD,eAAe,GAAG7E,MAAM,CAACC,IAAP,CAAY,KAAKtB,MAAL,CAAYuB,mBAAxB,CAAH,GAAkD,CAACS,GAAD,CAAtF;AAGA,UAAMoE,MAAM,GAAG,EAAf;AAIA,UAAM/D,OAAO,GAAG,EAAhB;;AAEA,SAAK,MAAMgE,CAAX,IAAgBF,YAAhB,EAA8B;AAC5B,UAAI;AACF,cAAM,qBAAS,KAAKhG,YAAd,EAA6B,eAAc,KAAKD,IAAK,KAAImG,CAAE,gBAA3D,EAA4E,MAAM;AACtF,cAAI,KAAKrG,MAAL,CAAYuB,mBAAZ,CAAgC8E,CAAhC,EAAmClE,WAAnC,KAAmDC,iCAAvD,EAAyE;AACvE,kBAAM,IAAIzC,iBAAJ,EAAN;AACD;AACF,SAJK,CAAN;AAKA,cAAMqG,MAAM,GAAG,MAAM,qBAAS,KAAK7F,YAAd,EAA6B,eAAc,KAAKD,IAAK,KAAImG,CAAE,mBAA3D,EAA+E,YAAY;AAC9G,gBAAML,MAAM,GAAG,MAAM,KAAK1D,uBAAL,CAA6B+D,CAA7B,CAArB;;AACA,cAAI,EAAEL,MAAM,CAACxD,UAAP,IAAqBwD,MAAM,CAACtD,YAA9B,CAAJ,EAAiD;AAC/C,kBAAM,IAAI7C,uBAAJ,EAAN;AACD;;AACD,iBAAOmG,MAAP;AACD,SANoB,CAArB;;AAOA,YAAI,CAACC,MAAD,IAAW,CAACD,MAAM,CAACxD,UAAvB,EAAmC;AACjC,gBAAM,IAAI5C,KAAJ,CAAW,OAAM,KAAKM,IAAK,KAAImG,CAAE,gCAAvB,GACC,IAAGL,MAAM,CAACM,OAAQ,OAAMN,MAAM,CAACtD,YAAa,yBAD7C,GAEC,0EAFX,CAAN;AAGD;;AACD,cAAM6D,SAAS,GAAGN,MAAM,IAAID,MAAM,CAACtD,YAAjB,GAAgCsD,MAAM,CAACtD,YAAvC,GAAsDsD,MAAM,CAACxD,UAA/E;AACA,cAAM,qBACJ,KAAKrC,YADD,EAEH,oBAAmBkG,CAAE,UAASL,MAAM,CAACM,OAAQ,OAAMC,SAAU,EAF1D,EAGJ,YAAY,MAAM,KAAKC,eAAL,CAAqBH,CAArB,EAAwBE,SAAxB,CAHd,CAAN;AAKAlE,QAAAA,OAAO,CAACgE,CAAD,CAAP,GAAa;AAACI,UAAAA,IAAI,EAAET,MAAM,CAACM,OAAd;AAAuBI,UAAAA,EAAE,EAAEH;AAA3B,SAAb;AACD,OAzBD,CAyBE,OAAOhB,GAAP,EAAY;AACZa,QAAAA,MAAM,CAACC,CAAD,CAAN,GAAYd,GAAZ;AACD;AACF;;AAED,oBAAI,KAAKpF,YAAT,EAAuB,gBAAvB;;AACA,SAAK,MAAM,CAACkG,CAAD,EAAIL,MAAJ,CAAX,IAA0BrF,gBAAEuB,OAAF,CAAUG,OAAV,CAA1B,EAA8C;AAC5C,sBAAI,KAAKlC,YAAT,EAAwB,KAAI,KAAKD,IAAK,IAAGmG,CAAE,aAAYL,MAAM,CAACS,IAAK,OAAMT,MAAM,CAACU,EAAG,EAA5D,CAA8DrD,KAArF;AACD;;AACD,SAAK,MAAM,CAACgD,CAAD,EAAId,GAAJ,CAAX,IAAuB5E,gBAAEuB,OAAF,CAAUkE,MAAV,CAAvB,EAA0C;AACxC,UAAIb,GAAG,YAAY5F,iBAAnB,EAAsC;AACpC,wBAAI,KAAKQ,YAAT,EAAwB,MAAKkG,CAAE,qDAAR,GACC,aAAD,CAAcpD,MADrC;AAED,OAHD,MAGO,IAAIsC,GAAG,YAAY1F,uBAAnB,EAA4C;AACjD,wBAAI,KAAKM,YAAT,EAAwB,MAAKkG,CAAE,4BAAR,CAAoCpD,MAA3D;AACD,OAFM,MAEA;AAEL,wBAAI,KAAK9C,YAAT,EAAwB,MAAKkG,CAAE,uBAAsBd,GAAI,EAAlC,CAAoCoB,GAA3D;AACD;AACF;;AAED,WAAO;AAACtE,MAAAA,OAAD;AAAU+D,MAAAA;AAAV,KAAP;AACD;;AAgBD,QAAM9D,uBAAN,CAA+BN,GAA/B,EAAoC;AAIlC,UAAM;AAACa,MAAAA,OAAD;AAAUd,MAAAA;AAAV,QAAqB,KAAK/B,MAAL,CAAYuB,mBAAZ,CAAgCS,GAAhC,CAA3B;AACA,QAAIU,YAAY,GAAG,MAAM,KAAKtC,GAAL,CAASwG,gBAAT,CAA0B7E,OAA1B,CAAzB;AACA,QAAIS,UAAU,GAAG,MAAM,KAAKpC,GAAL,CAASyG,2BAAT,CAAqC9E,OAArC,EAA8Cc,OAA9C,CAAvB;;AACA,QAAI,CAACiE,oBAAKC,eAAL,CAAqBrE,YAArB,EAAmC,GAAnC,EAAwCG,OAAxC,CAAL,EAAuD;AAErDH,MAAAA,YAAY,GAAG,IAAf;AACAF,MAAAA,UAAU,GAAG,IAAb;AACD;;AACD,QAAIE,YAAY,IAAIA,YAAY,KAAKF,UAArC,EAAiD;AAE/CE,MAAAA,YAAY,GAAG,IAAf;AACD;;AACD,QAAIF,UAAU,IAAI,CAACsE,oBAAKC,eAAL,CAAqBvE,UAArB,EAAiC,GAAjC,EAAsCK,OAAtC,CAAnB,EAAmE;AAEjEL,MAAAA,UAAU,GAAG,IAAb;AACD;;AACD,WAAO;AAAC8D,MAAAA,OAAO,EAAEzD,OAAV;AAAmBL,MAAAA,UAAnB;AAA+BE,MAAAA;AAA/B,KAAP;AACD;;AASD,QAAM8D,eAAN,CAAuBxE,GAAvB,EAA4Ba,OAA5B,EAAqC;AACnC,UAAM;AAACd,MAAAA;AAAD,QAAY,KAAK/B,MAAL,CAAYuB,mBAAZ,CAAgCS,GAAhC,CAAlB;AACA,UAAM,KAAKuC,aAAL,CAAmB;AAACvC,MAAAA,GAAD;AAAMD,MAAAA,OAAN;AAAe0C,MAAAA,MAAM,EAAE5B;AAAvB,KAAnB,CAAN;AACA,SAAK7C,MAAL,CAAYuB,mBAAZ,CAAgCS,GAAhC,EAAqCa,OAArC,GAA+CA,OAA/C;AACA,UAAM,KAAK7C,MAAL,CAAYgH,KAAZ,EAAN;AACD;;AAxdmC","sourcesContent":["/* eslint-disable no-console */\n\nimport _ from 'lodash';\nimport NPM from './npm';\nimport path from 'path';\nimport { fs, util } from 'appium-support';\nimport { log, spinWith } from './utils';\nimport { INSTALL_TYPE_NPM, INSTALL_TYPE_GIT, INSTALL_TYPE_GITHUB,\n         INSTALL_TYPE_LOCAL } from '../extension-config';\n\nconst UPDATE_ALL = 'installed';\n\nclass NotUpdatableError extends Error {}\nclass NoUpdatesAvailableError extends Error {}\n\nexport default class ExtensionCommand {\n\n  /**\n   * @typedef {Object} ExtensionCommandConstructor\n   * @property {Object} config - the DriverConfig or PluginConfig object used for this command\n   * @property {boolean} json - whether the output of this command should be JSON or text\n   * @property {string} type - DRIVER_TYPE or PLUGIN_TYPE\n   */\n\n  /**\n   * Build an ExtensionCommand\n   *\n   * @param {ExtensionCommandConstructor} opts\n   * @return {ExtensionCommand}\n   */\n  constructor ({config, json, type}) {\n    this.config = config;\n    this.type = type;\n    this.isJsonOutput = json;\n    this.npm = new NPM(this.config.appiumHome);\n    this.knownExtensions = {}; // this needs to be overridden in final class\n  }\n\n  /**\n   * Take a CLI parse and run an extension command based on its type\n   *\n   * @param {object} args - a key/value object with CLI flags and values\n   * @return {object} the result of the specific command which is executed\n   */\n  async execute (args) {\n    const cmd = args[`${this.type}Command`];\n    if (!_.isFunction(ExtensionCommand.prototype[cmd])) {\n      throw new Error(`Cannot handle ${this.type} command ${cmd}`);\n    }\n    const executeCmd = this[cmd].bind(this);\n    return await executeCmd(args);\n  }\n\n  /**\n   * @typedef {Object} ListArgs\n   * @property {boolean} showInstalled - whether should show only installed extensions\n   * @property {boolean} showUpdates - whether should show available updates\n   */\n\n  /**\n   * List extensions\n   *\n   * @param {ListArgs} args\n   * @return {object} map of extension names to extension data\n   */\n  async list ({showInstalled, showUpdates}) {\n    const lsMsg = `Listing ${showInstalled ? 'installed' : 'available'} ${this.type}s`;\n    const installedNames = Object.keys(this.config.installedExtensions);\n    const knownNames = Object.keys(this.knownExtensions);\n    const exts = [...installedNames, ...knownNames].reduce((acc, name) => {\n      if (!acc[name]) {\n        if (installedNames.includes(name)) {\n          acc[name] = {...this.config.installedExtensions[name], installed: true};\n        } else if (!showInstalled) {\n          acc[name] = {pkgName: this.knownExtensions[name], installed: false};\n        }\n      }\n      return acc;\n    }, {});\n\n    // if we want to show whether updates are available, put that behind a spinner\n    await spinWith(this.isJsonOutput, lsMsg, async () => {\n      if (!showUpdates) {\n        return;\n      }\n      for (const [ext, data] of _.toPairs(exts)) {\n        const {installed, installType} = data;\n        if (!installed || installType !== INSTALL_TYPE_NPM) {\n          // don't need to check for updates on exts that aren't installed\n          // also don't need to check for updates on non-npm exts\n          continue;\n        }\n        const updates = await this.checkForExtensionUpdate(ext);\n        data.updateVersion = updates.safeUpdate;\n        data.unsafeUpdateVersion = updates.unsafeUpdate;\n        data.upToDate = updates.safeUpdate === null && updates.unsafeUpdate === null;\n      }\n    });\n\n    // if we're just getting the data, short circuit return here since we don't need to do any\n    // formatting logic\n    if (this.isJsonOutput) {\n      return exts;\n    }\n\n    for (const [\n      name,\n      {installType, installSpec, installed, updateVersion, unsafeUpdateVersion, version, upToDate}\n    ] of _.toPairs(exts)) {\n      let typeTxt;\n      switch (installType) {\n        case INSTALL_TYPE_GIT:\n        case INSTALL_TYPE_GITHUB:\n          typeTxt = `(cloned from ${installSpec})`.yellow;\n          break;\n        case INSTALL_TYPE_LOCAL:\n          typeTxt = `(linked from ${installSpec})`.magenta;\n          break;\n        default:\n          typeTxt = '(NPM)';\n      }\n      const installTxt = installed ?\n        `@${version.yellow} ${('[installed ' + typeTxt + ']').green}` :\n        ' [not installed]'.grey;\n      const updateTxt = showUpdates && updateVersion ?\n        ` [${updateVersion} available]`.magenta :\n        '';\n      const upToDateTxt = showUpdates && upToDate ?\n        ` [Up to date]`.green :\n        '';\n      const unsafeUpdateTxt = showUpdates && unsafeUpdateVersion ?\n        ` [${unsafeUpdateVersion} available (potentially unsafe)]`.cyan :\n        '';\n\n      console.log(`- ${name.yellow}${installTxt}${updateTxt}${upToDateTxt}${unsafeUpdateTxt}`);\n    }\n\n    return exts;\n  }\n\n  /**\n   * @typedef {Object} InstallArgs\n   * @property {string} ext - the name or spec of an extension to install\n   * @property {string} installType - how to install this extension. One of the INSTALL_TYPES\n   * @property {string} [packageName] - for git/github installs, the extension node package name\n   */\n\n  /**\n   * Install an extension\n   *\n   * @param {InstallArgs} args\n   * @return {object} map of all installed extension names to extension data\n   */\n  async install ({ext, installType, packageName}) {\n    log(this.isJsonOutput, `Attempting to find and install ${this.type} '${ext}'`);\n\n    let extData;\n    let installSpec = ext;\n\n    if (packageName && [INSTALL_TYPE_LOCAL, INSTALL_TYPE_NPM].includes(installType)) {\n      throw new Error(`When using --source=${installType}, cannot also use --package`);\n    }\n\n    if (!packageName && [INSTALL_TYPE_GIT, INSTALL_TYPE_GITHUB].includes(installType)) {\n      throw new Error(`When using --source=${installType}, must also use --package`);\n    }\n\n    if (installType === INSTALL_TYPE_LOCAL) {\n      const msg = `Linking ${this.type} from local path`;\n      const pkgJsonData = await spinWith(this.isJsonOutput, msg, async () => (\n        await this.npm.linkPackage(installSpec))\n      );\n      extData = this.getExtensionFields(pkgJsonData);\n      extData.installPath = extData.pkgName;\n    } else if (installType === INSTALL_TYPE_GITHUB) {\n      if (installSpec.split('/').length !== 2) {\n        throw new Error(`Github ${this.type} spec ${installSpec} appeared to be invalid; ` +\n                        'it should be of the form <org>/<repo>');\n      }\n      extData = await this.installViaNpm({ext: installSpec, pkgName: packageName});\n    } else if (installType === INSTALL_TYPE_GIT) {\n      // git urls can have '.git' at the end, but this is not necessary and would complicate the\n      // way we download and name directories, so we can just remove it\n      installSpec = installSpec.replace(/\\.git$/, '');\n      extData = await this.installViaNpm({ext: installSpec, pkgName: packageName});\n    } else {\n      // at this point we have either an npm package or an appium verified extension\n      // name. both of which will be installed via npm.\n      // extensions installed via npm can include versions or tags after the '@'\n      // sign, so check for that. We also need to be careful that package names themselves can\n      // contain the '@' symbol, as in `npm install @appium/fake-driver@1.2.0`\n      let name, pkgVer;\n      const splits = installSpec.split('@');\n      if (installSpec[0] === '@') {\n        // this is the case where we have an npm org included in the package name\n        [name, pkgVer] = [`@${splits[1]}`, splits[2]];\n      } else {\n        // this is the case without an npm org\n        [name, pkgVer] = splits;\n      }\n      let pkgName;\n\n      if (installType === INSTALL_TYPE_NPM) {\n        // if we're installing a named package from npm, we don't need to check\n        // against the appium extension list; just use the installSpec as is\n        pkgName = name;\n      } else {\n        // if we're installing a named appium driver (like 'xcuitest') we need to\n        // dereference the actual npm package ('appiupm-xcuitest-driver'), so\n        // check it exists and get the correct package\n        const knownNames = Object.keys(this.knownExtensions);\n        if (!_.includes(knownNames, name)) {\n          const msg = `Could not resolve ${this.type}; are you sure it's in the list ` +\n                      `of supported ${this.type}s? ${JSON.stringify(knownNames)}`;\n          throw new Error(msg);\n        }\n        pkgName = this.knownExtensions[name];\n        // given that we'll use the install type in the driver json, store it as\n        // 'npm' now\n        installType = INSTALL_TYPE_NPM;\n      }\n\n      extData = await this.installViaNpm({ext, pkgName, pkgVer});\n    }\n\n    const extName = extData[`${this.type}Name`];\n    delete extData[`${this.type}Name`];\n\n    if (this.config.isInstalled(extName)) {\n      throw new Error(`A ${this.type} named '${extName}' is already installed. ` +\n                      `Did you mean to update? 'appium ${this.type} update'. See ` +\n                      `installed ${this.type}s with 'appium ${this.type} list --installed'.`);\n    }\n\n    extData.installType = installType;\n    extData.installSpec = installSpec;\n    await this.config.addExtension(extName, extData);\n\n    // log info for the user\n    log(this.isJsonOutput, this.getPostInstallText({extName, extData}));\n\n    return this.config.installedExtensions;\n  }\n\n  /**\n   * @typedef {Object} InstallViaNpmArgs\n   * @property {string} ext - the name or spec of an extension to install\n   * @property {string} pkgName - the NPM package name of the extension\n   * @property {string} [pkgVer] - the specific version of the NPM package\n   */\n\n  /**\n   * Install an extension via NPM\n   *\n   * @param {InstallViaNpmArgs} args\n   */\n  async installViaNpm ({ext, pkgName, pkgVer}) {\n    const npmSpec = `${pkgName}${pkgVer ? '@' + pkgVer : ''}`;\n    const specMsg = npmSpec === ext ? '' : ` using NPM install spec '${npmSpec}'`;\n    const msg = `Installing '${ext}'${specMsg}`;\n    try {\n      const pkgJsonData = await spinWith(this.isJsonOutput, msg, async () => (\n        await this.npm.installPackage({\n          pkgDir: path.resolve(this.config.appiumHome, pkgName),\n          pkgName,\n          pkgVer\n        })\n      ));\n      const extData = this.getExtensionFields(pkgJsonData);\n      extData.installPath = pkgName;\n      return extData;\n    } catch (err) {\n      throw new Error(`Encountered an error when installing package: ${err.message}`);\n    }\n  }\n\n  /**\n   * @typedef {Object} ExtensionArgs\n   * @property {string} extName - the name of an extension\n   * @property {object} extData - the data for an installed extension\n   */\n\n  /**\n   * Get the text which should be displayed to the user after an extension has been installed. This\n   * is designed to be overridden by drivers/plugins with their own particular text.\n   *\n   * @param {ExtensionArgs} args\n   */\n  getPostInstallText (/*{extName, extData}*/) {\n    throw new Error('Must be implemented in final class');\n  }\n\n  /**\n   * Take an NPM module's package.json and extract Appium driver information from a special\n   * 'appium' field in the JSON data. We need this information to e.g. determine which class to\n   * load as the main driver class, or to be able to detect incompatibilities between driver and\n   * appium versions.\n   *\n   * @param {object} pkgJsonData - the package.json data for a driver module, as if it had been\n   * straightforwardly 'require'd\n   */\n  getExtensionFields (pkgJsonData) {\n    if (!pkgJsonData.appium) {\n      throw new Error(`Installed driver did not have an 'appium' section in its ` +\n                      `package.json file as expected`);\n    }\n    const {appium, name, version} = pkgJsonData;\n    this.validateExtensionFields(appium);\n\n    return {...appium, pkgName: name, version};\n  }\n\n  /**\n   * For any package.json fields which a particular type of extension requires, validate the\n   * presence and form of those fields on the package.json data, throwing an error if anything is\n   * amiss.\n   *\n   * @param {object} appiumPkgData - the data in the \"appium\" field of package.json for an\n   * extension\n   */\n  validateExtensionFields (/*appiumPkgData*/) {\n    throw new Error('Must be implemented in final class');\n  }\n\n  /**\n   * @typedef {Object} UninstallArgs\n   * @property {string} ext - the name or spec of an extension to uninstall\n   */\n\n  /**\n   * Uninstall an extension\n   *\n   * @param {UninstallArgs} args\n   * @return {object} map of all installed extension names to extension data\n   */\n  async uninstall ({ext}) {\n    if (!this.config.isInstalled(ext)) {\n      throw new Error(`Can't uninstall ${this.type} '${ext}'; it is not installed`);\n    }\n    try {\n      await fs.rimraf(this.config.getInstallPath(ext));\n    } finally {\n      await this.config.removeExtension(ext);\n    }\n    log(this.isJsonOutput, `Successfully uninstalled ${this.type} '${ext}'`.green);\n    return this.config.installedExtensions;\n  }\n\n  /**\n   * @typedef {Object} ExtensionUpdateOpts\n   * @property {string} ext - the name of the extension to update\n   * @property {boolean} unsafe - if true, will perform unsafe updates past major revision\n   * boundaries\n   */\n\n  /**\n   * @typedef {Object} UpdateReport\n   * @property {string} from - version updated from\n   * @property {string} to - version updated to\n   */\n\n  /**\n   * @typedef {Object} ExtensionUpdateResult\n   * @property {Object} errors - map of ext names to error objects\n   * @property {Object} updates - map of ext names to {@link UpdateReport}s\n   */\n\n  /**\n   * Attempt to update one or more drivers using NPM\n   *\n   * @param {ExtensionUpdateOpts} updateSpec\n   * @return {ExtensionUpdateResult}\n   */\n  async update ({ext, unsafe}) {\n    const shouldUpdateAll = ext === UPDATE_ALL;\n    // if we're specifically requesting an update for an extension, make sure it's installed\n    if (!shouldUpdateAll && !this.config.isInstalled(ext)) {\n      throw new Error(`The ${this.type} '${ext}' was not installed, so can't be updated`);\n    }\n    const extsToUpdate = shouldUpdateAll ? Object.keys(this.config.installedExtensions) : [ext];\n\n    // 'errors' will have ext names as keys and error objects as values\n    const errors = {};\n\n    // 'updates' will have ext names as keys and update objects as values, where an update\n    // object is of the form {from: versionString, to: versionString}\n    const updates = {};\n\n    for (const e of extsToUpdate) {\n      try {\n        await spinWith(this.isJsonOutput, `Checking if ${this.type} '${e}' is updatable`, () => {\n          if (this.config.installedExtensions[e].installType !== INSTALL_TYPE_NPM) {\n            throw new NotUpdatableError();\n          }\n        });\n        const update = await spinWith(this.isJsonOutput, `Checking if ${this.type} '${e}' needs an update`, async () => {\n          const update = await this.checkForExtensionUpdate(e);\n          if (!(update.safeUpdate || update.unsafeUpdate)) {\n            throw new NoUpdatesAvailableError();\n          }\n          return update;\n        });\n        if (!unsafe && !update.safeUpdate) {\n          throw new Error(`The ${this.type} '${e}' has a major revision update ` +\n                          `(${update.current} => ${update.unsafeUpdate}), which could include ` +\n                          `breaking changes. If you want to apply this update, re-run with --unsafe`);\n        }\n        const updateVer = unsafe && update.unsafeUpdate ? update.unsafeUpdate : update.safeUpdate;\n        await spinWith(\n          this.isJsonOutput,\n          `Updating driver '${e}' from ${update.current} to ${updateVer}`,\n          async () => await this.updateExtension(e, updateVer)\n        );\n        updates[e] = {from: update.current, to: updateVer};\n      } catch (err) {\n        errors[e] = err;\n      }\n    }\n\n    log(this.isJsonOutput, 'Update report:');\n    for (const [e, update] of _.toPairs(updates)) {\n      log(this.isJsonOutput, `- ${this.type} ${e} updated: ${update.from} => ${update.to}`.green);\n    }\n    for (const [e, err] of _.toPairs(errors)) {\n      if (err instanceof NotUpdatableError) {\n        log(this.isJsonOutput, `- '${e}' was not installed via npm, so we could not check ` +\n                               `for updates`.yellow);\n      } else if (err instanceof NoUpdatesAvailableError) {\n        log(this.isJsonOutput, `- '${e}' had no updates available`.yellow);\n      } else {\n        // otherwise, make it pop with red!\n        log(this.isJsonOutput, `- '${e}' failed to update: ${err}`.red);\n      }\n    }\n\n    return {updates, errors};\n  }\n\n  /**\n   * @typedef PossibleUpdates\n   * @property {string} current - current version\n   * @property {string|null} safeUpdate - version we can safely update to if it exists, or null\n   * @property {string|null} unsafeUpdate - version we can unsafely update to if it exists, or null\n   */\n\n  /**\n   * Given an extension name, figure out what its highest possible version upgrade is, and also the\n   * highest possible safe upgrade.\n   *\n   * @param {string} ext - name of extension\n   * @return {PossibleUpdates}\n   */\n  async checkForExtensionUpdate (ext) {\n    // TODO decide how we want to handle beta versions?\n    // this is a helper method, 'ext' is assumed to already be installed here, and of the npm\n    // install type\n    const {version, pkgName} = this.config.installedExtensions[ext];\n    let unsafeUpdate = await this.npm.getLatestVersion(pkgName);\n    let safeUpdate = await this.npm.getLatestSafeUpgradeVersion(pkgName, version);\n    if (!util.compareVersions(unsafeUpdate, '>', version)) {\n      // the latest version is not greater than the current version, so there's no possible update\n      unsafeUpdate = null;\n      safeUpdate = null;\n    }\n    if (unsafeUpdate && unsafeUpdate === safeUpdate) {\n      // the latest update is the same as the safe update, which means it's not actually unsafe\n      unsafeUpdate = null;\n    }\n    if (safeUpdate && !util.compareVersions(safeUpdate, '>', version)) {\n      // even the safe update is not later than the current, so it is not actually an update\n      safeUpdate = null;\n    }\n    return {current: version, safeUpdate, unsafeUpdate};\n  }\n\n  /**\n   * Actually update an extension installed by NPM, using the NPM cli. And update the installation\n   * manifest.\n   *\n   * @param {string} ext - name of extension to update\n   * @param {string} version - version string identifier to update extension to\n   */\n  async updateExtension (ext, version) {\n    const {pkgName} = this.config.installedExtensions[ext];\n    await this.installViaNpm({ext, pkgName, pkgVer: version});\n    this.config.installedExtensions[ext].version = version;\n    await this.config.write();\n  }\n}\n"],"file":"lib/cli/extension-command.js","sourceRoot":"../../.."}
531
+ exports.ExtensionCommand = ExtensionCommand;
532
+ var _default = ExtensionCommand;
533
+ exports.default = _default;
534
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["UPDATE_ALL","NotUpdatableError","Error","NoUpdatesAvailableError","ExtensionCommand","config","knownExtensions","isJsonOutput","constructor","json","log","console","CliConsole","jsonMode","Boolean","type","extensionType","_createFatalError","message","decorate","execute","args","cmd","_","isFunction","executeCmd","bind","list","showInstalled","showUpdates","lsMsg","installedNames","Object","keys","installedExtensions","knownNames","exts","reduce","acc","name","includes","installed","pkgName","spinWith","ext","data","toPairs","installType","INSTALL_TYPE_NPM","updates","checkForExtensionUpdate","updateVersion","safeUpdate","unsafeUpdateVersion","unsafeUpdate","upToDate","listData","installTxt","grey","updateTxt","upToDateTxt","unsafeUpdateTxt","installSpec","version","typeTxt","INSTALL_TYPE_GIT","INSTALL_TYPE_GITHUB","yellow","INSTALL_TYPE_LOCAL","magenta","green","cyan","_install","packageName","extData","installOpts","probableExtName","split","length","replace","pkgVer","path","isAbsolute","resolve","splits","msg","JSON","stringify","isInstalled","installViaNpm","extName","extManifest","errors","warnings","B","all","getProblems","getWarnings","errorMap","Map","warningMap","errorSummaries","warningSummaries","getValidationResultSummaries","isEmpty","join","warn","addExtension","env","hasAppiumDependency","appiumHome","packageDidChange","info","getPostInstallText","npmSpec","specMsg","pkgJsonData","npm","installPackage","validatePackageJson","getExtensionFields","err","pkgJson","appium","peerDependencies","result","appiumVersion","createMissingFieldError","field","ReferenceError","validateExtensionFields","extMetadata","_uninstall","uninstallPackage","removeExtension","ok","_update","unsafe","shouldUpdateAll","extsToUpdate","e","update","current","updateVer","updateExtension","from","to","error","red","getLatestVersion","getLatestSafeUpgradeVersion","util","compareVersions","_run","scriptName","extConfig","scripts","extScripts","isPlainObject","has","runner","SubProcess","process","execPath","cwd","getInstallPath","output","RingBuffer","on","line","enqueue","start","getBuff"],"sources":["../../../lib/cli/extension-command.js"],"sourcesContent":["/* eslint-disable no-console */\nimport B from 'bluebird';\nimport _ from 'lodash';\nimport path from 'path';\nimport {npm, util, env, console} from '@appium/support';\nimport {spinWith, RingBuffer} from './utils';\nimport {SubProcess} from 'teen_process';\nimport {\n  INSTALL_TYPE_NPM,\n  INSTALL_TYPE_GIT,\n  INSTALL_TYPE_GITHUB,\n  INSTALL_TYPE_LOCAL,\n} from '../extension/extension-config';\nimport {packageDidChange} from '../extension/package-changed';\n\nconst UPDATE_ALL = 'installed';\n\nclass NotUpdatableError extends Error {}\nclass NoUpdatesAvailableError extends Error {}\n\n/**\n * @template {ExtensionType} ExtType\n */\nclass ExtensionCommand {\n  /**\n   * This is the `DriverConfig` or `PluginConfig`, depending on `ExtType`.\n   * @type {ExtensionConfig<ExtType>}\n   */\n  config;\n\n  /**\n   * {@linkcode Record} of official plugins or drivers.\n   * @type {KnownExtensions<ExtType>}\n   */\n  knownExtensions;\n\n  /**\n   * If `true`, command output has been requested as JSON.\n   * @type {boolean}\n   */\n  isJsonOutput;\n\n  /**\n   * Build an ExtensionCommand\n   * @param {ExtensionCommandOptions<ExtType>} opts\n   */\n  constructor({config, json}) {\n    this.config = config;\n    this.log = new console.CliConsole({jsonMode: json});\n    this.isJsonOutput = Boolean(json);\n  }\n\n  /**\n   * `driver` or `plugin`, depending on the `ExtensionConfig`.\n   */\n  get type() {\n    return this.config.extensionType;\n  }\n\n  /**\n   * Logs a message and returns an {@linkcode Error} to throw.\n   *\n   * For TS to understand that a function throws an exception, it must actually throw an exception--\n   * in other words, _calling_ a function which is guaranteed to throw an exception is not enough--\n   * nor is something like `@returns {never}` which does not imply a thrown exception.\n   * @param {string} message\n   * @protected\n   * @returns {Error}\n   */\n  _createFatalError(message) {\n    return new Error(this.log.decorate(message, 'error'));\n  }\n\n  /**\n   * Take a CLI parse and run an extension command based on its type\n   *\n   * @param {object} args - a key/value object with CLI flags and values\n   * @return {Promise<object>} the result of the specific command which is executed\n   */\n  async execute(args) {\n    const cmd = args[`${this.type}Command`];\n    if (!_.isFunction(this[cmd])) {\n      throw this._createFatalError(`Cannot handle ${this.type} command ${cmd}`);\n    }\n    const executeCmd = this[cmd].bind(this);\n    return await executeCmd(args);\n  }\n\n  /**\n   * List extensions\n   *\n   * @param {ListOptions} opts\n   * @return {Promise<ExtensionListData>} map of extension names to extension data\n   */\n  async list({showInstalled, showUpdates}) {\n    const lsMsg = `Listing ${showInstalled ? 'installed' : 'available'} ${this.type}s`;\n    const installedNames = Object.keys(this.config.installedExtensions);\n    const knownNames = Object.keys(this.knownExtensions);\n    const exts = [...installedNames, ...knownNames].reduce(\n      (acc, name) => {\n        if (!acc[name]) {\n          if (installedNames.includes(name)) {\n            acc[name] = {\n              ...this.config.installedExtensions[name],\n              installed: true,\n            };\n          } else if (!showInstalled) {\n            acc[name] = {pkgName: this.knownExtensions[name], installed: false};\n          }\n        }\n        return acc;\n      },\n      /**\n       * This accumulator contains either {@linkcode UninstalledExtensionLIstData} _or_\n       * {@linkcode InstalledExtensionListData} without upgrade information (which is added by the below code block)\n       * @type {Record<string,Partial<InstalledExtensionListData>|UninstalledExtensionListData>}\n       */ ({})\n    );\n\n    // if we want to show whether updates are available, put that behind a spinner\n    await spinWith(this.isJsonOutput, lsMsg, async () => {\n      if (!showUpdates) {\n        return;\n      }\n      for (const [ext, data] of _.toPairs(exts)) {\n        if (!data.installed || data.installType !== INSTALL_TYPE_NPM) {\n          // don't need to check for updates on exts that aren't installed\n          // also don't need to check for updates on non-npm exts\n          continue;\n        }\n        const updates = await this.checkForExtensionUpdate(ext);\n        data.updateVersion = updates.safeUpdate;\n        data.unsafeUpdateVersion = updates.unsafeUpdate;\n        data.upToDate = updates.safeUpdate === null && updates.unsafeUpdate === null;\n      }\n    });\n\n    const listData = /** @type {ExtensionListData} */ (exts);\n\n    // if we're just getting the data, short circuit return here since we don't need to do any\n    // formatting logic\n    if (this.isJsonOutput) {\n      return listData;\n    }\n\n    for (const [name, data] of _.toPairs(listData)) {\n      let installTxt = ' [not installed]'.grey;\n      let updateTxt = '';\n      let upToDateTxt = '';\n      let unsafeUpdateTxt = '';\n      if (data.installed) {\n        const {installType, installSpec, updateVersion, unsafeUpdateVersion, version, upToDate} =\n          data;\n        let typeTxt;\n        switch (installType) {\n          case INSTALL_TYPE_GIT:\n          case INSTALL_TYPE_GITHUB:\n            typeTxt = `(cloned from ${installSpec})`.yellow;\n            break;\n          case INSTALL_TYPE_LOCAL:\n            typeTxt = `(linked from ${installSpec})`.magenta;\n            break;\n          default:\n            typeTxt = '(NPM)';\n        }\n        installTxt = `@${version.yellow} ${('[installed ' + typeTxt + ']').green}`;\n\n        if (showUpdates) {\n          if (updateVersion) {\n            updateTxt = ` [${updateVersion} available]`.magenta;\n          }\n          if (upToDate) {\n            upToDateTxt = ` [Up to date]`.green;\n          }\n          if (unsafeUpdateVersion) {\n            unsafeUpdateTxt = ` [${unsafeUpdateVersion} available (potentially unsafe)]`.cyan;\n          }\n        }\n      }\n\n      this.log.log(`- ${name.yellow}${installTxt}${updateTxt}${upToDateTxt}${unsafeUpdateTxt}`);\n    }\n\n    return listData;\n  }\n\n  /**\n   * Install an extension\n   *\n   * @param {InstallArgs} args\n   * @return {Promise<ExtRecord<ExtType>>} map of all installed extension names to extension data\n   */\n  async _install({installSpec, installType, packageName}) {\n    /** @type {ExtensionFields<ExtType>} */\n    let extData;\n\n    if (packageName && [INSTALL_TYPE_LOCAL, INSTALL_TYPE_NPM].includes(installType)) {\n      throw this._createFatalError(`When using --source=${installType}, cannot also use --package`);\n    }\n\n    if (!packageName && [INSTALL_TYPE_GIT, INSTALL_TYPE_GITHUB].includes(installType)) {\n      throw this._createFatalError(`When using --source=${installType}, must also use --package`);\n    }\n\n    /**\n     * @type {InstallViaNpmArgs}\n     */\n    let installOpts;\n\n    /**\n     * The probable (?) name of the extension derived from the install spec.\n     *\n     * If using a local install type, this will remain empty.\n     * @type {string}\n     */\n    let probableExtName = '';\n\n    // depending on `installType`, build the options to pass into `installViaNpm`\n    if (installType === INSTALL_TYPE_GITHUB) {\n      if (installSpec.split('/').length !== 2) {\n        throw this._createFatalError(\n          `Github ${this.type} spec ${installSpec} appeared to be invalid; ` +\n            'it should be of the form <org>/<repo>'\n        );\n      }\n      installOpts = {\n        installSpec,\n        pkgName: /** @type {string} */ (packageName),\n      };\n      probableExtName = installSpec;\n    } else if (installType === INSTALL_TYPE_GIT) {\n      // git urls can have '.git' at the end, but this is not necessary and would complicate the\n      // way we download and name directories, so we can just remove it\n      installSpec = installSpec.replace(/\\.git$/, '');\n      installOpts = {\n        installSpec,\n        pkgName: /** @type {string} */ (packageName),\n      };\n      probableExtName = installSpec;\n    } else {\n      let pkgName, pkgVer;\n      if (installType === INSTALL_TYPE_LOCAL) {\n        pkgName = path.isAbsolute(installSpec) ? installSpec : path.resolve(installSpec);\n      } else {\n        // at this point we have either an npm package or an appium verified extension\n        // name or a local path. both of which will be installed via npm.\n        // extensions installed via npm can include versions or tags after the '@'\n        // sign, so check for that. We also need to be careful that package names themselves can\n        // contain the '@' symbol, as in `npm install @appium/fake-driver@1.2.0`\n        let name;\n        const splits = installSpec.split('@');\n        if (installSpec[0] === '@') {\n          // this is the case where we have an npm org included in the package name\n          [name, pkgVer] = [`@${splits[1]}`, splits[2]];\n        } else {\n          // this is the case without an npm org\n          [name, pkgVer] = splits;\n        }\n\n        if (installType === INSTALL_TYPE_NPM) {\n          // if we're installing a named package from npm, we don't need to check\n          // against the appium extension list; just use the installSpec as is\n          pkgName = name;\n        } else {\n          // if we're installing a named appium driver (like 'xcuitest') we need to\n          // dereference the actual npm package ('appiupm-xcuitest-driver'), so\n          // check it exists and get the correct package\n          const knownNames = Object.keys(this.knownExtensions);\n          if (!_.includes(knownNames, name)) {\n            const msg =\n              `Could not resolve ${this.type}; are you sure it's in the list ` +\n              `of supported ${this.type}s? ${JSON.stringify(knownNames)}`;\n            throw this._createFatalError(msg);\n          }\n          probableExtName = name;\n          pkgName = this.knownExtensions[name];\n          // given that we'll use the install type in the driver json, store it as\n          // 'npm' now\n          installType = INSTALL_TYPE_NPM;\n        }\n      }\n      installOpts = {installSpec, pkgName, pkgVer};\n    }\n\n    // fail fast here if we can\n    if (probableExtName && this.config.isInstalled(probableExtName)) {\n      throw this._createFatalError(\n        `A ${this.type} named \"${probableExtName}\" is already installed. ` +\n          `Did you mean to update? Run \"appium ${this.type} update\". See ` +\n          `installed ${this.type}s with \"appium ${this.type} list --installed\".`\n      );\n    }\n\n    extData = await this.installViaNpm(installOpts);\n\n    // this _should_ be the same as `probablyExtName` as the one derived above unless\n    // install type is local.\n    const extName = extData[/** @type {string} */ (`${this.type}Name`)];\n\n    // check _a second time_ with the more-accurate extName\n    if (this.config.isInstalled(extName)) {\n      throw this._createFatalError(\n        `A ${this.type} named \"${extName}\" is already installed. ` +\n          `Did you mean to update? Run \"appium ${this.type} update\". See ` +\n          `installed ${this.type}s with \"appium ${this.type} list --installed\".`\n      );\n    }\n\n    // this field does not exist as such in the manifest (it's used as a property name instead)\n    // so that's why it's being removed here.\n    delete extData[/** @type {string} */ (`${this.type}Name`)];\n\n    /** @type {ExtManifest<ExtType>} */\n    const extManifest = {...extData, installType, installSpec};\n    const [errors, warnings] = await B.all([\n      this.config.getProblems(extName, extManifest),\n      this.config.getWarnings(extName, extManifest),\n    ]);\n    const errorMap = new Map([[extName, errors]]);\n    const warningMap = new Map([[extName, warnings]]);\n    const {errorSummaries, warningSummaries} = this.config.getValidationResultSummaries(\n      errorMap,\n      warningMap\n    );\n\n    if (!_.isEmpty(errorSummaries)) {\n      throw this._createFatalError(errorSummaries.join('\\n'));\n    }\n\n    // note that we won't show any warnings if there were errors.\n    if (!_.isEmpty(warningSummaries)) {\n      this.log.warn(warningSummaries.join('\\n'));\n    }\n\n    await this.config.addExtension(extName, extManifest);\n\n    // update the if we've changed the local `package.json`\n    if (await env.hasAppiumDependency(this.config.appiumHome)) {\n      await packageDidChange(this.config.appiumHome);\n    }\n\n    // log info for the user\n    this.log.info(this.getPostInstallText({extName, extData}));\n\n    return this.config.installedExtensions;\n  }\n\n  /**\n   * Install an extension via NPM\n   *\n   * @param {InstallViaNpmArgs} args\n   */\n  async installViaNpm({installSpec, pkgName, pkgVer}) {\n    const npmSpec = `${pkgName}${pkgVer ? '@' + pkgVer : ''}`;\n    const specMsg = npmSpec === installSpec ? '' : ` using NPM install spec '${npmSpec}'`;\n    const msg = `Installing '${installSpec}'${specMsg}`;\n    try {\n      const pkgJsonData = await spinWith(this.isJsonOutput, msg, async () => {\n        const pkgJsonData = await npm.installPackage(this.config.appiumHome, pkgName, {\n          pkgVer,\n        });\n        this.validatePackageJson(pkgJsonData, installSpec);\n        return pkgJsonData;\n      });\n\n      return this.getExtensionFields(pkgJsonData);\n    } catch (err) {\n      throw this._createFatalError(`Encountered an error when installing package: ${err.message}`);\n    }\n  }\n\n  /**\n   * Get the text which should be displayed to the user after an extension has been installed. This\n   * is designed to be overridden by drivers/plugins with their own particular text.\n   *\n   * @param {ExtensionArgs} args\n   * @returns {string}\n   */\n  // eslint-disable-next-line no-unused-vars\n  getPostInstallText(args) {\n    throw this._createFatalError('Must be implemented in final class');\n  }\n\n  /**\n   * Take an NPM module's package.json and extract Appium driver information from a special\n   * 'appium' field in the JSON data. We need this information to e.g. determine which class to\n   * load as the main driver class, or to be able to detect incompatibilities between driver and\n   * appium versions.\n   *\n   * @param {ExtPackageJson<ExtType>} pkgJson - the package.json data for a driver module, as if it had been straightforwardly 'require'd\n   * @returns {ExtensionFields<ExtType>}\n   */\n  getExtensionFields(pkgJson) {\n    const {appium, name, version, peerDependencies} = pkgJson;\n\n    /** @type {unknown} */\n    const result = {\n      ...appium,\n      pkgName: name,\n      version,\n      appiumVersion: peerDependencies?.appium,\n    };\n    return /** @type {ExtensionFields<ExtType>} */ (result);\n  }\n\n  /**\n   * Validates the _required_ root fields of an extension's `package.json` file.\n   *\n   * These required fields are:\n   * - `name`\n   * - `version`\n   * - `appium`\n   * @param {import('type-fest').PackageJson} pkgJson - `package.json` of extension\n   * @param {string} installSpec - Extension name/spec\n   * @throws {ReferenceError} If `package.json` has a missing or invalid field\n   * @returns {pkgJson is ExtPackageJson<ExtType>}\n   */\n  validatePackageJson(pkgJson, installSpec) {\n    const {appium, name, version} = /** @type {ExtPackageJson<ExtType>} */ (pkgJson);\n\n    /**\n     *\n     * @param {string} field\n     * @returns {ReferenceError}\n     */\n    const createMissingFieldError = (field) =>\n      new ReferenceError(\n        `${this.type} \"${installSpec}\" invalid; missing a \\`${field}\\` field of its \\`package.json\\``\n      );\n\n    if (!name) {\n      throw createMissingFieldError('name');\n    }\n    if (!version) {\n      throw createMissingFieldError('version');\n    }\n    if (!appium) {\n      throw createMissingFieldError('appium');\n    }\n\n    this.validateExtensionFields(appium, installSpec);\n\n    return true;\n  }\n\n  /**\n   * For any `package.json` fields which a particular type of extension requires, validate the\n   * presence and form of those fields on the `package.json` data, throwing an error if anything is\n   * amiss.\n   *\n   * @param {ExtMetadata<ExtType>} extMetadata - the data in the \"appium\" field of `package.json` for an extension\n   * @param {string} installSpec - Extension name/spec\n   */\n  // eslint-disable-next-line no-unused-vars\n  validateExtensionFields(extMetadata, installSpec) {\n    throw this._createFatalError('Must be implemented in final class');\n  }\n\n  /**\n   * Uninstall an extension.\n   *\n   * First tries to do this via `npm uninstall`, but if that fails, just `rm -rf`'s the extension dir.\n   *\n   * Will only remove the extension from the manifest if it has been successfully removed.\n   *\n   * @param {UninstallOpts} opts\n   * @return {Promise<ExtRecord<ExtType>>} map of all installed extension names to extension data (without the extension just uninstalled)\n   */\n  async _uninstall({installSpec}) {\n    if (!this.config.isInstalled(installSpec)) {\n      throw this._createFatalError(\n        `Can't uninstall ${this.type} '${installSpec}'; it is not installed`\n      );\n    }\n    const pkgName = this.config.installedExtensions[installSpec].pkgName;\n    await npm.uninstallPackage(this.config.appiumHome, pkgName);\n    await this.config.removeExtension(installSpec);\n    this.log.ok(`Successfully uninstalled ${this.type} '${installSpec}'`.green);\n    return this.config.installedExtensions;\n  }\n\n  /**\n   * Attempt to update one or more drivers using NPM\n   *\n   * @param {ExtensionUpdateOpts} updateSpec\n   * @return {Promise<ExtensionUpdateResult>}\n   */\n  async _update({installSpec, unsafe}) {\n    const shouldUpdateAll = installSpec === UPDATE_ALL;\n    // if we're specifically requesting an update for an extension, make sure it's installed\n    if (!shouldUpdateAll && !this.config.isInstalled(installSpec)) {\n      throw this._createFatalError(\n        `The ${this.type} \"${installSpec}\" was not installed, so can't be updated`\n      );\n    }\n    const extsToUpdate = shouldUpdateAll\n      ? Object.keys(this.config.installedExtensions)\n      : [installSpec];\n\n    // 'errors' will have ext names as keys and error objects as values\n    /** @type {Record<string,Error>} */\n    const errors = {};\n\n    // 'updates' will have ext names as keys and update objects as values, where an update\n    // object is of the form {from: versionString, to: versionString}\n    /** @type {Record<string,UpdateReport>} */\n    const updates = {};\n\n    for (const e of extsToUpdate) {\n      try {\n        await spinWith(this.isJsonOutput, `Checking if ${this.type} '${e}' is updatable`, () => {\n          if (this.config.installedExtensions[e].installType !== INSTALL_TYPE_NPM) {\n            throw new NotUpdatableError();\n          }\n        });\n        const update = await spinWith(\n          this.isJsonOutput,\n          `Checking if ${this.type} '${e}' needs an update`,\n          async () => {\n            const update = await this.checkForExtensionUpdate(e);\n            if (!(update.safeUpdate || update.unsafeUpdate)) {\n              throw new NoUpdatesAvailableError();\n            }\n            return update;\n          }\n        );\n        if (!unsafe && !update.safeUpdate) {\n          throw this._createFatalError(\n            `The ${this.type} '${e}' has a major revision update ` +\n              `(${update.current} => ${update.unsafeUpdate}), which could include ` +\n              `breaking changes. If you want to apply this update, re-run with --unsafe`\n          );\n        }\n        const updateVer = unsafe && update.unsafeUpdate ? update.unsafeUpdate : update.safeUpdate;\n        await spinWith(\n          this.isJsonOutput,\n          `Updating driver '${e}' from ${update.current} to ${updateVer}`,\n          async () => await this.updateExtension(e, updateVer)\n        );\n        updates[e] = {from: update.current, to: updateVer};\n      } catch (err) {\n        errors[e] = err;\n      }\n    }\n\n    this.log.info('Update report:');\n\n    for (const [e, update] of _.toPairs(updates)) {\n      this.log.ok(`  - ${this.type} ${e} updated: ${update.from} => ${update.to}`.green);\n    }\n\n    for (const [e, err] of _.toPairs(errors)) {\n      if (err instanceof NotUpdatableError) {\n        this.log.warn(\n          `  - '${e}' was not installed via npm, so we could not check ` + `for updates`.yellow\n        );\n      } else if (err instanceof NoUpdatesAvailableError) {\n        this.log.info(`  - '${e}' had no updates available`.yellow);\n      } else {\n        // otherwise, make it pop with red!\n        this.log.error(`  - '${e}' failed to update: ${err}`.red);\n      }\n    }\n    return {updates, errors};\n  }\n\n  /**\n   * Given an extension name, figure out what its highest possible version upgrade is, and also the\n   * highest possible safe upgrade.\n   *\n   * @param {string} ext - name of extension\n   * @return {Promise<PossibleUpdates>}\n   */\n  async checkForExtensionUpdate(ext) {\n    // TODO decide how we want to handle beta versions?\n    // this is a helper method, 'ext' is assumed to already be installed here, and of the npm\n    // install type\n    const {version, pkgName} = this.config.installedExtensions[ext];\n    let unsafeUpdate = await npm.getLatestVersion(this.config.appiumHome, pkgName);\n    let safeUpdate = await npm.getLatestSafeUpgradeVersion(\n      this.config.appiumHome,\n      pkgName,\n      version\n    );\n    if (!util.compareVersions(unsafeUpdate, '>', version)) {\n      // the latest version is not greater than the current version, so there's no possible update\n      unsafeUpdate = null;\n      safeUpdate = null;\n    }\n    if (unsafeUpdate && unsafeUpdate === safeUpdate) {\n      // the latest update is the same as the safe update, which means it's not actually unsafe\n      unsafeUpdate = null;\n    }\n    if (safeUpdate && !util.compareVersions(safeUpdate, '>', version)) {\n      // even the safe update is not later than the current, so it is not actually an update\n      safeUpdate = null;\n    }\n    return {current: version, safeUpdate, unsafeUpdate};\n  }\n\n  /**\n   * Actually update an extension installed by NPM, using the NPM cli. And update the installation\n   * manifest.\n   *\n   * @param {string} installSpec - name of extension to update\n   * @param {string} version - version string identifier to update extension to\n   * @returns {Promise<void>}\n   */\n  async updateExtension(installSpec, version) {\n    const {pkgName} = this.config.installedExtensions[installSpec];\n    const extData = await this.installViaNpm({\n      installSpec,\n      pkgName,\n      pkgVer: version,\n    });\n    delete extData[/** @type {string} */ (`${this.type}Name`)];\n    await this.config.updateExtension(installSpec, extData);\n  }\n\n  /**\n   * Runs a script cached inside the \"scripts\" field under \"appium\"\n   * inside of the driver/plugins \"package.json\" file. Will throw\n   * an error if the driver/plugin does not contain a \"scripts\" field\n   * underneath the \"appium\" field in its package.json, if the\n   * \"scripts\" field is not a plain object, or if the scriptName is\n   * not found within \"scripts\" object.\n   *\n   * @param {RunOptions} opts\n   * @return {Promise<RunOutput>}\n   */\n  async _run({installSpec, scriptName}) {\n    if (!this.config.isInstalled(installSpec)) {\n      throw this._createFatalError(`The ${this.type} \"${installSpec}\" is not installed`);\n    }\n\n    const extConfig = this.config.installedExtensions[installSpec];\n\n    // note: TS cannot understand that _.has() is a type guard\n    if (!extConfig.scripts) {\n      throw this._createFatalError(\n        `The ${this.type} named '${installSpec}' does not contain the ` +\n          `\"scripts\" field underneath the \"appium\" field in its package.json`\n      );\n    }\n\n    const extScripts = extConfig.scripts;\n\n    if (!_.isPlainObject(extScripts)) {\n      throw this._createFatalError(\n        `The ${this.type} named '${installSpec}' \"scripts\" field must be a plain object`\n      );\n    }\n\n    if (!_.has(extScripts, scriptName)) {\n      throw this._createFatalError(\n        `The ${this.type} named '${installSpec}' does not support the script: '${scriptName}'`\n      );\n    }\n\n    const runner = new SubProcess(process.execPath, [extScripts[scriptName]], {\n      cwd: this.config.getInstallPath(installSpec),\n    });\n\n    const output = new RingBuffer(50);\n\n    runner.on('stream-line', (line) => {\n      output.enqueue(line);\n      this.log.log(line);\n    });\n\n    await runner.start(0);\n\n    try {\n      await runner.join();\n      this.log.ok(`${scriptName} successfully ran`.green);\n      return {output: output.getBuff()};\n    } catch (err) {\n      this.log.error(`Encountered an error when running '${scriptName}': ${err.message}`.red);\n      return {error: err.message, output: output.getBuff()};\n    }\n  }\n}\n\nexport default ExtensionCommand;\nexport {ExtensionCommand};\n\n/**\n * Options for the {@linkcode ExtensionCommand} constructor\n * @template {ExtensionType} ExtType\n * @typedef ExtensionCommandOptions\n * @property {ExtensionConfig<ExtType>} config - the `DriverConfig` or `PluginConfig` instance used for this command\n * @property {boolean} json - whether the output of this command should be JSON or text\n */\n\n/**\n * Extra stuff about extensions; used indirectly by {@linkcode ExtensionCommand.list}.\n *\n * @typedef ExtensionMetadata\n * @property {boolean} installed - If `true`, the extension is installed\n * @property {string?} updateVersion - If the extension is installed, the version it can be updated to\n * @property {string?} unsafeUpdateVersion - Same as above, but a major version bump\n * @property {boolean} upToDate - If the extension is installed and the latest\n */\n\n/**\n * @typedef {import('@appium/types').ExtensionType} ExtensionType\n * @typedef {import('@appium/types').DriverType} DriverType\n * @typedef {import('@appium/types').PluginType} PluginType\n */\n\n/**\n * @template {ExtensionType} ExtType\n * @typedef {import('appium/types').ExtRecord<ExtType>} ExtRecord\n */\n\n/**\n * @template {ExtensionType} ExtType\n * @typedef {import('../extension/extension-config').ExtensionConfig<ExtType>} ExtensionConfig\n */\n\n/**\n * @template {ExtensionType} ExtType\n * @typedef {import('appium/types').ExtMetadata<ExtType>} ExtMetadata\n */\n\n/**\n * @template {ExtensionType} ExtType\n * @typedef {import('appium/types').ExtManifest<ExtType>} ExtManifest\n */\n\n/**\n * @template {ExtensionType} ExtType\n * @typedef {import('appium/types').ExtPackageJson<ExtType>} ExtPackageJson\n */\n\n/**\n * Possible return value for {@linkcode ExtensionCommand.list}\n * @typedef UninstalledExtensionListData\n * @property {string} pkgName\n * @property {false} installed\n */\n\n/**\n * Possible return value for {@linkcode ExtensionCommand.list}\n * @typedef {import('appium/types').InternalMetadata & ExtensionMetadata} InstalledExtensionListData\n */\n\n/**\n * Return value of {@linkcode ExtensionCommand.list}.\n * @typedef {Record<string,InstalledExtensionListData|UninstalledExtensionListData>} ExtensionListData\n */\n\n/**\n * Options for {@linkcode ExtensionCommand._run}.\n * @typedef RunOptions\n * @property {string} installSpec - name of the extension to run a script from\n * @property {string} scriptName - name of the script to run\n */\n\n/**\n * Return value of {@linkcode ExtensionCommand._run}\n *\n * @typedef RunOutput\n * @property {string} [error] - error message if script ran unsuccessfully, otherwise undefined\n * @property {string[]} output - script output\n */\n\n/**\n * Options for {@linkcode ExtensionCommand._update}.\n * @typedef ExtensionUpdateOpts\n * @property {string} installSpec - the name of the extension to update\n * @property {boolean} unsafe - if true, will perform unsafe updates past major revision boundaries\n */\n\n/**\n * Return value of {@linkcode ExtensionCommand._update}.\n * @typedef ExtensionUpdateResult\n * @property {Record<string,Error>} errors - map of ext names to error objects\n * @property {Record<string,UpdateReport>} updates - map of ext names to {@linkcode UpdateReport}s\n */\n\n/**\n * Part of result of {@linkcode ExtensionCommand._update}.\n * @typedef UpdateReport\n * @property {string} from - version the extension was updated from\n * @property {string} to - version the extension was updated to\n */\n\n/**\n * Options for {@linkcode ExtensionCommand._uninstall}.\n * @typedef UninstallOpts\n * @property {string} installSpec - the name or spec of an extension to uninstall\n */\n\n/**\n * Used by {@linkcode ExtensionCommand.getPostInstallText}\n * @typedef ExtensionArgs\n * @property {string} extName - the name of an extension\n * @property {object} extData - the data for an installed extension\n */\n\n/**\n * Options for {@linkcode ExtensionCommand.installViaNpm}\n * @typedef InstallViaNpmArgs\n * @property {string} installSpec - the name or spec of an extension to install\n * @property {string} pkgName - the NPM package name of the extension\n * @property {string} [pkgVer] - the specific version of the NPM package\n */\n\n/**\n * Object returned by {@linkcode ExtensionCommand.checkForExtensionUpdate}\n * @typedef PossibleUpdates\n * @property {string} current - current version\n * @property {string?} safeUpdate - version we can safely update to if it exists, or null\n * @property {string?} unsafeUpdate - version we can unsafely update to if it exists, or null\n */\n\n/**\n * Options for {@linkcode ExtensionCommand._install}\n * @typedef InstallArgs\n * @property {string} installSpec - the name or spec of an extension to install\n * @property {import('appium/types').InstallType} installType - how to install this extension. One of the INSTALL_TYPES\n * @property {string} [packageName] - for git/github installs, the extension node package name\n */\n\n/**\n * Returned by {@linkcode ExtensionCommand.getExtensionFields}\n * @template {ExtensionType} ExtType\n * @typedef {ExtMetadata<ExtType> & { pkgName: string, version: string, appiumVersion: string } & import('appium/types').CommonExtMetadata} ExtensionFields\n */\n\n/**\n * @template {ExtensionType} ExtType\n * @typedef {ExtType extends DriverType ? typeof import('../constants').KNOWN_DRIVERS : ExtType extends PluginType ? typeof import('../constants').KNOWN_PLUGINS : never} KnownExtensions\n */\n\n/**\n * @typedef ListOptions\n * @property {boolean} showInstalled - whether should show only installed extensions\n * @property {boolean} showUpdates - whether should show available updates\n */\n"],"mappings":"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"}