appium 2.0.0-beta.46 → 2.0.0-beta.48

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 (141) hide show
  1. package/README.md +145 -44
  2. package/build/lib/appium.d.ts +3 -103
  3. package/build/lib/appium.d.ts.map +1 -1
  4. package/build/lib/appium.js +679 -549
  5. package/build/lib/appium.js.map +1 -1
  6. package/build/lib/cli/args.js +247 -127
  7. package/build/lib/cli/args.js.map +1 -1
  8. package/build/lib/cli/driver-command.d.ts +24 -5
  9. package/build/lib/cli/driver-command.d.ts.map +1 -1
  10. package/build/lib/cli/driver-command.js +78 -88
  11. package/build/lib/cli/driver-command.js.map +1 -1
  12. package/build/lib/cli/extension-command.d.ts +33 -24
  13. package/build/lib/cli/extension-command.d.ts.map +1 -1
  14. package/build/lib/cli/extension-command.js +729 -512
  15. package/build/lib/cli/extension-command.js.map +1 -1
  16. package/build/lib/cli/extension.d.ts +7 -6
  17. package/build/lib/cli/extension.d.ts.map +1 -1
  18. package/build/lib/cli/extension.js +68 -65
  19. package/build/lib/cli/extension.js.map +1 -1
  20. package/build/lib/cli/parser.d.ts +3 -3
  21. package/build/lib/cli/parser.d.ts.map +1 -1
  22. package/build/lib/cli/parser.js +234 -192
  23. package/build/lib/cli/parser.js.map +1 -1
  24. package/build/lib/cli/plugin-command.js +58 -87
  25. package/build/lib/cli/plugin-command.js.map +1 -1
  26. package/build/lib/cli/utils.js +66 -69
  27. package/build/lib/cli/utils.js.map +1 -1
  28. package/build/lib/config-file.d.ts.map +1 -1
  29. package/build/lib/config-file.js +189 -120
  30. package/build/lib/config-file.js.map +1 -1
  31. package/build/lib/config.d.ts.map +1 -1
  32. package/build/lib/config.js +254 -213
  33. package/build/lib/config.js.map +1 -1
  34. package/build/lib/constants.d.ts +6 -5
  35. package/build/lib/constants.d.ts.map +1 -1
  36. package/build/lib/constants.js +65 -59
  37. package/build/lib/constants.js.map +1 -1
  38. package/build/lib/extension/driver-config.js +199 -164
  39. package/build/lib/extension/driver-config.js.map +1 -1
  40. package/build/lib/extension/extension-config.d.ts +33 -26
  41. package/build/lib/extension/extension-config.d.ts.map +1 -1
  42. package/build/lib/extension/extension-config.js +541 -396
  43. package/build/lib/extension/extension-config.js.map +1 -1
  44. package/build/lib/extension/index.js +98 -68
  45. package/build/lib/extension/index.js.map +1 -1
  46. package/build/lib/extension/manifest-migrations.d.ts +27 -0
  47. package/build/lib/extension/manifest-migrations.d.ts.map +1 -0
  48. package/build/lib/extension/manifest-migrations.js +118 -0
  49. package/build/lib/extension/manifest-migrations.js.map +1 -0
  50. package/build/lib/extension/manifest.d.ts +35 -63
  51. package/build/lib/extension/manifest.d.ts.map +1 -1
  52. package/build/lib/extension/manifest.js +500 -240
  53. package/build/lib/extension/manifest.js.map +1 -1
  54. package/build/lib/extension/package-changed.js +57 -61
  55. package/build/lib/extension/package-changed.js.map +1 -1
  56. package/build/lib/extension/plugin-config.d.ts +2 -3
  57. package/build/lib/extension/plugin-config.d.ts.map +1 -1
  58. package/build/lib/extension/plugin-config.js +94 -70
  59. package/build/lib/extension/plugin-config.js.map +1 -1
  60. package/build/lib/grid-register.js +119 -137
  61. package/build/lib/grid-register.js.map +1 -1
  62. package/build/lib/logger.d.ts +1 -1
  63. package/build/lib/logger.d.ts.map +1 -1
  64. package/build/lib/logger.js +5 -15
  65. package/build/lib/logger.js.map +1 -1
  66. package/build/lib/logsink.d.ts.map +1 -1
  67. package/build/lib/logsink.js +189 -183
  68. package/build/lib/logsink.js.map +1 -1
  69. package/build/lib/main.d.ts +19 -12
  70. package/build/lib/main.d.ts.map +1 -1
  71. package/build/lib/main.js +330 -304
  72. package/build/lib/main.js.map +1 -1
  73. package/build/lib/schema/arg-spec.js +153 -108
  74. package/build/lib/schema/arg-spec.js.map +1 -1
  75. package/build/lib/schema/cli-args.js +203 -164
  76. package/build/lib/schema/cli-args.js.map +1 -1
  77. package/build/lib/schema/cli-transformers.js +117 -72
  78. package/build/lib/schema/cli-transformers.js.map +1 -1
  79. package/build/lib/schema/index.js +17 -32
  80. package/build/lib/schema/index.js.map +1 -1
  81. package/build/lib/schema/keywords.js +125 -67
  82. package/build/lib/schema/keywords.js.map +1 -1
  83. package/build/lib/schema/schema.d.ts.map +1 -1
  84. package/build/lib/schema/schema.js +582 -417
  85. package/build/lib/schema/schema.js.map +1 -1
  86. package/build/lib/utils.d.ts +41 -255
  87. package/build/lib/utils.d.ts.map +1 -1
  88. package/build/lib/utils.js +342 -193
  89. package/build/lib/utils.js.map +1 -1
  90. package/build/tsconfig.tsbuildinfo +1 -1
  91. package/build/types/cli.d.ts +45 -34
  92. package/build/types/cli.d.ts.map +1 -1
  93. package/build/types/cli.js +3 -0
  94. package/build/types/cli.js.map +1 -0
  95. package/build/types/index.d.ts +1 -2
  96. package/build/types/index.d.ts.map +1 -1
  97. package/build/types/index.js +19 -0
  98. package/build/types/index.js.map +1 -0
  99. package/build/types/manifest/base.d.ts +135 -0
  100. package/build/types/manifest/base.d.ts.map +1 -0
  101. package/build/types/manifest/base.js +3 -0
  102. package/build/types/manifest/base.js.map +1 -0
  103. package/build/types/manifest/index.d.ts +19 -0
  104. package/build/types/manifest/index.d.ts.map +1 -0
  105. package/build/types/manifest/index.js +40 -0
  106. package/build/types/manifest/index.js.map +1 -0
  107. package/build/types/manifest/v3.d.ts +139 -0
  108. package/build/types/manifest/v3.d.ts.map +1 -0
  109. package/build/types/manifest/v3.js +3 -0
  110. package/build/types/manifest/v3.js.map +1 -0
  111. package/lib/appium.js +1 -1
  112. package/lib/cli/args.js +1 -1
  113. package/lib/cli/driver-command.js +17 -0
  114. package/lib/cli/extension-command.js +119 -65
  115. package/lib/cli/extension.js +9 -8
  116. package/lib/cli/parser.js +2 -2
  117. package/lib/config-file.js +2 -3
  118. package/lib/config.js +3 -2
  119. package/lib/constants.js +7 -5
  120. package/lib/extension/extension-config.js +52 -47
  121. package/lib/extension/manifest-migrations.js +120 -0
  122. package/lib/extension/manifest.js +184 -103
  123. package/lib/extension/plugin-config.js +1 -2
  124. package/lib/logsink.js +26 -5
  125. package/lib/main.js +58 -50
  126. package/lib/schema/schema.js +6 -1
  127. package/lib/utils.js +62 -0
  128. package/package.json +24 -25
  129. package/scripts/autoinstall-extensions.js +78 -26
  130. package/types/cli.ts +81 -42
  131. package/types/index.ts +1 -2
  132. package/types/manifest/README.md +30 -0
  133. package/types/manifest/base.ts +158 -0
  134. package/types/manifest/index.ts +27 -0
  135. package/types/manifest/v3.ts +161 -0
  136. package/build/types/appium-manifest.d.ts +0 -59
  137. package/build/types/appium-manifest.d.ts.map +0 -1
  138. package/build/types/extension-manifest.d.ts +0 -55
  139. package/build/types/extension-manifest.d.ts.map +0 -1
  140. package/types/appium-manifest.ts +0 -73
  141. package/types/extension-manifest.ts +0 -64
@@ -1,261 +1,521 @@
1
1
  "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
2
+ /**
3
+ * Module containing {@link Manifest} which handles reading & writing of extension config files.
4
+ */
5
+ var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
6
+ if (kind === "m") throw new TypeError("Private method is not writable");
7
+ if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
8
+ if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
9
+ return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
10
+ };
11
+ var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
12
+ if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
13
+ if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
14
+ return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
15
+ };
16
+ var __importDefault = (this && this.__importDefault) || function (mod) {
17
+ return (mod && mod.__esModule) ? mod : { "default": mod };
18
+ };
19
+ var _Manifest_instances, _Manifest_data, _Manifest_appiumHome, _Manifest_manifestPath, _Manifest_writing, _Manifest_reading, _Manifest_setManifestPath;
20
+ Object.defineProperty(exports, "__esModule", { value: true });
6
21
  exports.Manifest = void 0;
7
-
8
- require("source-map-support/register");
9
-
10
- var _bluebird = _interopRequireDefault(require("bluebird"));
11
-
12
- var _glob = _interopRequireDefault(require("glob"));
13
-
14
- var _support = require("@appium/support");
15
-
16
- var _lodash = _interopRequireDefault(require("lodash"));
17
-
18
- var _path = _interopRequireDefault(require("path"));
19
-
20
- var _yaml = _interopRequireDefault(require("yaml"));
21
-
22
- var _constants = require("../constants");
23
-
24
- var _logger = _interopRequireDefault(require("../logger"));
25
-
26
- var _extensionConfig = require("./extension-config");
27
-
28
- var _packageChanged = require("./package-changed");
29
-
30
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
31
-
32
- const CONFIG_SCHEMA_REV = 2;
33
- const CONFIG_DATA_DRIVER_KEY = `${_constants.DRIVER_TYPE}s`;
34
- const CONFIG_DATA_PLUGIN_KEY = `${_constants.PLUGIN_TYPE}s`;
22
+ const bluebird_1 = __importDefault(require("bluebird"));
23
+ const glob_1 = __importDefault(require("glob"));
24
+ const support_1 = require("@appium/support");
25
+ const lodash_1 = __importDefault(require("lodash"));
26
+ const path_1 = __importDefault(require("path"));
27
+ const yaml_1 = __importDefault(require("yaml"));
28
+ const constants_1 = require("../constants");
29
+ const logger_1 = __importDefault(require("../logger"));
30
+ const extension_config_1 = require("./extension-config");
31
+ const package_changed_1 = require("./package-changed");
32
+ const manifest_migrations_1 = require("./manifest-migrations");
33
+ /**
34
+ * The name of the prop (`drivers`) used in `extensions.yaml` for drivers.
35
+ * @type {`${typeof DRIVER_TYPE}s`}
36
+ */
37
+ const CONFIG_DATA_DRIVER_KEY = `${constants_1.DRIVER_TYPE}s`;
38
+ /**
39
+ * The name of the prop (`plugins`) used in `extensions.yaml` for plugins.
40
+ * @type {`${typeof PLUGIN_TYPE}s`}
41
+ */
42
+ const CONFIG_DATA_PLUGIN_KEY = `${constants_1.PLUGIN_TYPE}s`;
43
+ /**
44
+ * @type {Readonly<ManifestData>}
45
+ */
35
46
  const INITIAL_MANIFEST_DATA = Object.freeze({
36
- [CONFIG_DATA_DRIVER_KEY]: Object.freeze({}),
37
- [CONFIG_DATA_PLUGIN_KEY]: Object.freeze({}),
38
- schemaRev: CONFIG_SCHEMA_REV
47
+ [CONFIG_DATA_DRIVER_KEY]: Object.freeze({}),
48
+ [CONFIG_DATA_PLUGIN_KEY]: Object.freeze({}),
49
+ schemaRev: constants_1.CURRENT_SCHEMA_REV,
39
50
  });
40
-
51
+ /**
52
+ * Given a `package.json` return `true` if it represents an Appium Extension (either a driver or plugin).
53
+ *
54
+ * _This is a type guard; not a validator._
55
+ *
56
+ * The `package.json` must have an `appium` property which is an object.
57
+ * @param {any} value
58
+ * @returns {value is ExtPackageJson<ExtensionType>}
59
+ */
41
60
  function isExtension(value) {
42
- return _lodash.default.isPlainObject(value) && _lodash.default.isPlainObject(value.appium) && _lodash.default.isString(value.name) && _lodash.default.isString(value.version);
61
+ return (lodash_1.default.isPlainObject(value) &&
62
+ lodash_1.default.isPlainObject(value.appium) &&
63
+ lodash_1.default.isString(value.name) &&
64
+ lodash_1.default.isString(value.version));
43
65
  }
44
-
66
+ /**
67
+ * Given a `package.json`, return `true` if it represents an Appium Driver.
68
+ *
69
+ * _This is a type guard; not a validator._
70
+ *
71
+ * To be considered a driver, a `package.json` must have an `appium.driverName` field.
72
+ *
73
+ * Further validation of the `appium` property happens elsewhere.
74
+ * @param {any} value - Value to test
75
+ * @returns {value is ExtPackageJson<DriverType>}
76
+ */
45
77
  function isDriver(value) {
46
- return isExtension(value) && _lodash.default.isString(_lodash.default.get(value, 'appium.driverName')) && _lodash.default.isString(_lodash.default.get(value, 'appium.automationName')) && _lodash.default.isArray(_lodash.default.get(value, 'appium.platformNames'));
78
+ return isExtension(value) && 'driverName' in value.appium && lodash_1.default.isString(value.appium.driverName);
47
79
  }
48
-
80
+ /**
81
+ * Given a `package.json`, return `true` if it represents an Appium Plugin.
82
+ *
83
+ * _This is a type guard; not a validator._
84
+ *
85
+ * To be considered a plugin, a `package.json` must have an `appium.pluginName` field.
86
+ *
87
+ * Further validation of the `appium` property happens elsewhere.
88
+ * @param {any} value - Value to test
89
+ * @returns {value is ExtPackageJson<PluginType>}
90
+ */
49
91
  function isPlugin(value) {
50
- return isExtension(value) && _lodash.default.isString(_lodash.default.get(value, 'appium.pluginName'));
92
+ return isExtension(value) && 'pluginName' in value.appium && lodash_1.default.isString(value.appium.pluginName);
51
93
  }
52
-
94
+ /**
95
+ * Handles reading & writing of extension config files.
96
+ *
97
+ * Only one instance of this class exists per value of `APPIUM_HOME`.
98
+ */
53
99
  class Manifest {
54
- _data;
55
- _appiumHome;
56
- _manifestPath;
57
- _writing;
58
- _reading;
59
-
60
- constructor(appiumHome) {
61
- this._appiumHome = appiumHome;
62
- this._data = _lodash.default.cloneDeep(INITIAL_MANIFEST_DATA);
63
- }
64
-
65
- static getInstance = _lodash.default.memoize(function _getInstance(appiumHome) {
66
- return new Manifest(appiumHome);
67
- });
68
-
69
- async syncWithInstalledExtensions() {
70
- let didChange = false;
71
-
72
- const onMatch = async filepath => {
73
- try {
74
- const pkg = JSON.parse(await _support.fs.readFile(filepath, 'utf8'));
75
-
76
- if (isDriver(pkg) || isPlugin(pkg)) {
77
- const changed = this.addExtensionFromPackage(pkg, filepath);
78
- didChange = didChange || changed;
79
- }
80
- } catch {}
81
- };
82
-
83
- const queue = [onMatch(_path.default.join(this._appiumHome, 'package.json'))];
84
- await new _bluebird.default((resolve, reject) => {
85
- (0, _glob.default)('node_modules/{*,@*/*}/package.json', {
86
- cwd: this._appiumHome,
87
- silent: true,
88
- absolute: true
89
- }, err => {
90
- if (err) {
91
- reject(err);
92
- }
93
-
94
- resolve();
95
- }).on('error', reject).on('match', filepath => {
96
- queue.push(onMatch(filepath));
97
- });
98
- });
99
- await _bluebird.default.all(queue);
100
- return didChange;
101
- }
102
-
103
- hasDriver(name) {
104
- return Boolean(this._data.drivers[name]);
105
- }
106
-
107
- hasPlugin(name) {
108
- return Boolean(this._data.plugins[name]);
109
- }
110
-
111
- addExtensionFromPackage(pkgJson, pkgPath) {
112
- var _pkgJson$peerDependen;
113
-
114
- const extensionPath = _path.default.dirname(pkgPath);
115
-
116
- const internal = {
117
- pkgName: pkgJson.name,
118
- version: pkgJson.version,
119
- appiumVersion: (_pkgJson$peerDependen = pkgJson.peerDependencies) === null || _pkgJson$peerDependen === void 0 ? void 0 : _pkgJson$peerDependen.appium,
120
- installType: _extensionConfig.INSTALL_TYPE_NPM,
121
- installSpec: `${pkgJson.name}@${pkgJson.version}`
122
- };
123
-
124
- if (isDriver(pkgJson)) {
125
- if (!this.hasDriver(pkgJson.appium.driverName)) {
126
- this.addExtension(_constants.DRIVER_TYPE, pkgJson.appium.driverName, { ..._lodash.default.omit(pkgJson.appium, 'driverName'),
127
- ...internal
128
- });
129
- return true;
130
- }
131
-
132
- return false;
133
- } else if (isPlugin(pkgJson)) {
134
- if (!this.hasPlugin(pkgJson.appium.pluginName)) {
135
- this.addExtension(_constants.PLUGIN_TYPE, pkgJson.appium.pluginName, { ..._lodash.default.omit(pkgJson.appium, 'pluginName'),
136
- ...internal
100
+ /**
101
+ * Sets internal data to a fresh clone of {@link INITIAL_MANIFEST_DATA}
102
+ *
103
+ * Use {@link Manifest.getInstance} instead.
104
+ * @param {string} appiumHome
105
+ * @private
106
+ */
107
+ constructor(appiumHome) {
108
+ _Manifest_instances.add(this);
109
+ /**
110
+ * The entire contents of a parsed YAML extension config file.
111
+ *
112
+ * Contains proxies for automatic persistence on disk
113
+ * @type {ManifestData}
114
+ */
115
+ _Manifest_data.set(this, void 0);
116
+ /**
117
+ * Path to `APPIUM_HOME`.
118
+ * @type {Readonly<string>}
119
+ */
120
+ _Manifest_appiumHome.set(this, void 0);
121
+ /**
122
+ * Path to `extensions.yaml`
123
+ * @type {string}
124
+ * Not set until {@link Manifest.read} is called.
125
+ */
126
+ _Manifest_manifestPath.set(this, void 0);
127
+ /**
128
+ * Helps avoid writing multiple times.
129
+ *
130
+ * If this is `undefined`, calling {@link Manifest.write} will cause it to be
131
+ * set to a `Promise`. When the call to `write()` is complete, the `Promise`
132
+ * will resolve and then this value will be set to `undefined`. Concurrent calls
133
+ * made while this value is a `Promise` will return the `Promise` itself.
134
+ * @type {Promise<boolean>|undefined}
135
+ */
136
+ _Manifest_writing.set(this, void 0);
137
+ /**
138
+ * Helps avoid reading multiple times.
139
+ *
140
+ * If this is `undefined`, calling {@link Manifest.read} will cause it to be
141
+ * set to a `Promise`. When the call to `read()` is complete, the `Promise`
142
+ * will resolve and then this value will be set to `undefined`. Concurrent calls
143
+ * made while this value is a `Promise` will return the `Promise` itself.
144
+ * @type {Promise<void>|undefined}
145
+ */
146
+ _Manifest_reading.set(this, void 0);
147
+ __classPrivateFieldSet(this, _Manifest_appiumHome, appiumHome, "f");
148
+ __classPrivateFieldSet(this, _Manifest_data, lodash_1.default.cloneDeep(INITIAL_MANIFEST_DATA), "f");
149
+ }
150
+ /**
151
+ * Searches `APPIUM_HOME` for installed extensions and adds them to the manifest.
152
+ * @returns {Promise<boolean>} `true` if any extensions were added, `false` otherwise.
153
+ */
154
+ async syncWithInstalledExtensions() {
155
+ // this could be parallelized, but we can't use fs.walk as an async iterator
156
+ let didChange = false;
157
+ /**
158
+ * Listener for the `match` event of a `glob` instance
159
+ * @param {string} filepath - Path to a `package.json`
160
+ * @returns {Promise<void>}
161
+ */
162
+ const onMatch = async (filepath) => {
163
+ try {
164
+ const pkg = JSON.parse(await support_1.fs.readFile(filepath, 'utf8'));
165
+ if (isExtension(pkg)) {
166
+ const extType = isDriver(pkg) ? constants_1.DRIVER_TYPE : constants_1.PLUGIN_TYPE;
167
+ /**
168
+ * this should only be 'unknown' if the extension's `package.json` is invalid
169
+ * @type {string}
170
+ */
171
+ const name = isDriver(pkg)
172
+ ? pkg.appium.driverName
173
+ : isPlugin(pkg)
174
+ ? pkg.appium.pluginName
175
+ : '(unknown)';
176
+ if ((isDriver(pkg) && !this.hasDriver(name)) ||
177
+ (isPlugin(pkg) && !this.hasPlugin(name))) {
178
+ logger_1.default.info(`Discovered installed ${extType} "${name}"`);
179
+ }
180
+ const changed = this.addExtensionFromPackage(pkg, filepath);
181
+ didChange = didChange || changed;
182
+ }
183
+ }
184
+ catch { }
185
+ };
186
+ /**
187
+ * A list of `Promise`s which read `package.json` files looking for Appium extensions.
188
+ * @type {Promise<void>[]}
189
+ */
190
+ const queue = [
191
+ // look at `package.json` in `APPIUM_HOME` only
192
+ onMatch(path_1.default.join(__classPrivateFieldGet(this, _Manifest_appiumHome, "f"), 'package.json')),
193
+ ];
194
+ // add dependencies to the queue
195
+ await new bluebird_1.default((resolve, reject) => {
196
+ (0, glob_1.default)('node_modules/{*,@*/*}/package.json', { cwd: __classPrivateFieldGet(this, _Manifest_appiumHome, "f"), silent: true, absolute: true },
197
+ // eslint-disable-next-line promise/prefer-await-to-callbacks
198
+ (err) => {
199
+ if (err) {
200
+ reject(err);
201
+ }
202
+ resolve();
203
+ })
204
+ .on('error', reject)
205
+ .on('match', (filepath) => {
206
+ queue.push(onMatch(filepath));
207
+ });
137
208
  });
138
- return true;
139
- }
140
-
141
- return false;
142
- } else {
143
- throw new TypeError(`The extension in ${extensionPath} is neither a valid driver nor a valid plugin.`);
209
+ // wait for everything to finish
210
+ await bluebird_1.default.all(queue);
211
+ return didChange;
212
+ }
213
+ /**
214
+ * Returns `true` if driver with name `name` is registered.
215
+ * @param {string} name - Driver name
216
+ * @returns {boolean}
217
+ */
218
+ hasDriver(name) {
219
+ return Boolean(__classPrivateFieldGet(this, _Manifest_data, "f").drivers[name]);
144
220
  }
145
- }
146
-
147
- addExtension(extType, extName, extData) {
148
- const data = _lodash.default.clone(extData);
149
-
150
- this._data[`${extType}s`][extName] = data;
151
- return data;
152
- }
153
-
154
- get appiumHome() {
155
- return this._appiumHome;
156
- }
157
-
158
- get manifestPath() {
159
- return this._manifestPath;
160
- }
161
-
162
- getExtensionData(extType) {
163
- return this._data[`${extType}s`];
164
- }
165
-
166
- async read() {
167
- if (this._reading) {
168
- await this._reading;
169
- return this._data;
221
+ /**
222
+ * Returns `true` if plugin with name `name` is registered.
223
+ * @param {string} name - Plugin name
224
+ * @returns {boolean}
225
+ */
226
+ hasPlugin(name) {
227
+ return Boolean(__classPrivateFieldGet(this, _Manifest_data, "f").plugins[name]);
170
228
  }
171
-
172
- this._reading = (async () => {
173
- let data;
174
- let isNewFile = false;
175
- await this._setManifestPath();
176
-
177
- try {
178
- _logger.default.debug(`Reading ${this._manifestPath}...`);
179
-
180
- const yaml = await _support.fs.readFile(this._manifestPath, 'utf8');
181
- data = _yaml.default.parse(yaml);
182
-
183
- _logger.default.debug(`Parsed manifest file: ${JSON.stringify(data, null, 2)}`);
184
- } catch (err) {
185
- if (err.code === 'ENOENT') {
186
- data = _lodash.default.cloneDeep(INITIAL_MANIFEST_DATA);
187
- isNewFile = true;
188
- } else {
189
- if (this._manifestPath) {
190
- throw new Error(`Appium had trouble loading the extension installation ` + `cache file (${this._manifestPath}). It may be invalid YAML. Specific error: ${err.message}`);
191
- } else {
192
- throw new Error(`Appium encountered an unknown problem. Specific error: ${err.message}`);
193
- }
229
+ /**
230
+ * Given a path to a `package.json`, add it as either a driver or plugin to the manifest.
231
+ *
232
+ * @template {ExtensionType} ExtType
233
+ * @param {ExtPackageJson<ExtType>} pkgJson
234
+ * @param {string} pkgPath
235
+ * @returns {boolean} - `true` if this method did anything.
236
+ */
237
+ addExtensionFromPackage(pkgJson, pkgPath) {
238
+ const extensionPath = path_1.default.dirname(pkgPath);
239
+ /**
240
+ * @type {InternalMetadata}
241
+ */
242
+ const internal = {
243
+ pkgName: pkgJson.name,
244
+ version: pkgJson.version,
245
+ appiumVersion: pkgJson.peerDependencies?.appium,
246
+ installType: extension_config_1.INSTALL_TYPE_NPM,
247
+ installSpec: `${pkgJson.name}@${pkgJson.version}`,
248
+ installPath: extensionPath,
249
+ };
250
+ if (isDriver(pkgJson)) {
251
+ const value = {
252
+ ...lodash_1.default.omit(pkgJson.appium, 'driverName'),
253
+ ...internal,
254
+ };
255
+ if (!lodash_1.default.isEqual(value, __classPrivateFieldGet(this, _Manifest_data, "f").drivers[pkgJson.appium.driverName])) {
256
+ this.setExtension(constants_1.DRIVER_TYPE, pkgJson.appium.driverName, value);
257
+ return true;
258
+ }
259
+ return false;
194
260
  }
195
- }
196
-
197
- this._data = data;
198
- let installedExtensionsChanged = false;
199
-
200
- if ((await _support.env.hasAppiumDependency(this.appiumHome)) && (await (0, _packageChanged.packageDidChange)(this.appiumHome))) {
201
- installedExtensionsChanged = await this.syncWithInstalledExtensions();
202
- }
203
-
204
- if (isNewFile || installedExtensionsChanged) {
205
- await this.write();
206
- }
207
- })();
208
-
209
- try {
210
- await this._reading;
211
- return this._data;
212
- } finally {
213
- this._reading = undefined;
261
+ else if (isPlugin(pkgJson)) {
262
+ const value = {
263
+ ...lodash_1.default.omit(pkgJson.appium, 'pluginName'),
264
+ ...internal,
265
+ };
266
+ if (!lodash_1.default.isEqual(value, __classPrivateFieldGet(this, _Manifest_data, "f").plugins[pkgJson.appium.pluginName])) {
267
+ this.setExtension(constants_1.PLUGIN_TYPE, pkgJson.appium.pluginName, value);
268
+ return true;
269
+ }
270
+ return false;
271
+ }
272
+ else {
273
+ throw new TypeError(`The extension in ${extensionPath} is neither a valid ${constants_1.DRIVER_TYPE} nor a valid ${constants_1.PLUGIN_TYPE}.`);
274
+ }
275
+ }
276
+ /**
277
+ * Adds an extension to the manifest as was installed by the `appium` CLI. The
278
+ * `extData`, `extType`, and `extName` have already been determined.
279
+ *
280
+ * See {@link Manifest.addExtensionFromPackage} for adding an extension from an on-disk package.
281
+ * @template {ExtensionType} ExtType
282
+ * @param {ExtType} extType - `driver` or `plugin`
283
+ * @param {string} extName - Name of extension
284
+ * @param {ExtManifest<ExtType>} extData - Extension metadata
285
+ * @returns {ExtManifest<ExtType>} A clone of `extData`, potentially with a mutated `appiumVersion` field
286
+ */
287
+ setExtension(extType, extName, extData) {
288
+ const data = lodash_1.default.cloneDeep(extData);
289
+ __classPrivateFieldGet(this, _Manifest_data, "f")[`${extType}s`][extName] = data;
290
+ return data;
291
+ }
292
+ /**
293
+ * Sets the schema revision
294
+ * @param {keyof import('./manifest-migrations').ManifestDataVersions} rev
295
+ */
296
+ setSchemaRev(rev) {
297
+ __classPrivateFieldGet(this, _Manifest_data, "f").schemaRev = rev;
298
+ }
299
+ /**
300
+ * Remove an extension from the manifest.
301
+ * @param {ExtensionType} extType
302
+ * @param {string} extName
303
+ */
304
+ deleteExtension(extType, extName) {
305
+ delete __classPrivateFieldGet(this, _Manifest_data, "f")[`${extType}s`][extName];
306
+ }
307
+ /**
308
+ * Returns the `APPIUM_HOME` path
309
+ */
310
+ get appiumHome() {
311
+ return __classPrivateFieldGet(this, _Manifest_appiumHome, "f");
214
312
  }
215
- }
216
-
217
- async _setManifestPath() {
218
- if (!this._manifestPath) {
219
- this._manifestPath = await _support.env.resolveManifestPath(this._appiumHome);
220
-
221
- if (_path.default.relative(this._appiumHome, this._manifestPath).startsWith('.')) {
222
- throw new Error(`Mismatch between location of APPIUM_HOME and manifest file. APPIUM_HOME: ${this.appiumHome}, manifest file: ${this._manifestPath}`);
223
- }
313
+ /**
314
+ * Returns the path to the manifest file (`extensions.yaml`)
315
+ */
316
+ get manifestPath() {
317
+ return __classPrivateFieldGet(this, _Manifest_manifestPath, "f");
224
318
  }
225
-
226
- return this._manifestPath;
227
- }
228
-
229
- async write() {
230
- if (this._writing) {
231
- return this._writing;
319
+ /**
320
+ * Returns the schema rev of this manifest
321
+ */
322
+ get schemaRev() {
323
+ return __classPrivateFieldGet(this, _Manifest_data, "f").schemaRev;
232
324
  }
233
-
234
- this._writing = (async () => {
235
- await this._setManifestPath();
236
-
237
- try {
238
- await _support.fs.mkdirp(_path.default.dirname(this._manifestPath));
239
- } catch (err) {
240
- throw new Error(`Appium could not create the directory for the manifest file: ${_path.default.dirname(this._manifestPath)}. Original error: ${err.message}`);
241
- }
242
-
243
- try {
244
- await _support.fs.writeFile(this._manifestPath, _yaml.default.stringify(this._data), 'utf8');
245
- return true;
246
- } catch (err) {
247
- throw new Error(`Appium could not write to manifest at ${this._manifestPath} using APPIUM_HOME ${this._appiumHome}. ` + `Please ensure it is writable. Original error: ${err.message}`);
248
- }
249
- })();
250
-
251
- try {
252
- return await this._writing;
253
- } finally {
254
- this._writing = undefined;
325
+ /**
326
+ * Returns extension data for a particular type.
327
+ *
328
+ * @template {ExtensionType} ExtType
329
+ * @param {ExtType} extType
330
+ * @returns {Readonly<ExtRecord<ExtType>>}
331
+ */
332
+ getExtensionData(extType) {
333
+ return __classPrivateFieldGet(this, _Manifest_data, "f")[ /** @type {string} */(`${extType}s`)];
334
+ }
335
+ /**
336
+ * Reads manifest from disk and _overwrites_ the internal data.
337
+ *
338
+ * If the manifest does not exist on disk, an
339
+ * {@link INITIAL_MANIFEST_DATA "empty"} manifest file will be created, as
340
+ * well as its directory if needed.
341
+ *
342
+ * This will also, if necessary:
343
+ * 1. perform a migration of the manifest data
344
+ * 2. sync the manifest with extensions on-disk (kind of like "auto
345
+ * discovery")
346
+ * 3. write the manifest to disk.
347
+ *
348
+ * Only one read operation can happen at a time.
349
+ *
350
+ * @returns {Promise<ManifestData>} The data
351
+ */
352
+ async read() {
353
+ if (__classPrivateFieldGet(this, _Manifest_reading, "f")) {
354
+ await __classPrivateFieldGet(this, _Manifest_reading, "f");
355
+ return __classPrivateFieldGet(this, _Manifest_data, "f");
356
+ }
357
+ __classPrivateFieldSet(this, _Manifest_reading, (async () => {
358
+ /** @type {ManifestData} */
359
+ let data;
360
+ /**
361
+ * This will be `true` if, after reading, we need to update the manifest data
362
+ * and write it again to disk.
363
+ */
364
+ let shouldWrite = false;
365
+ await __classPrivateFieldGet(this, _Manifest_instances, "m", _Manifest_setManifestPath).call(this);
366
+ try {
367
+ const yaml = await support_1.fs.readFile(__classPrivateFieldGet(this, _Manifest_manifestPath, "f"), 'utf8');
368
+ data = yaml_1.default.parse(yaml);
369
+ logger_1.default.debug(`Parsed manifest file at ${__classPrivateFieldGet(this, _Manifest_manifestPath, "f")}: ${JSON.stringify(data, null, 2)}`);
370
+ }
371
+ catch (err) {
372
+ if (err.code === 'ENOENT') {
373
+ logger_1.default.debug(`No manifest file found at ${__classPrivateFieldGet(this, _Manifest_manifestPath, "f")}; creating`);
374
+ data = lodash_1.default.cloneDeep(INITIAL_MANIFEST_DATA);
375
+ shouldWrite = true;
376
+ }
377
+ else {
378
+ if (__classPrivateFieldGet(this, _Manifest_manifestPath, "f")) {
379
+ throw new Error(`Appium had trouble loading the extension installation ` +
380
+ `cache file (${__classPrivateFieldGet(this, _Manifest_manifestPath, "f")}). It may be invalid YAML. Specific error: ${err.message}`);
381
+ }
382
+ else {
383
+ throw new Error(`Appium encountered an unknown problem. Specific error: ${err.message}`);
384
+ }
385
+ }
386
+ }
387
+ __classPrivateFieldSet(this, _Manifest_data, data, "f");
388
+ /**
389
+ * the only way `shouldWrite` is `true` is if we have a new file. a new
390
+ * file will get the latest schema revision, so we can skip the migration.
391
+ */
392
+ if (!shouldWrite && (data.schemaRev ?? 0) < constants_1.CURRENT_SCHEMA_REV) {
393
+ logger_1.default.debug(`Updating manifest schema from rev ${data.schemaRev ?? '(none)'} to ${constants_1.CURRENT_SCHEMA_REV}`);
394
+ shouldWrite = await (0, manifest_migrations_1.migrate)(this);
395
+ }
396
+ /**
397
+ * we still may want to sync with installed extensions even if we have a
398
+ * new file. right now this is limited to the following cases:
399
+ * 1. we have a brand new manifest file
400
+ * 2. we have performed a migration on a manifest file
401
+ * 3. `appium` is a dependency within `package.json`, and `package.json`
402
+ * has changed since last time we checked.
403
+ *
404
+ * It may also make sense to sync with the extensions in an arbitrary
405
+ * `APPIUM_HOME`, but we don't do that here.
406
+ */
407
+ if (shouldWrite ||
408
+ ((await support_1.env.hasAppiumDependency(this.appiumHome)) &&
409
+ (await (0, package_changed_1.packageDidChange)(this.appiumHome)))) {
410
+ logger_1.default.debug('Discovering newly installed extensions...');
411
+ shouldWrite = (await this.syncWithInstalledExtensions()) || shouldWrite;
412
+ }
413
+ if (shouldWrite) {
414
+ await this.write();
415
+ }
416
+ })(), "f");
417
+ try {
418
+ await __classPrivateFieldGet(this, _Manifest_reading, "f");
419
+ return __classPrivateFieldGet(this, _Manifest_data, "f");
420
+ }
421
+ finally {
422
+ __classPrivateFieldSet(this, _Manifest_reading, undefined, "f");
423
+ }
424
+ }
425
+ /**
426
+ * Writes the data if it need s writing.
427
+ *
428
+ * If the `schemaRev` prop needs updating, the file will be written.
429
+ *
430
+ * @todo If this becomes too much of a bottleneck, throttle it.
431
+ * @returns {Promise<boolean>} Whether the data was written
432
+ */
433
+ async write() {
434
+ if (__classPrivateFieldGet(this, _Manifest_writing, "f")) {
435
+ return __classPrivateFieldGet(this, _Manifest_writing, "f");
436
+ }
437
+ __classPrivateFieldSet(this, _Manifest_writing, (async () => {
438
+ await __classPrivateFieldGet(this, _Manifest_instances, "m", _Manifest_setManifestPath).call(this);
439
+ try {
440
+ await support_1.fs.mkdirp(path_1.default.dirname(__classPrivateFieldGet(this, _Manifest_manifestPath, "f")));
441
+ }
442
+ catch (err) {
443
+ throw new Error(`Appium could not create the directory for the manifest file: ${path_1.default.dirname(__classPrivateFieldGet(this, _Manifest_manifestPath, "f"))}. Original error: ${err.message}`);
444
+ }
445
+ try {
446
+ await support_1.fs.writeFile(__classPrivateFieldGet(this, _Manifest_manifestPath, "f"), yaml_1.default.stringify(__classPrivateFieldGet(this, _Manifest_data, "f")), 'utf8');
447
+ return true;
448
+ }
449
+ catch (err) {
450
+ throw new Error(`Appium could not write to manifest at ${__classPrivateFieldGet(this, _Manifest_manifestPath, "f")} using APPIUM_HOME ${__classPrivateFieldGet(this, _Manifest_appiumHome, "f")}. Please ensure it is writable. Original error: ${err.message}`);
451
+ }
452
+ })(), "f");
453
+ try {
454
+ return await __classPrivateFieldGet(this, _Manifest_writing, "f");
455
+ }
456
+ finally {
457
+ __classPrivateFieldSet(this, _Manifest_writing, undefined, "f");
458
+ }
255
459
  }
256
- }
257
-
258
460
  }
259
-
260
461
  exports.Manifest = Manifest;
261
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJDT05GSUdfU0NIRU1BX1JFViIsIkNPTkZJR19EQVRBX0RSSVZFUl9LRVkiLCJEUklWRVJfVFlQRSIsIkNPTkZJR19EQVRBX1BMVUdJTl9LRVkiLCJQTFVHSU5fVFlQRSIsIklOSVRJQUxfTUFOSUZFU1RfREFUQSIsIk9iamVjdCIsImZyZWV6ZSIsInNjaGVtYVJldiIsImlzRXh0ZW5zaW9uIiwidmFsdWUiLCJfIiwiaXNQbGFpbk9iamVjdCIsImFwcGl1bSIsImlzU3RyaW5nIiwibmFtZSIsInZlcnNpb24iLCJpc0RyaXZlciIsImdldCIsImlzQXJyYXkiLCJpc1BsdWdpbiIsIk1hbmlmZXN0IiwiX2RhdGEiLCJfYXBwaXVtSG9tZSIsIl9tYW5pZmVzdFBhdGgiLCJfd3JpdGluZyIsIl9yZWFkaW5nIiwiY29uc3RydWN0b3IiLCJhcHBpdW1Ib21lIiwiY2xvbmVEZWVwIiwiZ2V0SW5zdGFuY2UiLCJtZW1vaXplIiwiX2dldEluc3RhbmNlIiwic3luY1dpdGhJbnN0YWxsZWRFeHRlbnNpb25zIiwiZGlkQ2hhbmdlIiwib25NYXRjaCIsImZpbGVwYXRoIiwicGtnIiwiSlNPTiIsInBhcnNlIiwiZnMiLCJyZWFkRmlsZSIsImNoYW5nZWQiLCJhZGRFeHRlbnNpb25Gcm9tUGFja2FnZSIsInF1ZXVlIiwicGF0aCIsImpvaW4iLCJCIiwicmVzb2x2ZSIsInJlamVjdCIsImdsb2IiLCJjd2QiLCJzaWxlbnQiLCJhYnNvbHV0ZSIsImVyciIsIm9uIiwicHVzaCIsImFsbCIsImhhc0RyaXZlciIsIkJvb2xlYW4iLCJkcml2ZXJzIiwiaGFzUGx1Z2luIiwicGx1Z2lucyIsInBrZ0pzb24iLCJwa2dQYXRoIiwiZXh0ZW5zaW9uUGF0aCIsImRpcm5hbWUiLCJpbnRlcm5hbCIsInBrZ05hbWUiLCJhcHBpdW1WZXJzaW9uIiwicGVlckRlcGVuZGVuY2llcyIsImluc3RhbGxUeXBlIiwiSU5TVEFMTF9UWVBFX05QTSIsImluc3RhbGxTcGVjIiwiZHJpdmVyTmFtZSIsImFkZEV4dGVuc2lvbiIsIm9taXQiLCJwbHVnaW5OYW1lIiwiVHlwZUVycm9yIiwiZXh0VHlwZSIsImV4dE5hbWUiLCJleHREYXRhIiwiZGF0YSIsImNsb25lIiwibWFuaWZlc3RQYXRoIiwiZ2V0RXh0ZW5zaW9uRGF0YSIsInJlYWQiLCJpc05ld0ZpbGUiLCJfc2V0TWFuaWZlc3RQYXRoIiwibG9nIiwiZGVidWciLCJ5YW1sIiwiWUFNTCIsInN0cmluZ2lmeSIsImNvZGUiLCJFcnJvciIsIm1lc3NhZ2UiLCJpbnN0YWxsZWRFeHRlbnNpb25zQ2hhbmdlZCIsImVudiIsImhhc0FwcGl1bURlcGVuZGVuY3kiLCJwYWNrYWdlRGlkQ2hhbmdlIiwid3JpdGUiLCJ1bmRlZmluZWQiLCJyZXNvbHZlTWFuaWZlc3RQYXRoIiwicmVsYXRpdmUiLCJzdGFydHNXaXRoIiwibWtkaXJwIiwid3JpdGVGaWxlIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vbGliL2V4dGVuc2lvbi9tYW5pZmVzdC5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE1vZHVsZSBjb250YWluaW5nIHtAbGluayBNYW5pZmVzdH0gd2hpY2ggaGFuZGxlcyByZWFkaW5nICYgd3JpdGluZyBvZiBleHRlbnNpb24gY29uZmlnIGZpbGVzLlxuICovXG5cbmltcG9ydCBCIGZyb20gJ2JsdWViaXJkJztcbmltcG9ydCBnbG9iIGZyb20gJ2dsb2InO1xuaW1wb3J0IHtlbnYsIGZzfSBmcm9tICdAYXBwaXVtL3N1cHBvcnQnO1xuaW1wb3J0IF8gZnJvbSAnbG9kYXNoJztcbmltcG9ydCBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0IFlBTUwgZnJvbSAneWFtbCc7XG5pbXBvcnQge0RSSVZFUl9UWVBFLCBQTFVHSU5fVFlQRX0gZnJvbSAnLi4vY29uc3RhbnRzJztcbmltcG9ydCBsb2cgZnJvbSAnLi4vbG9nZ2VyJztcbmltcG9ydCB7SU5TVEFMTF9UWVBFX05QTX0gZnJvbSAnLi9leHRlbnNpb24tY29uZmlnJztcbmltcG9ydCB7cGFja2FnZURpZENoYW5nZX0gZnJvbSAnLi9wYWNrYWdlLWNoYW5nZWQnO1xuXG4vKipcbiAqIEN1cnJlbnQgY29uZmlndXJhdGlvbiBzY2hlbWEgcmV2aXNpb24hXG4gKi9cbmNvbnN0IENPTkZJR19TQ0hFTUFfUkVWID0gMjtcblxuLyoqXG4gKiBUaGUgbmFtZSBvZiB0aGUgcHJvcCAoYGRyaXZlcnNgKSB1c2VkIGluIGBleHRlbnNpb25zLnlhbWxgIGZvciBkcml2ZXJzLlxuICogQHR5cGUge2Ake3R5cGVvZiBEUklWRVJfVFlQRX1zYH1cbiAqL1xuY29uc3QgQ09ORklHX0RBVEFfRFJJVkVSX0tFWSA9IGAke0RSSVZFUl9UWVBFfXNgO1xuXG4vKipcbiAqIFRoZSBuYW1lIG9mIHRoZSBwcm9wIChgcGx1Z2luc2ApIHVzZWQgaW4gYGV4dGVuc2lvbnMueWFtbGAgZm9yIHBsdWdpbnMuXG4gKiBAdHlwZSB7YCR7dHlwZW9mIFBMVUdJTl9UWVBFfXNgfVxuICovXG5jb25zdCBDT05GSUdfREFUQV9QTFVHSU5fS0VZID0gYCR7UExVR0lOX1RZUEV9c2A7XG5cbi8qKlxuICogQHR5cGUge1JlYWRvbmx5PE1hbmlmZXN0RGF0YT59XG4gKi9cbmNvbnN0IElOSVRJQUxfTUFOSUZFU1RfREFUQSA9IE9iamVjdC5mcmVlemUoe1xuICBbQ09ORklHX0RBVEFfRFJJVkVSX0tFWV06IE9iamVjdC5mcmVlemUoe30pLFxuICBbQ09ORklHX0RBVEFfUExVR0lOX0tFWV06IE9iamVjdC5mcmVlemUoe30pLFxuICBzY2hlbWFSZXY6IENPTkZJR19TQ0hFTUFfUkVWLFxufSk7XG5cbi8qKlxuICogR2l2ZW4gYSBgcGFja2FnZS5qc29uYCByZXR1cm4gYHRydWVgIGlmIGl0IHJlcHJlc2VudHMgYW4gQXBwaXVtIEV4dGVuc2lvbiAoZWl0aGVyIGEgZHJpdmVyIG9yIHBsdWdpbikuXG4gKlxuICogVGhlIGBwYWNrYWdlLmpzb25gIG11c3QgaGF2ZSBhbiBgYXBwaXVtYCBwcm9wZXJ0eSB3aGljaCBpcyBhbiBvYmplY3QuXG4gKiBAcGFyYW0ge2FueX0gdmFsdWVcbiAqIEByZXR1cm5zIHt2YWx1ZSBpcyBFeHRQYWNrYWdlSnNvbjxFeHRlbnNpb25UeXBlPn1cbiAqL1xuZnVuY3Rpb24gaXNFeHRlbnNpb24odmFsdWUpIHtcbiAgcmV0dXJuIChcbiAgICBfLmlzUGxhaW5PYmplY3QodmFsdWUpICYmXG4gICAgXy5pc1BsYWluT2JqZWN0KHZhbHVlLmFwcGl1bSkgJiZcbiAgICBfLmlzU3RyaW5nKHZhbHVlLm5hbWUpICYmXG4gICAgXy5pc1N0cmluZyh2YWx1ZS52ZXJzaW9uKVxuICApO1xufVxuLyoqXG4gKiBHaXZlbiBhIGBwYWNrYWdlLmpzb25gLCByZXR1cm4gYHRydWVgIGlmIGl0IHJlcHJlc2VudHMgYW4gQXBwaXVtIERyaXZlci5cbiAqXG4gKiBUbyBiZSBjb25zaWRlcmVkIGEgZHJpdmVyLCBhIGBwYWNrYWdlLmpzb25gIG11c3QgaGF2ZSBhIGZpZWxkc1xuICogYGFwcGl1bS5kcml2ZXJOYW1lYCwgYGFwcGl1bS5hdXRvbWF0aW9uTmFtZWAgYW5kIGBhcHBpdW0ucGxhdGZvcm1OYW1lc2AuXG4gKiBAcGFyYW0ge2FueX0gdmFsdWUgLSBWYWx1ZSB0byB0ZXN0XG4gKiBAcmV0dXJucyB7dmFsdWUgaXMgRXh0UGFja2FnZUpzb248RHJpdmVyVHlwZT59XG4gKi9cbmZ1bmN0aW9uIGlzRHJpdmVyKHZhbHVlKSB7XG4gIHJldHVybiAoXG4gICAgaXNFeHRlbnNpb24odmFsdWUpICYmXG4gICAgXy5pc1N0cmluZyhfLmdldCh2YWx1ZSwgJ2FwcGl1bS5kcml2ZXJOYW1lJykpICYmXG4gICAgXy5pc1N0cmluZyhfLmdldCh2YWx1ZSwgJ2FwcGl1bS5hdXRvbWF0aW9uTmFtZScpKSAmJlxuICAgIF8uaXNBcnJheShfLmdldCh2YWx1ZSwgJ2FwcGl1bS5wbGF0Zm9ybU5hbWVzJykpXG4gICk7XG59XG5cbi8qKlxuICogR2l2ZW4gYSBgcGFja2FnZS5qc29uYCwgcmV0dXJuIGB0cnVlYCBpZiBpdCByZXByZXNlbnRzIGFuIEFwcGl1bSBQbHVnaW4uXG4gKlxuICogVG8gYmUgY29uc2lkZXJlZCBhIHBsdWdpbiwgYSBgcGFja2FnZS5qc29uYCBtdXN0IGhhdmUgYW4gYGFwcGl1bS5wbHVnaW5OYW1lYCBmaWVsZC5cbiAqIEBwYXJhbSB7YW55fSB2YWx1ZSAtIFZhbHVlIHRvIHRlc3RcbiAqIEByZXR1cm5zIHt2YWx1ZSBpcyBFeHRQYWNrYWdlSnNvbjxQbHVnaW5UeXBlPn1cbiAqL1xuZnVuY3Rpb24gaXNQbHVnaW4odmFsdWUpIHtcbiAgcmV0dXJuIGlzRXh0ZW5zaW9uKHZhbHVlKSAmJiBfLmlzU3RyaW5nKF8uZ2V0KHZhbHVlLCAnYXBwaXVtLnBsdWdpbk5hbWUnKSk7XG59XG5cbi8qKlxuICogSGFuZGxlcyByZWFkaW5nICYgd3JpdGluZyBvZiBleHRlbnNpb24gY29uZmlnIGZpbGVzLlxuICpcbiAqIE9ubHkgb25lIGluc3RhbmNlIG9mIHRoaXMgY2xhc3MgZXhpc3RzIHBlciB2YWx1ZSBvZiBgQVBQSVVNX0hPTUVgLlxuICovXG5leHBvcnQgY2xhc3MgTWFuaWZlc3Qge1xuICAvKipcbiAgICogVGhlIGVudGlyZSBjb250ZW50cyBvZiBhIHBhcnNlZCBZQU1MIGV4dGVuc2lvbiBjb25maWcgZmlsZS5cbiAgICpcbiAgICogQ29udGFpbnMgcHJveGllcyBmb3IgYXV0b21hdGljIHBlcnNpc3RlbmNlIG9uIGRpc2tcbiAgICogQHR5cGUge01hbmlmZXN0RGF0YX1cbiAgICogQHByaXZhdGVcbiAgICovXG4gIF9kYXRhO1xuXG4gIC8qKlxuICAgKiBQYXRoIHRvIGBBUFBJVU1fSE9NRWAuXG4gICAqIEBwcml2YXRlXG4gICAqIEB0eXBlIHtSZWFkb25seTxzdHJpbmc+fVxuICAgKi9cbiAgX2FwcGl1bUhvbWU7XG5cbiAgLyoqXG4gICAqIFBhdGggdG8gYGV4dGVuc2lvbnMueWFtbGBcbiAgICogQHR5cGUge3N0cmluZ31cbiAgICogTm90IHNldCB1bnRpbCB7QGxpbmsgTWFuaWZlc3QucmVhZH0gaXMgY2FsbGVkLlxuICAgKi9cbiAgX21hbmlmZXN0UGF0aDtcblxuICAvKipcbiAgICogSGVscHMgYXZvaWQgd3JpdGluZyBtdWx0aXBsZSB0aW1lcy5cbiAgICpcbiAgICogSWYgdGhpcyBpcyBgdW5kZWZpbmVkYCwgY2FsbGluZyB7QGxpbmsgTWFuaWZlc3Qud3JpdGV9IHdpbGwgY2F1c2UgaXQgdG8gYmVcbiAgICogc2V0IHRvIGEgYFByb21pc2VgLiBXaGVuIHRoZSBjYWxsIHRvIGB3cml0ZSgpYCBpcyBjb21wbGV0ZSwgdGhlIGBQcm9taXNlYFxuICAgKiB3aWxsIHJlc29sdmUgYW5kIHRoZW4gdGhpcyB2YWx1ZSB3aWxsIGJlIHNldCB0byBgdW5kZWZpbmVkYC4gIENvbmN1cnJlbnQgY2FsbHNcbiAgICogbWFkZSB3aGlsZSB0aGlzIHZhbHVlIGlzIGEgYFByb21pc2VgIHdpbGwgcmV0dXJuIHRoZSBgUHJvbWlzZWAgaXRzZWxmLlxuICAgKiBAcHJpdmF0ZVxuICAgKiBAdHlwZSB7UHJvbWlzZTxib29sZWFuPnx1bmRlZmluZWR9XG4gICAqL1xuICBfd3JpdGluZztcblxuICAvKipcbiAgICogSGVscHMgYXZvaWQgcmVhZGluZyBtdWx0aXBsZSB0aW1lcy5cbiAgICpcbiAgICogSWYgdGhpcyBpcyBgdW5kZWZpbmVkYCwgY2FsbGluZyB7QGxpbmsgTWFuaWZlc3QucmVhZH0gd2lsbCBjYXVzZSBpdCB0byBiZVxuICAgKiBzZXQgdG8gYSBgUHJvbWlzZWAuIFdoZW4gdGhlIGNhbGwgdG8gYHJlYWQoKWAgaXMgY29tcGxldGUsIHRoZSBgUHJvbWlzZWBcbiAgICogd2lsbCByZXNvbHZlIGFuZCB0aGVuIHRoaXMgdmFsdWUgd2lsbCBiZSBzZXQgdG8gYHVuZGVmaW5lZGAuICBDb25jdXJyZW50IGNhbGxzXG4gICAqIG1hZGUgd2hpbGUgdGhpcyB2YWx1ZSBpcyBhIGBQcm9taXNlYCB3aWxsIHJldHVybiB0aGUgYFByb21pc2VgIGl0c2VsZi5cbiAgICogQHByaXZhdGVcbiAgICogQHR5cGUge1Byb21pc2U8dm9pZD58dW5kZWZpbmVkfVxuICAgKi9cbiAgX3JlYWRpbmc7XG5cbiAgLyoqXG4gICAqIFNldHMgaW50ZXJuYWwgZGF0YSB0byBhIGZyZXNoIGNsb25lIG9mIHtAbGluayBJTklUSUFMX01BTklGRVNUX0RBVEF9XG4gICAqXG4gICAqIFVzZSB7QGxpbmsgTWFuaWZlc3QuZ2V0SW5zdGFuY2V9IGluc3RlYWQuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBhcHBpdW1Ib21lXG4gICAqIEBwcml2YXRlXG4gICAqL1xuICBjb25zdHJ1Y3RvcihhcHBpdW1Ib21lKSB7XG4gICAgdGhpcy5fYXBwaXVtSG9tZSA9IGFwcGl1bUhvbWU7XG4gICAgdGhpcy5fZGF0YSA9IF8uY2xvbmVEZWVwKElOSVRJQUxfTUFOSUZFU1RfREFUQSk7XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyBhIG5ldyBvciBleGlzdGluZyB7QGxpbmsgTWFuaWZlc3R9IGluc3RhbmNlLCBiYXNlZCBvbiB0aGUgdmFsdWUgb2YgYGFwcGl1bUhvbWVgLlxuICAgKlxuICAgKiBNYWludGFpbnMgb25lIGluc3RhbmNlIHBlciB2YWx1ZSBvZiBgYXBwaXVtSG9tZWAuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBhcHBpdW1Ib21lIC0gUGF0aCB0byBgQVBQSVVNX0hPTUVgXG4gICAqIEByZXR1cm5zIHtNYW5pZmVzdH1cbiAgICovXG4gIHN0YXRpYyBnZXRJbnN0YW5jZSA9IF8ubWVtb2l6ZShmdW5jdGlvbiBfZ2V0SW5zdGFuY2UoYXBwaXVtSG9tZSkge1xuICAgIHJldHVybiBuZXcgTWFuaWZlc3QoYXBwaXVtSG9tZSk7XG4gIH0pO1xuXG4gIC8qKlxuICAgKiBTZWFyY2hlcyBgQVBQSVVNX0hPTUVgIGZvciBpbnN0YWxsZWQgZXh0ZW5zaW9ucyBhbmQgYWRkcyB0aGVtIHRvIHRoZSBtYW5pZmVzdC5cbiAgICogQHJldHVybnMge1Byb21pc2U8Ym9vbGVhbj59IGB0cnVlYCBpZiBhbnkgZXh0ZW5zaW9ucyB3ZXJlIGFkZGVkLCBgZmFsc2VgIG90aGVyd2lzZS5cbiAgICovXG4gIGFzeW5jIHN5bmNXaXRoSW5zdGFsbGVkRXh0ZW5zaW9ucygpIHtcbiAgICAvLyB0aGlzIGNvdWxkIGJlIHBhcmFsbGVsaXplZCwgYnV0IHdlIGNhbid0IHVzZSBmcy53YWxrIGFzIGFuIGFzeW5jIGl0ZXJhdG9yXG4gICAgbGV0IGRpZENoYW5nZSA9IGZhbHNlO1xuXG4gICAgLyoqXG4gICAgICogTGlzdGVuZXIgZm9yIHRoZSBgbWF0Y2hgIGV2ZW50IG9mIGEgYGdsb2JgIGluc3RhbmNlXG4gICAgICogQHBhcmFtIHtzdHJpbmd9IGZpbGVwYXRoIC0gUGF0aCB0byBhIGBwYWNrYWdlLmpzb25gXG4gICAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59XG4gICAgICovXG4gICAgY29uc3Qgb25NYXRjaCA9IGFzeW5jIChmaWxlcGF0aCkgPT4ge1xuICAgICAgdHJ5IHtcbiAgICAgICAgY29uc3QgcGtnID0gSlNPTi5wYXJzZShhd2FpdCBmcy5yZWFkRmlsZShmaWxlcGF0aCwgJ3V0ZjgnKSk7XG4gICAgICAgIGlmIChpc0RyaXZlcihwa2cpIHx8IGlzUGx1Z2luKHBrZykpIHtcbiAgICAgICAgICBjb25zdCBjaGFuZ2VkID0gdGhpcy5hZGRFeHRlbnNpb25Gcm9tUGFja2FnZShwa2csIGZpbGVwYXRoKTtcbiAgICAgICAgICBkaWRDaGFuZ2UgPSBkaWRDaGFuZ2UgfHwgY2hhbmdlZDtcbiAgICAgICAgfVxuICAgICAgfSBjYXRjaCB7fVxuICAgIH07XG5cbiAgICAvKipcbiAgICAgKiBBIGxpc3Qgb2YgYFByb21pc2VgcyB3aGljaCByZWFkIGBwYWNrYWdlLmpzb25gIGZpbGVzIGxvb2tpbmcgZm9yIEFwcGl1bSBleHRlbnNpb25zLlxuICAgICAqIEB0eXBlIHtQcm9taXNlPHZvaWQ+W119XG4gICAgICovXG4gICAgY29uc3QgcXVldWUgPSBbXG4gICAgICAvLyBsb29rIGF0IGBwYWNrYWdlLmpzb25gIGluIGBBUFBJVU1fSE9NRWAgb25seVxuICAgICAgb25NYXRjaChwYXRoLmpvaW4odGhpcy5fYXBwaXVtSG9tZSwgJ3BhY2thZ2UuanNvbicpKSxcbiAgICBdO1xuXG4gICAgLy8gYWRkIGRlcGVuZGVuY2llcyB0byB0aGUgcXVldWVcbiAgICBhd2FpdCBuZXcgQigocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICBnbG9iKFxuICAgICAgICAnbm9kZV9tb2R1bGVzL3sqLEAqLyp9L3BhY2thZ2UuanNvbicsXG4gICAgICAgIHtjd2Q6IHRoaXMuX2FwcGl1bUhvbWUsIHNpbGVudDogdHJ1ZSwgYWJzb2x1dGU6IHRydWV9LFxuICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgcHJvbWlzZS9wcmVmZXItYXdhaXQtdG8tY2FsbGJhY2tzXG4gICAgICAgIChlcnIpID0+IHtcbiAgICAgICAgICBpZiAoZXJyKSB7XG4gICAgICAgICAgICByZWplY3QoZXJyKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgcmVzb2x2ZSgpO1xuICAgICAgICB9XG4gICAgICApXG4gICAgICAgIC5vbignZXJyb3InLCByZWplY3QpXG4gICAgICAgIC5vbignbWF0Y2gnLCAoZmlsZXBhdGgpID0+IHtcbiAgICAgICAgICBxdWV1ZS5wdXNoKG9uTWF0Y2goZmlsZXBhdGgpKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICAvLyB3YWl0IGZvciBldmVyeXRoaW5nIHRvIGZpbmlzaFxuICAgIGF3YWl0IEIuYWxsKHF1ZXVlKTtcblxuICAgIHJldHVybiBkaWRDaGFuZ2U7XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyBgdHJ1ZWAgaWYgZHJpdmVyIHdpdGggbmFtZSBgbmFtZWAgaXMgcmVnaXN0ZXJlZC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IG5hbWUgLSBEcml2ZXIgbmFtZVxuICAgKiBAcmV0dXJucyB7Ym9vbGVhbn1cbiAgICovXG4gIGhhc0RyaXZlcihuYW1lKSB7XG4gICAgcmV0dXJuIEJvb2xlYW4odGhpcy5fZGF0YS5kcml2ZXJzW25hbWVdKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIGB0cnVlYCBpZiBwbHVnaW4gd2l0aCBuYW1lIGBuYW1lYCBpcyByZWdpc3RlcmVkLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gbmFtZSAtIFBsdWdpbiBuYW1lXG4gICAqIEByZXR1cm5zIHtib29sZWFufVxuICAgKi9cbiAgaGFzUGx1Z2luKG5hbWUpIHtcbiAgICByZXR1cm4gQm9vbGVhbih0aGlzLl9kYXRhLnBsdWdpbnNbbmFtZV0pO1xuICB9XG5cbiAgLyoqXG4gICAqIEdpdmVuIGEgcGF0aCB0byBhIGBwYWNrYWdlLmpzb25gLCBhZGQgaXQgYXMgZWl0aGVyIGEgZHJpdmVyIG9yIHBsdWdpbiB0byB0aGUgbWFuaWZlc3QuXG4gICAqXG4gICAqIFdpbGwgX25vdF8gb3ZlcndyaXRlIGV4aXN0aW5nIGVudHJpZXMuXG4gICAqIEB0ZW1wbGF0ZSB7RXh0ZW5zaW9uVHlwZX0gRXh0VHlwZVxuICAgKiBAcGFyYW0ge0V4dFBhY2thZ2VKc29uPEV4dFR5cGU+fSBwa2dKc29uXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBwa2dQYXRoXG4gICAqIEByZXR1cm5zIHtib29sZWFufSAtIGB0cnVlYCB1cG9uIHN1Y2Nlc3MsIGBmYWxzZWAgaWYgdGhlIGV4dGVuc2lvbiBpcyBhbHJlYWR5IHJlZ2lzdGVyZWQuXG4gICAqL1xuICBhZGRFeHRlbnNpb25Gcm9tUGFja2FnZShwa2dKc29uLCBwa2dQYXRoKSB7XG4gICAgY29uc3QgZXh0ZW5zaW9uUGF0aCA9IHBhdGguZGlybmFtZShwa2dQYXRoKTtcblxuICAgIC8qKlxuICAgICAqIEB0eXBlIHtJbnRlcm5hbE1ldGFkYXRhfVxuICAgICAqL1xuICAgIGNvbnN0IGludGVybmFsID0ge1xuICAgICAgcGtnTmFtZTogcGtnSnNvbi5uYW1lLFxuICAgICAgdmVyc2lvbjogcGtnSnNvbi52ZXJzaW9uLFxuICAgICAgYXBwaXVtVmVyc2lvbjogcGtnSnNvbi5wZWVyRGVwZW5kZW5jaWVzPy5hcHBpdW0sXG4gICAgICBpbnN0YWxsVHlwZTogSU5TVEFMTF9UWVBFX05QTSxcbiAgICAgIGluc3RhbGxTcGVjOiBgJHtwa2dKc29uLm5hbWV9QCR7cGtnSnNvbi52ZXJzaW9ufWAsXG4gICAgfTtcblxuICAgIGlmIChpc0RyaXZlcihwa2dKc29uKSkge1xuICAgICAgaWYgKCF0aGlzLmhhc0RyaXZlcihwa2dKc29uLmFwcGl1bS5kcml2ZXJOYW1lKSkge1xuICAgICAgICB0aGlzLmFkZEV4dGVuc2lvbihEUklWRVJfVFlQRSwgcGtnSnNvbi5hcHBpdW0uZHJpdmVyTmFtZSwge1xuICAgICAgICAgIC4uLl8ub21pdChwa2dKc29uLmFwcGl1bSwgJ2RyaXZlck5hbWUnKSxcbiAgICAgICAgICAuLi5pbnRlcm5hbCxcbiAgICAgICAgfSk7XG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH0gZWxzZSBpZiAoaXNQbHVnaW4ocGtnSnNvbikpIHtcbiAgICAgIGlmICghdGhpcy5oYXNQbHVnaW4ocGtnSnNvbi5hcHBpdW0ucGx1Z2luTmFtZSkpIHtcbiAgICAgICAgdGhpcy5hZGRFeHRlbnNpb24oUExVR0lOX1RZUEUsIHBrZ0pzb24uYXBwaXVtLnBsdWdpbk5hbWUsIHtcbiAgICAgICAgICAuLi5fLm9taXQocGtnSnNvbi5hcHBpdW0sICdwbHVnaW5OYW1lJyksXG4gICAgICAgICAgLi4uaW50ZXJuYWwsXG4gICAgICAgIH0pO1xuICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICAgIH1cbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhyb3cgbmV3IFR5cGVFcnJvcihcbiAgICAgICAgYFRoZSBleHRlbnNpb24gaW4gJHtleHRlbnNpb25QYXRofSBpcyBuZWl0aGVyIGEgdmFsaWQgZHJpdmVyIG5vciBhIHZhbGlkIHBsdWdpbi5gXG4gICAgICApO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBBZGRzIGFuIGV4dGVuc2lvbiB0byB0aGUgbWFuaWZlc3QgYXMgd2FzIGluc3RhbGxlZCBieSB0aGUgYGFwcGl1bWAgQ0xJLiAgVGhlXG4gICAqIGBleHREYXRhYCwgYGV4dFR5cGVgLCBhbmQgYGV4dE5hbWVgIGhhdmUgYWxyZWFkeSBiZWVuIGRldGVybWluZWQuXG4gICAqXG4gICAqIFNlZSB7QGxpbmsgTWFuaWZlc3QuYWRkRXh0ZW5zaW9uRnJvbVBhY2thZ2V9IGZvciBhZGRpbmcgYW4gZXh0ZW5zaW9uIGZyb20gYW4gb24tZGlzayBwYWNrYWdlLlxuICAgKiBAdGVtcGxhdGUge0V4dGVuc2lvblR5cGV9IEV4dFR5cGVcbiAgICogQHBhcmFtIHtFeHRUeXBlfSBleHRUeXBlIC0gYGRyaXZlcmAgb3IgYHBsdWdpbmBcbiAgICogQHBhcmFtIHtzdHJpbmd9IGV4dE5hbWUgLSBOYW1lIG9mIGV4dGVuc2lvblxuICAgKiBAcGFyYW0ge0V4dE1hbmlmZXN0PEV4dFR5cGU+fSBleHREYXRhIC0gRXh0ZW5zaW9uIG1ldGFkYXRhXG4gICAqIEByZXR1cm5zIHtFeHRNYW5pZmVzdDxFeHRUeXBlPn0gQSBjbG9uZSBvZiBgZXh0RGF0YWAsIHBvdGVudGlhbGx5IHdpdGggYSBtdXRhdGVkIGBhcHBpdW1WZXJzaW9uYCBmaWVsZFxuICAgKi9cbiAgYWRkRXh0ZW5zaW9uKGV4dFR5cGUsIGV4dE5hbWUsIGV4dERhdGEpIHtcbiAgICBjb25zdCBkYXRhID0gXy5jbG9uZShleHREYXRhKTtcbiAgICB0aGlzLl9kYXRhW2Ake2V4dFR5cGV9c2BdW2V4dE5hbWVdID0gZGF0YTtcbiAgICByZXR1cm4gZGF0YTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIHRoZSBBUFBJVU1fSE9NRSBwYXRoXG4gICAqL1xuICBnZXQgYXBwaXVtSG9tZSgpIHtcbiAgICByZXR1cm4gdGhpcy5fYXBwaXVtSG9tZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIHRoZSBwYXRoIHRvIHRoZSBtYW5pZmVzdCBmaWxlXG4gICAqL1xuICBnZXQgbWFuaWZlc3RQYXRoKCkge1xuICAgIHJldHVybiB0aGlzLl9tYW5pZmVzdFBhdGg7XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyBleHRlbnNpb24gZGF0YSBmb3IgYSBwYXJ0aWN1bGFyIHR5cGUuXG4gICAqXG4gICAqIEB0ZW1wbGF0ZSB7RXh0ZW5zaW9uVHlwZX0gRXh0VHlwZVxuICAgKiBAcGFyYW0ge0V4dFR5cGV9IGV4dFR5cGVcbiAgICogQHJldHVybnMge0V4dFJlY29yZDxFeHRUeXBlPn1cbiAgICovXG4gIGdldEV4dGVuc2lvbkRhdGEoZXh0VHlwZSkge1xuICAgIHJldHVybiB0aGlzLl9kYXRhWy8qKiBAdHlwZSB7c3RyaW5nfSAqLyAoYCR7ZXh0VHlwZX1zYCldO1xuICB9XG5cbiAgLyoqXG4gICAqIFJlYWRzIG1hbmlmZXN0IGZyb20gZGlzayBhbmQgX292ZXJ3cml0ZXNfIHRoZSBpbnRlcm5hbCBkYXRhLlxuICAgKlxuICAgKiBJZiB0aGUgbWFuaWZlc3QgZG9lcyBub3QgZXhpc3Qgb24gZGlzaywgYW4ge0BsaW5rIElOSVRJQUxfTUFOSUZFU1RfREFUQSBcImVtcHR5XCJ9IG1hbmlmZXN0IGZpbGUgd2lsbCBiZSBjcmVhdGVkLlxuICAgKlxuICAgKiBJZiBgQVBQSVVNX0hPTUVgIGNvbnRhaW5zIGEgYHBhY2thZ2UuanNvbmAgd2l0aCBhbiBgYXBwaXVtYCBkZXBlbmRlbmN5LCB0aGVuIGEgaGFzaCBvZiB0aGUgYHBhY2thZ2UuanNvbmAgd2lsbCBiZSB0YWtlbi4gSWYgdGhpcyBoYXNoIGRpZmZlcnMgZnJvbSB0aGUgbGFzdCBoYXNoLCB0aGUgY29udGVudHMgb2YgYEFQUElVTV9IT01FL25vZGVfbW9kdWxlc2Agd2lsbCBiZSBzY2FubmVkIGZvciBleHRlbnNpb25zIHRoYXQgbWF5IGhhdmUgYmVlbiBpbnN0YWxsZWQgb3V0c2lkZSBvZiB0aGUgYGFwcGl1bWAgQ0xJLiAgQW55IGZvdW5kIGV4dGVuc2lvbnMgd2lsbCBiZSBhZGRlZCB0byB0aGUgbWFuaWZlc3QgZmlsZSwgYW5kIGlmIHNvLCB0aGUgbWFuaWZlc3QgZmlsZSB3aWxsIGJlIHdyaXR0ZW4gdG8gZGlzay5cbiAgICpcbiAgICogT25seSBvbmUgcmVhZCBvcGVyYXRpb24gc2hvdWxkIGhhcHBlbiBhdCBhIHRpbWUuICBUaGlzIGlzIGNvbnRyb2xsZWQgdmlhIHRoZSB7QGxpbmsgTWFuaWZlc3QuX3JlYWRpbmd9IHByb3BlcnR5LlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTxNYW5pZmVzdERhdGE+fSBUaGUgZGF0YVxuICAgKi9cbiAgYXN5bmMgcmVhZCgpIHtcbiAgICBpZiAodGhpcy5fcmVhZGluZykge1xuICAgICAgYXdhaXQgdGhpcy5fcmVhZGluZztcbiAgICAgIHJldHVybiB0aGlzLl9kYXRhO1xuICAgIH1cblxuICAgIHRoaXMuX3JlYWRpbmcgPSAoYXN5bmMgKCkgPT4ge1xuICAgICAgLyoqIEB0eXBlIHtNYW5pZmVzdERhdGF9ICovXG4gICAgICBsZXQgZGF0YTtcbiAgICAgIGxldCBpc05ld0ZpbGUgPSBmYWxzZTtcbiAgICAgIGF3YWl0IHRoaXMuX3NldE1hbmlmZXN0UGF0aCgpO1xuICAgICAgdHJ5IHtcbiAgICAgICAgbG9nLmRlYnVnKGBSZWFkaW5nICR7dGhpcy5fbWFuaWZlc3RQYXRofS4uLmApO1xuICAgICAgICBjb25zdCB5YW1sID0gYXdhaXQgZnMucmVhZEZpbGUodGhpcy5fbWFuaWZlc3RQYXRoLCAndXRmOCcpO1xuICAgICAgICBkYXRhID0gWUFNTC5wYXJzZSh5YW1sKTtcbiAgICAgICAgbG9nLmRlYnVnKGBQYXJzZWQgbWFuaWZlc3QgZmlsZTogJHtKU09OLnN0cmluZ2lmeShkYXRhLCBudWxsLCAyKX1gKTtcbiAgICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgICBpZiAoZXJyLmNvZGUgPT09ICdFTk9FTlQnKSB7XG4gICAgICAgICAgZGF0YSA9IF8uY2xvbmVEZWVwKElOSVRJQUxfTUFOSUZFU1RfREFUQSk7XG4gICAgICAgICAgaXNOZXdGaWxlID0gdHJ1ZTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBpZiAodGhpcy5fbWFuaWZlc3RQYXRoKSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICAgIGBBcHBpdW0gaGFkIHRyb3VibGUgbG9hZGluZyB0aGUgZXh0ZW5zaW9uIGluc3RhbGxhdGlvbiBgICtcbiAgICAgICAgICAgICAgICBgY2FjaGUgZmlsZSAoJHt0aGlzLl9tYW5pZmVzdFBhdGh9KS4gSXQgbWF5IGJlIGludmFsaWQgWUFNTC4gU3BlY2lmaWMgZXJyb3I6ICR7ZXJyLm1lc3NhZ2V9YFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAgICAgICBgQXBwaXVtIGVuY291bnRlcmVkIGFuIHVua25vd24gcHJvYmxlbS4gU3BlY2lmaWMgZXJyb3I6ICR7ZXJyLm1lc3NhZ2V9YFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgdGhpcy5fZGF0YSA9IGRhdGE7XG4gICAgICBsZXQgaW5zdGFsbGVkRXh0ZW5zaW9uc0NoYW5nZWQgPSBmYWxzZTtcbiAgICAgIGlmIChcbiAgICAgICAgKGF3YWl0IGVudi5oYXNBcHBpdW1EZXBlbmRlbmN5KHRoaXMuYXBwaXVtSG9tZSkpICYmXG4gICAgICAgIChhd2FpdCBwYWNrYWdlRGlkQ2hhbmdlKHRoaXMuYXBwaXVtSG9tZSkpXG4gICAgICApIHtcbiAgICAgICAgaW5zdGFsbGVkRXh0ZW5zaW9uc0NoYW5nZWQgPSBhd2FpdCB0aGlzLnN5bmNXaXRoSW5zdGFsbGVkRXh0ZW5zaW9ucygpO1xuICAgICAgfVxuXG4gICAgICBpZiAoaXNOZXdGaWxlIHx8IGluc3RhbGxlZEV4dGVuc2lvbnNDaGFuZ2VkKSB7XG4gICAgICAgIGF3YWl0IHRoaXMud3JpdGUoKTtcbiAgICAgIH1cbiAgICB9KSgpO1xuICAgIHRyeSB7XG4gICAgICBhd2FpdCB0aGlzLl9yZWFkaW5nO1xuICAgICAgcmV0dXJuIHRoaXMuX2RhdGE7XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIHRoaXMuX3JlYWRpbmcgPSB1bmRlZmluZWQ7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEVuc3VyZXMge0BsaW5rIE1hbmlmZXN0Ll9tYW5pZmVzdFBhdGh9IGlzIHNldC5cbiAgICpcbiAgICogQ3JlYXRlcyB0aGUgZGlyZWN0b3J5IGlmIG5lY2Vzc2FyeS5cbiAgICogQHByaXZhdGVcbiAgICogQHJldHVybnMge1Byb21pc2U8c3RyaW5nPn1cbiAgICovXG4gIGFzeW5jIF9zZXRNYW5pZmVzdFBhdGgoKSB7XG4gICAgaWYgKCF0aGlzLl9tYW5pZmVzdFBhdGgpIHtcbiAgICAgIHRoaXMuX21hbmlmZXN0UGF0aCA9IGF3YWl0IGVudi5yZXNvbHZlTWFuaWZlc3RQYXRoKHRoaXMuX2FwcGl1bUhvbWUpO1xuXG4gICAgICAvKiBpc3RhbmJ1bCBpZ25vcmUgaWYgKi9cbiAgICAgIGlmIChwYXRoLnJlbGF0aXZlKHRoaXMuX2FwcGl1bUhvbWUsIHRoaXMuX21hbmlmZXN0UGF0aCkuc3RhcnRzV2l0aCgnLicpKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICBgTWlzbWF0Y2ggYmV0d2VlbiBsb2NhdGlvbiBvZiBBUFBJVU1fSE9NRSBhbmQgbWFuaWZlc3QgZmlsZS4gQVBQSVVNX0hPTUU6ICR7dGhpcy5hcHBpdW1Ib21lfSwgbWFuaWZlc3QgZmlsZTogJHt0aGlzLl9tYW5pZmVzdFBhdGh9YFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiB0aGlzLl9tYW5pZmVzdFBhdGg7XG4gIH1cblxuICAvKipcbiAgICogV3JpdGVzIHRoZSBkYXRhIGlmIGl0IG5lZWQgcyB3cml0aW5nLlxuICAgKlxuICAgKiBJZiB0aGUgYHNjaGVtYVJldmAgcHJvcCBuZWVkcyB1cGRhdGluZywgdGhlIGZpbGUgd2lsbCBiZSB3cml0dGVuLlxuICAgKlxuICAgKiBAdG9kbyBJZiB0aGlzIGJlY29tZXMgdG9vIG11Y2ggb2YgYSBib3R0bGVuZWNrLCB0aHJvdHRsZSBpdC5cbiAgICogQHJldHVybnMge1Byb21pc2U8Ym9vbGVhbj59IFdoZXRoZXIgdGhlIGRhdGEgd2FzIHdyaXR0ZW5cbiAgICovXG4gIGFzeW5jIHdyaXRlKCkge1xuICAgIGlmICh0aGlzLl93cml0aW5nKSB7XG4gICAgICByZXR1cm4gdGhpcy5fd3JpdGluZztcbiAgICB9XG4gICAgdGhpcy5fd3JpdGluZyA9IChhc3luYyAoKSA9PiB7XG4gICAgICBhd2FpdCB0aGlzLl9zZXRNYW5pZmVzdFBhdGgoKTtcbiAgICAgIHRyeSB7XG4gICAgICAgIGF3YWl0IGZzLm1rZGlycChwYXRoLmRpcm5hbWUodGhpcy5fbWFuaWZlc3RQYXRoKSk7XG4gICAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAgIGBBcHBpdW0gY291bGQgbm90IGNyZWF0ZSB0aGUgZGlyZWN0b3J5IGZvciB0aGUgbWFuaWZlc3QgZmlsZTogJHtwYXRoLmRpcm5hbWUoXG4gICAgICAgICAgICB0aGlzLl9tYW5pZmVzdFBhdGhcbiAgICAgICAgICApfS4gT3JpZ2luYWwgZXJyb3I6ICR7ZXJyLm1lc3NhZ2V9YFxuICAgICAgICApO1xuICAgICAgfVxuICAgICAgdHJ5IHtcbiAgICAgICAgYXdhaXQgZnMud3JpdGVGaWxlKHRoaXMuX21hbmlmZXN0UGF0aCwgWUFNTC5zdHJpbmdpZnkodGhpcy5fZGF0YSksICd1dGY4Jyk7XG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgfSBjYXRjaCAoZXJyKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICBgQXBwaXVtIGNvdWxkIG5vdCB3cml0ZSB0byBtYW5pZmVzdCBhdCAke3RoaXMuX21hbmlmZXN0UGF0aH0gdXNpbmcgQVBQSVVNX0hPTUUgJHt0aGlzLl9hcHBpdW1Ib21lfS4gYCArXG4gICAgICAgICAgICBgUGxlYXNlIGVuc3VyZSBpdCBpcyB3cml0YWJsZS4gT3JpZ2luYWwgZXJyb3I6ICR7ZXJyLm1lc3NhZ2V9YFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH0pKCk7XG4gICAgdHJ5IHtcbiAgICAgIHJldHVybiBhd2FpdCB0aGlzLl93cml0aW5nO1xuICAgIH0gZmluYWxseSB7XG4gICAgICB0aGlzLl93cml0aW5nID0gdW5kZWZpbmVkO1xuICAgIH1cbiAgfVxufVxuXG4vKipcbiAqIFR5cGUgb2YgdGhlIHN0cmluZyByZWZlcnJpbmcgdG8gYSBkcml2ZXIgKHR5cGljYWxseSBhcyBhIGtleSBvciB0eXBlIHN0cmluZylcbiAqIEB0eXBlZGVmIHtpbXBvcnQoJ0BhcHBpdW0vdHlwZXMnKS5Ecml2ZXJUeXBlfSBEcml2ZXJUeXBlXG4gKi9cblxuLyoqXG4gKiBUeXBlIG9mIHRoZSBzdHJpbmcgcmVmZXJyaW5nIHRvIGEgcGx1Z2luICh0eXBpY2FsbHkgYXMgYSBrZXkgb3IgdHlwZSBzdHJpbmcpXG4gKiBAdHlwZWRlZiB7aW1wb3J0KCdAYXBwaXVtL3R5cGVzJykuUGx1Z2luVHlwZX0gUGx1Z2luVHlwZVxuICovXG5cbi8qKlxuICogQHR5cGVkZWYgU3luY1dpdGhJbnN0YWxsZWRFeHRlbnNpb25zT3B0c1xuICogQHByb3BlcnR5IHtudW1iZXJ9IFtkZXB0aExpbWl0XSAtIE1heGltdW0gZGVwdGggdG8gcmVjdXJzZSBpbnRvIHN1YmRpcmVjdG9yaWVzXG4gKi9cblxuLyoqXG4gKiBAdHlwZWRlZiB7aW1wb3J0KCdhcHBpdW0vdHlwZXMnKS5NYW5pZmVzdERhdGF9IE1hbmlmZXN0RGF0YVxuICogQHR5cGVkZWYge2ltcG9ydCgnYXBwaXVtL3R5cGVzJykuSW50ZXJuYWxNZXRhZGF0YX0gSW50ZXJuYWxNZXRhZGF0YVxuICovXG5cbi8qKlxuICogQHRlbXBsYXRlIFRcbiAqIEB0eXBlZGVmIHtpbXBvcnQoJ2FwcGl1bS90eXBlcycpLkV4dFBhY2thZ2VKc29uPFQ+fSBFeHRQYWNrYWdlSnNvblxuICovXG5cbi8qKlxuICogQHRlbXBsYXRlIFRcbiAqIEB0eXBlZGVmIHtpbXBvcnQoJ2FwcGl1bS90eXBlcycpLkV4dE1hbmlmZXN0PFQ+fSBFeHRNYW5pZmVzdFxuICovXG5cbi8qKlxuICogQHRlbXBsYXRlIFRcbiAqIEB0eXBlZGVmIHtpbXBvcnQoJ2FwcGl1bS90eXBlcycpLkV4dFJlY29yZDxUPn0gRXh0UmVjb3JkXG4gKi9cblxuLyoqXG4gKiBFaXRoZXIgYGRyaXZlcmAgb3IgYHBsdWdpbmAgcm5cbiAqIEB0eXBlZGVmIHtpbXBvcnQoJ0BhcHBpdW0vdHlwZXMnKS5FeHRlbnNpb25UeXBlfSBFeHRlbnNpb25UeXBlXG4gKi9cbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7O0FBSUE7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7Ozs7QUFLQSxNQUFNQSxpQkFBaUIsR0FBRyxDQUExQjtBQU1BLE1BQU1DLHNCQUFzQixHQUFJLEdBQUVDLHNCQUFZLEdBQTlDO0FBTUEsTUFBTUMsc0JBQXNCLEdBQUksR0FBRUMsc0JBQVksR0FBOUM7QUFLQSxNQUFNQyxxQkFBcUIsR0FBR0MsTUFBTSxDQUFDQyxNQUFQLENBQWM7RUFDMUMsQ0FBQ04sc0JBQUQsR0FBMEJLLE1BQU0sQ0FBQ0MsTUFBUCxDQUFjLEVBQWQsQ0FEZ0I7RUFFMUMsQ0FBQ0osc0JBQUQsR0FBMEJHLE1BQU0sQ0FBQ0MsTUFBUCxDQUFjLEVBQWQsQ0FGZ0I7RUFHMUNDLFNBQVMsRUFBRVI7QUFIK0IsQ0FBZCxDQUE5Qjs7QUFhQSxTQUFTUyxXQUFULENBQXFCQyxLQUFyQixFQUE0QjtFQUMxQixPQUNFQyxlQUFBLENBQUVDLGFBQUYsQ0FBZ0JGLEtBQWhCLEtBQ0FDLGVBQUEsQ0FBRUMsYUFBRixDQUFnQkYsS0FBSyxDQUFDRyxNQUF0QixDQURBLElBRUFGLGVBQUEsQ0FBRUcsUUFBRixDQUFXSixLQUFLLENBQUNLLElBQWpCLENBRkEsSUFHQUosZUFBQSxDQUFFRyxRQUFGLENBQVdKLEtBQUssQ0FBQ00sT0FBakIsQ0FKRjtBQU1EOztBQVNELFNBQVNDLFFBQVQsQ0FBa0JQLEtBQWxCLEVBQXlCO0VBQ3ZCLE9B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
462
+ _Manifest_data = new WeakMap(), _Manifest_appiumHome = new WeakMap(), _Manifest_manifestPath = new WeakMap(), _Manifest_writing = new WeakMap(), _Manifest_reading = new WeakMap(), _Manifest_instances = new WeakSet(), _Manifest_setManifestPath =
463
+ /**
464
+ * Ensures the internal manifest path is set.
465
+ *
466
+ * Creates the directory if necessary.
467
+ * @returns {Promise<string>}
468
+ */
469
+ async function _Manifest_setManifestPath() {
470
+ if (!__classPrivateFieldGet(this, _Manifest_manifestPath, "f")) {
471
+ __classPrivateFieldSet(this, _Manifest_manifestPath, await support_1.env.resolveManifestPath(__classPrivateFieldGet(this, _Manifest_appiumHome, "f")), "f");
472
+ /* istanbul ignore if */
473
+ if (path_1.default.relative(__classPrivateFieldGet(this, _Manifest_appiumHome, "f"), __classPrivateFieldGet(this, _Manifest_manifestPath, "f")).startsWith('.')) {
474
+ throw new Error(`Mismatch between location of APPIUM_HOME and manifest file. APPIUM_HOME: ${this.appiumHome}, manifest file: ${__classPrivateFieldGet(this, _Manifest_manifestPath, "f")}`);
475
+ }
476
+ }
477
+ return __classPrivateFieldGet(this, _Manifest_manifestPath, "f");
478
+ };
479
+ /**
480
+ * Returns a new or existing {@link Manifest} instance, based on the value of `appiumHome`.
481
+ *
482
+ * Maintains one instance per value of `appiumHome`.
483
+ * @param {string} appiumHome - Path to `APPIUM_HOME`
484
+ * @returns {Manifest}
485
+ */
486
+ Manifest.getInstance = lodash_1.default.memoize(function _getInstance(appiumHome) {
487
+ return new Manifest(appiumHome);
488
+ });
489
+ /**
490
+ * Type of the string referring to a driver (typically as a key or type string)
491
+ * @typedef {import('@appium/types').DriverType} DriverType
492
+ */
493
+ /**
494
+ * Type of the string referring to a plugin (typically as a key or type string)
495
+ * @typedef {import('@appium/types').PluginType} PluginType
496
+ */
497
+ /**
498
+ * @typedef SyncWithInstalledExtensionsOpts
499
+ * @property {number} [depthLimit] - Maximum depth to recurse into subdirectories
500
+ */
501
+ /**
502
+ * @typedef {import('appium/types').ManifestData} ManifestData
503
+ * @typedef {import('appium/types').InternalMetadata} InternalMetadata
504
+ */
505
+ /**
506
+ * @template {ExtensionType} ExtType
507
+ * @typedef {import('appium/types').ExtPackageJson<ExtType>} ExtPackageJson
508
+ */
509
+ /**
510
+ * @template {ExtensionType} ExtType
511
+ * @typedef {import('appium/types').ExtManifest<ExtType>} ExtManifest
512
+ */
513
+ /**
514
+ * @template {ExtensionType} ExtType
515
+ * @typedef {import('appium/types').ExtRecord<ExtType>} ExtRecord
516
+ */
517
+ /**
518
+ * Either `driver` or `plugin` rn
519
+ * @typedef {import('@appium/types').ExtensionType} ExtensionType
520
+ */
521
+ //# sourceMappingURL=manifest.js.map