@appium/base-driver 8.1.2 → 8.2.3

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 (70) hide show
  1. package/build/lib/basedriver/capabilities.js +3 -1
  2. package/build/lib/basedriver/commands/index.js +2 -4
  3. package/build/lib/basedriver/driver.js +8 -10
  4. package/build/lib/basedriver/helpers.js +140 -82
  5. package/build/lib/express/express-logging.js +2 -2
  6. package/build/lib/index.js +126 -0
  7. package/build/lib/jsonwp-proxy/protocol-converter.js +2 -5
  8. package/build/lib/jsonwp-proxy/proxy.js +2 -5
  9. package/build/lib/protocol/errors.js +4 -2
  10. package/build/lib/protocol/helpers.js +3 -20
  11. package/build/lib/protocol/index.js +13 -1
  12. package/build/lib/protocol/protocol.js +26 -25
  13. package/build/lib/protocol/routes.js +1 -10
  14. package/build/test/basedriver/capability-specs.js +10 -10
  15. package/build/test/basedriver/commands/event-specs.js +10 -10
  16. package/build/test/basedriver/driver-e2e-specs.js +3 -3
  17. package/build/test/basedriver/driver-e2e-tests.js +6 -223
  18. package/build/test/basedriver/driver-specs.js +3 -3
  19. package/build/test/basedriver/driver-tests.js +6 -6
  20. package/build/test/basedriver/helpers-specs.js +5 -1
  21. package/build/test/basedriver/timeout-specs.js +7 -7
  22. package/build/test/basedriver/websockets-e2e-specs.js +5 -5
  23. package/build/test/express/server-e2e-specs.js +156 -0
  24. package/build/test/express/server-specs.js +151 -0
  25. package/build/test/express/static-specs.js +23 -0
  26. package/build/test/helpers.js +57 -0
  27. package/build/test/jsonwp-proxy/mock-request.js +93 -0
  28. package/build/test/jsonwp-proxy/protocol-converter-specs.js +173 -0
  29. package/build/test/jsonwp-proxy/proxy-e2e-specs.js +61 -0
  30. package/build/test/jsonwp-proxy/proxy-specs.js +294 -0
  31. package/build/test/jsonwp-proxy/url-specs.js +167 -0
  32. package/build/test/jsonwp-status/status-specs.js +36 -0
  33. package/build/test/protocol/errors-specs.js +388 -0
  34. package/build/test/protocol/fake-driver.js +168 -0
  35. package/build/test/protocol/helpers.js +27 -0
  36. package/build/test/protocol/protocol-e2e-specs.js +1182 -0
  37. package/build/test/protocol/routes-specs.js +82 -0
  38. package/build/test/protocol/validator-specs.js +151 -0
  39. package/index.d.ts +5 -3
  40. package/index.js +1 -62
  41. package/lib/basedriver/capabilities.js +3 -0
  42. package/lib/basedriver/commands/index.js +0 -2
  43. package/lib/basedriver/driver.js +6 -26
  44. package/lib/basedriver/helpers.js +202 -85
  45. package/lib/express/express-logging.js +1 -1
  46. package/lib/index.js +64 -0
  47. package/lib/jsonwp-proxy/protocol-converter.js +1 -5
  48. package/lib/jsonwp-proxy/proxy.js +1 -3
  49. package/lib/protocol/errors.js +1 -1
  50. package/lib/protocol/helpers.js +5 -25
  51. package/lib/protocol/index.js +3 -1
  52. package/lib/protocol/protocol.js +26 -31
  53. package/lib/protocol/routes.js +0 -3
  54. package/package.json +8 -16
  55. package/test/basedriver/capability-specs.js +1 -1
  56. package/test/basedriver/commands/event-specs.js +1 -1
  57. package/test/basedriver/driver-e2e-specs.js +1 -1
  58. package/test/basedriver/driver-e2e-tests.js +1 -179
  59. package/test/basedriver/driver-specs.js +1 -1
  60. package/test/basedriver/driver-tests.js +3 -3
  61. package/test/basedriver/helpers-specs.js +4 -0
  62. package/test/basedriver/timeout-specs.js +1 -1
  63. package/test/basedriver/websockets-e2e-specs.js +1 -1
  64. package/build/index.js +0 -120
  65. package/build/lib/basedriver/commands/execute-child.js +0 -137
  66. package/build/lib/basedriver/commands/execute.js +0 -119
  67. package/build/test/basedriver/fixtures/custom-element-finder-bad.js +0 -12
  68. package/build/test/basedriver/fixtures/custom-element-finder.js +0 -36
  69. package/lib/basedriver/commands/execute-child.js +0 -132
  70. package/lib/basedriver/commands/execute.js +0 -126
@@ -35,16 +35,16 @@ const CACHED_APPS_MAX_AGE = 1000 * 60 * 60 * 24;
35
35
  const APPLICATIONS_CACHE = new _lruCache.default({
36
36
  maxAge: CACHED_APPS_MAX_AGE,
37
37
  updateAgeOnGet: true,
38
- dispose: async (app, {
38
+ dispose: (app, {
39
39
  fullPath
40
40
  }) => {
41
- if (!(await _support.fs.exists(fullPath))) {
42
- return;
43
- }
44
-
45
- _logger.default.info(`The application '${app}' cached at '${fullPath}' has expired`);
41
+ _logger.default.info(`The application '${app}' cached at '${fullPath}' has ` + `expired after ${CACHED_APPS_MAX_AGE}ms`);
46
42
 
47
- await _support.fs.rimraf(fullPath);
43
+ setTimeout(async () => {
44
+ if (fullPath) {
45
+ await _support.fs.rimraf(fullPath);
46
+ }
47
+ });
48
48
  },
49
49
  noDisposeOnSet: true
50
50
  });
@@ -86,80 +86,116 @@ async function retrieveHeaders(link) {
86
86
  return {};
87
87
  }
88
88
 
89
- function getCachedApplicationPath(link, currentAppProps = {}) {
89
+ function getCachedApplicationPath(link, currentAppProps = {}, cachedAppInfo = {}) {
90
90
  const refresh = () => {
91
91
  _logger.default.debug(`A fresh copy of the application is going to be downloaded from ${link}`);
92
92
 
93
93
  return null;
94
94
  };
95
95
 
96
- if (APPLICATIONS_CACHE.has(link)) {
97
- const {
98
- lastModified: currentModified,
99
- immutable: currentImmutable,
100
- maxAge: currentMaxAge
101
- } = currentAppProps;
102
- const {
103
- lastModified,
104
- immutable,
105
- timestamp,
106
- fullPath
107
- } = APPLICATIONS_CACHE.get(link);
108
-
109
- if (lastModified && currentModified) {
110
- if (currentModified.getTime() <= lastModified.getTime()) {
111
- _logger.default.debug(`The application at ${link} has not been modified since ${lastModified}`);
112
-
113
- return fullPath;
114
- }
115
-
116
- _logger.default.debug(`The application at ${link} has been modified since ${lastModified}`);
96
+ if (!_lodash.default.isPlainObject(cachedAppInfo) || !_lodash.default.isPlainObject(currentAppProps)) {
97
+ return refresh();
98
+ }
117
99
 
118
- return refresh();
119
- }
100
+ const {
101
+ lastModified: currentModified,
102
+ immutable: currentImmutable,
103
+ maxAge: currentMaxAge
104
+ } = currentAppProps;
105
+ const {
106
+ lastModified,
107
+ immutable,
108
+ timestamp,
109
+ fullPath
110
+ } = cachedAppInfo;
120
111
 
121
- if (immutable && currentImmutable) {
122
- _logger.default.debug(`The application at ${link} is immutable`);
112
+ if (lastModified && currentModified) {
113
+ if (currentModified.getTime() <= lastModified.getTime()) {
114
+ _logger.default.debug(`The application at ${link} has not been modified since ${lastModified}`);
123
115
 
124
116
  return fullPath;
125
117
  }
126
118
 
127
- if (currentMaxAge && timestamp) {
128
- const msLeft = timestamp + currentMaxAge * 1000 - Date.now();
119
+ _logger.default.debug(`The application at ${link} has been modified since ${lastModified}`);
129
120
 
130
- if (msLeft > 0) {
131
- _logger.default.debug(`The cached application '${_path.default.basename(fullPath)}' will expire in ${msLeft / 1000}s`);
121
+ return refresh();
122
+ }
132
123
 
133
- return fullPath;
134
- }
124
+ if (immutable && currentImmutable) {
125
+ _logger.default.debug(`The application at ${link} is immutable`);
135
126
 
136
- _logger.default.debug(`The cached application '${_path.default.basename(fullPath)}' has expired`);
127
+ return fullPath;
128
+ }
129
+
130
+ if (currentMaxAge && timestamp) {
131
+ const msLeft = timestamp + currentMaxAge * 1000 - Date.now();
132
+
133
+ if (msLeft > 0) {
134
+ _logger.default.debug(`The cached application '${_path.default.basename(fullPath)}' will expire in ${msLeft / 1000}s`);
135
+
136
+ return fullPath;
137
137
  }
138
+
139
+ _logger.default.debug(`The cached application '${_path.default.basename(fullPath)}' has expired`);
138
140
  }
139
141
 
140
142
  return refresh();
141
143
  }
142
144
 
143
145
  function verifyAppExtension(app, supportedAppExtensions) {
144
- if (supportedAppExtensions.includes(_path.default.extname(app))) {
146
+ if (supportedAppExtensions.map(_lodash.default.toLower).includes(_lodash.default.toLower(_path.default.extname(app)))) {
145
147
  return app;
146
148
  }
147
149
 
148
150
  throw new Error(`New app path '${app}' did not have ` + `${_support.util.pluralize('extension', supportedAppExtensions.length, false)}: ` + supportedAppExtensions);
149
151
  }
150
152
 
151
- async function configureApp(app, supportedAppExtensions) {
153
+ async function calculateFolderIntegrity(folderPath) {
154
+ return (await _support.fs.glob('**/*', {
155
+ cwd: folderPath,
156
+ strict: false,
157
+ nosort: true
158
+ })).length;
159
+ }
160
+
161
+ async function calculateFileIntegrity(filePath) {
162
+ return await _support.fs.hash(filePath);
163
+ }
164
+
165
+ async function isAppIntegrityOk(currentPath, expectedIntegrity = {}) {
166
+ if (!(await _support.fs.exists(currentPath))) {
167
+ return false;
168
+ }
169
+
170
+ return (await _support.fs.stat(currentPath)).isDirectory() ? (await calculateFolderIntegrity(currentPath)) >= (expectedIntegrity === null || expectedIntegrity === void 0 ? void 0 : expectedIntegrity.folder) : (await calculateFileIntegrity(currentPath)) === (expectedIntegrity === null || expectedIntegrity === void 0 ? void 0 : expectedIntegrity.file);
171
+ }
172
+
173
+ async function configureApp(app, options = {}) {
152
174
  if (!_lodash.default.isString(app)) {
153
175
  return;
154
176
  }
155
177
 
156
- if (!_lodash.default.isArray(supportedAppExtensions)) {
157
- supportedAppExtensions = [supportedAppExtensions];
178
+ let supportedAppExtensions;
179
+ const {
180
+ onPostProcess
181
+ } = _lodash.default.isPlainObject(options) ? options : {};
182
+
183
+ if (_lodash.default.isString(options)) {
184
+ supportedAppExtensions = [options];
185
+ } else if (_lodash.default.isArray(options)) {
186
+ supportedAppExtensions = options;
187
+ } else if (_lodash.default.isPlainObject(options)) {
188
+ supportedAppExtensions = options.supportedExtensions;
189
+ }
190
+
191
+ if (_lodash.default.isEmpty(supportedAppExtensions)) {
192
+ throw new Error(`One or more supported app extensions must be provided`);
158
193
  }
159
194
 
160
195
  let newApp = app;
161
196
  let shouldUnzipApp = false;
162
- let archiveHash = null;
197
+ let packageHash = null;
198
+ let headers = null;
163
199
  const remoteAppProps = {
164
200
  lastModified: null,
165
201
  immutable: false,
@@ -172,11 +208,12 @@ async function configureApp(app, supportedAppExtensions) {
172
208
  } = _url.default.parse(newApp);
173
209
 
174
210
  const isUrl = ['http:', 'https:'].includes(protocol);
211
+ const cachedAppInfo = APPLICATIONS_CACHE.get(app);
175
212
  return await APPLICATIONS_CACHE_GUARD.acquire(app, async () => {
176
213
  if (isUrl) {
177
214
  _logger.default.info(`Using downloadable app '${newApp}'`);
178
215
 
179
- const headers = await retrieveHeaders(newApp);
216
+ headers = await retrieveHeaders(newApp);
180
217
 
181
218
  if (!_lodash.default.isEmpty(headers)) {
182
219
  if (headers['last-modified']) {
@@ -197,16 +234,16 @@ async function configureApp(app, supportedAppExtensions) {
197
234
  _logger.default.debug(`Cache-Control: ${headers['cache-control']}`);
198
235
  }
199
236
 
200
- const cachedPath = getCachedApplicationPath(app, remoteAppProps);
237
+ const cachedPath = getCachedApplicationPath(app, remoteAppProps, cachedAppInfo);
201
238
 
202
239
  if (cachedPath) {
203
- if (await _support.fs.exists(cachedPath)) {
240
+ if (await isAppIntegrityOk(cachedPath, cachedAppInfo === null || cachedAppInfo === void 0 ? void 0 : cachedAppInfo.integrity)) {
204
241
  _logger.default.info(`Reusing previously downloaded application at '${cachedPath}'`);
205
242
 
206
243
  return verifyAppExtension(cachedPath, supportedAppExtensions);
207
244
  }
208
245
 
209
- _logger.default.info(`The application at '${cachedPath}' does not exist anymore. Deleting it from the cache`);
246
+ _logger.default.info(`The application at '${cachedPath}' does not exist anymore ` + `or its integrity has been damaged. Deleting it from the internal cache`);
210
247
 
211
248
  APPLICATIONS_CACHE.del(app);
212
249
  }
@@ -283,16 +320,21 @@ async function configureApp(app, supportedAppExtensions) {
283
320
  throw new Error(errorMessage);
284
321
  }
285
322
 
286
- if (shouldUnzipApp) {
323
+ const isPackageAFile = (await _support.fs.stat(newApp)).isFile();
324
+
325
+ if (isPackageAFile) {
326
+ packageHash = await calculateFileIntegrity(newApp);
327
+ }
328
+
329
+ if (isPackageAFile && shouldUnzipApp && !_lodash.default.isFunction(onPostProcess)) {
287
330
  const archivePath = newApp;
288
- archiveHash = await _support.fs.hash(archivePath);
289
331
 
290
- if (APPLICATIONS_CACHE.has(app) && archiveHash === APPLICATIONS_CACHE.get(app).hash) {
332
+ if (packageHash === (cachedAppInfo === null || cachedAppInfo === void 0 ? void 0 : cachedAppInfo.packageHash)) {
291
333
  const {
292
334
  fullPath
293
- } = APPLICATIONS_CACHE.get(app);
335
+ } = cachedAppInfo;
294
336
 
295
- if (await _support.fs.exists(fullPath)) {
337
+ if (await isAppIntegrityOk(fullPath, cachedAppInfo === null || cachedAppInfo === void 0 ? void 0 : cachedAppInfo.integrity)) {
296
338
  if (archivePath !== app) {
297
339
  await _support.fs.rimraf(archivePath);
298
340
  }
@@ -302,7 +344,7 @@ async function configureApp(app, supportedAppExtensions) {
302
344
  return verifyAppExtension(fullPath, supportedAppExtensions);
303
345
  }
304
346
 
305
- _logger.default.info(`The application at '${fullPath}' does not exist anymore. Deleting it from the cache`);
347
+ _logger.default.info(`The application at '${fullPath}' does not exist anymore ` + `or its integrity has been damaged. Deleting it from the cache`);
306
348
 
307
349
  APPLICATIONS_CACHE.del(app);
308
350
  }
@@ -326,27 +368,42 @@ async function configureApp(app, supportedAppExtensions) {
326
368
  app = newApp;
327
369
  }
328
370
 
329
- verifyAppExtension(newApp, supportedAppExtensions);
371
+ const storeAppInCache = async appPathToCache => {
372
+ const cachedFullPath = cachedAppInfo === null || cachedAppInfo === void 0 ? void 0 : cachedAppInfo.fullPath;
330
373
 
331
- if (app !== newApp && (archiveHash || _lodash.default.values(remoteAppProps).some(Boolean))) {
332
- if (APPLICATIONS_CACHE.has(app)) {
333
- const {
334
- fullPath
335
- } = APPLICATIONS_CACHE.get(app);
374
+ if (cachedFullPath && cachedFullPath !== appPathToCache) {
375
+ await _support.fs.rimraf(cachedFullPath);
376
+ }
336
377
 
337
- if (fullPath !== newApp && (await _support.fs.exists(fullPath))) {
338
- await _support.fs.rimraf(fullPath);
339
- }
378
+ const integrity = {};
379
+
380
+ if ((await _support.fs.stat(appPathToCache)).isDirectory()) {
381
+ integrity.folder = await calculateFolderIntegrity(appPathToCache);
382
+ } else {
383
+ integrity.file = await calculateFileIntegrity(appPathToCache);
340
384
  }
341
385
 
342
386
  APPLICATIONS_CACHE.set(app, { ...remoteAppProps,
343
387
  timestamp: Date.now(),
344
- hash: archiveHash,
345
- fullPath: newApp
388
+ packageHash,
389
+ integrity,
390
+ fullPath: appPathToCache
391
+ });
392
+ return appPathToCache;
393
+ };
394
+
395
+ if (_lodash.default.isFunction(onPostProcess)) {
396
+ const result = await onPostProcess({
397
+ cachedAppInfo: _lodash.default.clone(cachedAppInfo),
398
+ isUrl,
399
+ headers: _lodash.default.clone(headers),
400
+ appPath: newApp
346
401
  });
402
+ return !(result !== null && result !== void 0 && result.appPath) || app === (result === null || result === void 0 ? void 0 : result.appPath) || !(await _support.fs.exists(result === null || result === void 0 ? void 0 : result.appPath)) ? newApp : await storeAppInCache(result.appPath);
347
403
  }
348
404
 
349
- return newApp;
405
+ verifyAppExtension(newApp, supportedAppExtensions);
406
+ return app !== newApp && (packageHash || _lodash.default.values(remoteAppProps).some(Boolean)) ? await storeAppInCache(newApp) : newApp;
350
407
  });
351
408
  }
352
409
 
@@ -379,8 +436,10 @@ async function unzipApp(zipPath, dstRoot, supportedAppExtensions) {
379
436
  _logger.default.debug(`Unzipping '${zipPath}'`);
380
437
 
381
438
  const timer = new _support.timing.Timer().start();
439
+ const useSystemUnzipEnv = process.env.APPIUM_PREFER_SYSTEM_UNZIP;
440
+ const useSystemUnzip = _lodash.default.isEmpty(useSystemUnzipEnv) || !['0', 'false'].includes(_lodash.default.toLower(useSystemUnzipEnv));
382
441
  const extractionOpts = {
383
- useSystemUnzip: !_support.system.isWindows()
442
+ useSystemUnzip
384
443
  };
385
444
 
386
445
  if (_path.default.extname(zipPath) === IPA_EXT) {
@@ -390,22 +449,21 @@ async function unzipApp(zipPath, dstRoot, supportedAppExtensions) {
390
449
  }
391
450
 
392
451
  await _support.zip.extractAllTo(zipPath, tmpRoot, extractionOpts);
393
- const duration = timer.getDuration();
394
- const allExtractedItems = await _support.fs.glob('**', {
395
- cwd: tmpRoot
396
- });
397
-
398
- _logger.default.debug(`Extracted ${_support.util.pluralize('item', allExtractedItems.length, true)} ` + `from '${zipPath}' in ${Math.round(duration.asMilliSeconds)}ms`);
399
-
400
- const allBundleItems = allExtractedItems.filter(relativePath => supportedAppExtensions.includes(_path.default.extname(relativePath))).sort((a, b) => a.split(_path.default.sep).length - b.split(_path.default.sep).length);
401
-
402
- if (_lodash.default.isEmpty(allBundleItems)) {
403
- throw new Error(`App zip unzipped OK, but we could not find '${supportedAppExtensions}' ` + _support.util.pluralize('bundle', supportedAppExtensions.length, false) + ` in it. Make sure your archive contains at least one package having ` + `'${supportedAppExtensions}' ${_support.util.pluralize('extension', supportedAppExtensions.length, false)}`);
452
+ const globPattern = `**/*.+(${supportedAppExtensions.map(ext => ext.replace(/^\./, '')).join('|')})`;
453
+ const sortedBundleItems = (await _support.fs.glob(globPattern, {
454
+ cwd: tmpRoot,
455
+ strict: false
456
+ })).sort((a, b) => a.split(_path.default.sep).length - b.split(_path.default.sep).length);
457
+
458
+ if (_lodash.default.isEmpty(sortedBundleItems)) {
459
+ _logger.default.errorAndThrow(`App unzipped OK, but we could not find any '${supportedAppExtensions}' ` + _support.util.pluralize('bundle', supportedAppExtensions.length, false) + ` in it. Make sure your archive contains at least one package having ` + `'${supportedAppExtensions}' ${_support.util.pluralize('extension', supportedAppExtensions.length, false)}`);
404
460
  }
405
461
 
406
- const matchedBundle = _lodash.default.first(allBundleItems);
462
+ _logger.default.debug(`Extracted ${_support.util.pluralize('bundle item', sortedBundleItems.length, true)} ` + `from '${zipPath}' in ${Math.round(timer.getDuration().asMilliSeconds)}ms: ${sortedBundleItems}`);
463
+
464
+ const matchedBundle = _lodash.default.first(sortedBundleItems);
407
465
 
408
- _logger.default.debug(`Matched ${_support.util.pluralize('item', allBundleItems.length, true)} in the extracted archive. ` + `Assuming '${matchedBundle}' is the correct bundle`);
466
+ _logger.default.info(`Assuming '${matchedBundle}' is the correct bundle`);
409
467
 
410
468
  const dstPath = _path.default.resolve(dstRoot, _path.default.basename(matchedBundle));
411
469
 
@@ -473,4 +531,4 @@ function parseCapsArray(cap) {
473
531
  }require('source-map-support').install();
474
532
 
475
533
 
476
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/basedriver/helpers.js"],"names":["IPA_EXT","ZIP_EXTS","ZIP_MIME_TYPES","CACHED_APPS_MAX_AGE","APPLICATIONS_CACHE","LRU","maxAge","updateAgeOnGet","dispose","app","fullPath","fs","exists","logger","info","rimraf","noDisposeOnSet","APPLICATIONS_CACHE_GUARD","AsyncLock","SANITIZE_REPLACEMENT","DEFAULT_BASENAME","APP_DOWNLOAD_TIMEOUT_MS","process","on","itemCount","appPaths","values","map","debug","length","util","pluralize","appPath","rimrafSync","e","warn","message","retrieveHeaders","link","url","method","timeout","headers","getCachedApplicationPath","currentAppProps","refresh","has","lastModified","currentModified","immutable","currentImmutable","currentMaxAge","timestamp","get","getTime","msLeft","Date","now","path","basename","verifyAppExtension","supportedAppExtensions","includes","extname","Error","configureApp","_","isString","isArray","newApp","shouldUnzipApp","archiveHash","remoteAppProps","protocol","pathname","parse","isUrl","acquire","isEmpty","test","maxAgeMatch","exec","parseInt","cachedPath","del","fileName","sanitizeName","decodeURIComponent","replacement","ct","some","mimeType","RegExp","escapeRegExp","match","resultingName","substring","resultingExt","first","targetPath","tempDir","prefix","suffix","downloadApp","errorMessage","archivePath","hash","tmpRoot","openDir","unzipApp","isAbsolute","resolve","cwd","Boolean","set","href","net","downloadFile","err","zipPath","dstRoot","zip","assertValidZip","timer","timing","Timer","start","extractionOpts","useSystemUnzip","system","isWindows","fileNamesEncoding","extractAllTo","duration","getDuration","allExtractedItems","glob","Math","round","asMilliSeconds","allBundleItems","filter","relativePath","sort","a","b","split","sep","matchedBundle","dstPath","mv","mkdirp","isPackageOrBundle","duplicateKeys","input","firstKey","secondKey","item","isPlainObject","resultObj","key","value","toPairs","recursivelyCalledValue","parseCapsArray","cap","parsedCaps","JSON","ign"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport path from 'path';\nimport url from 'url';\nimport logger from './logger';\nimport { system, tempDir, fs, util, zip, net, timing } from '@appium/support';\nimport LRU from 'lru-cache';\nimport AsyncLock from 'async-lock';\nimport axios from 'axios';\n\nconst IPA_EXT = '.ipa';\nconst ZIP_EXTS = ['.zip', IPA_EXT];\nconst ZIP_MIME_TYPES = [\n  'application/zip',\n  'application/x-zip-compressed',\n  'multipart/x-zip',\n];\nconst CACHED_APPS_MAX_AGE = 1000 * 60 * 60 * 24; // ms\nconst APPLICATIONS_CACHE = new LRU({\n  maxAge: CACHED_APPS_MAX_AGE, // expire after 24 hours\n  updateAgeOnGet: true,\n  dispose: async (app, {fullPath}) => {\n    if (!await fs.exists(fullPath)) {\n      return;\n    }\n\n    logger.info(`The application '${app}' cached at '${fullPath}' has expired`);\n    await fs.rimraf(fullPath);\n  },\n  noDisposeOnSet: true,\n});\nconst APPLICATIONS_CACHE_GUARD = new AsyncLock();\nconst SANITIZE_REPLACEMENT = '-';\nconst DEFAULT_BASENAME = 'appium-app';\nconst APP_DOWNLOAD_TIMEOUT_MS = 120 * 1000;\n\nprocess.on('exit', () => {\n  if (APPLICATIONS_CACHE.itemCount === 0) {\n    return;\n  }\n\n  const appPaths = APPLICATIONS_CACHE.values()\n    .map(({fullPath}) => fullPath);\n  logger.debug(`Performing cleanup of ${appPaths.length} cached ` +\n    util.pluralize('application', appPaths.length));\n  for (const appPath of appPaths) {\n    try {\n      // Asynchronous calls are not supported in onExit handler\n      fs.rimrafSync(appPath);\n    } catch (e) {\n      logger.warn(e.message);\n    }\n  }\n});\n\n\nasync function retrieveHeaders (link) {\n  try {\n    return (await axios({\n      url: link,\n      method: 'HEAD',\n      timeout: 5000,\n    })).headers;\n  } catch (e) {\n    logger.info(`Cannot send HEAD request to '${link}'. Original error: ${e.message}`);\n  }\n  return {};\n}\n\nfunction getCachedApplicationPath (link, currentAppProps = {}) {\n  const refresh = () => {\n    logger.debug(`A fresh copy of the application is going to be downloaded from ${link}`);\n    return null;\n  };\n\n  if (APPLICATIONS_CACHE.has(link)) {\n    const {\n      lastModified: currentModified,\n      immutable: currentImmutable,\n      // maxAge is in seconds\n      maxAge: currentMaxAge,\n    } = currentAppProps;\n    const {\n      // Date instance\n      lastModified,\n      // boolean\n      immutable,\n      // Unix time in milliseconds\n      timestamp,\n      fullPath,\n    } = APPLICATIONS_CACHE.get(link);\n    if (lastModified && currentModified) {\n      if (currentModified.getTime() <= lastModified.getTime()) {\n        logger.debug(`The application at ${link} has not been modified since ${lastModified}`);\n        return fullPath;\n      }\n      logger.debug(`The application at ${link} has been modified since ${lastModified}`);\n      return refresh();\n    }\n    if (immutable && currentImmutable) {\n      logger.debug(`The application at ${link} is immutable`);\n      return fullPath;\n    }\n    if (currentMaxAge && timestamp) {\n      const msLeft = timestamp + currentMaxAge * 1000 - Date.now();\n      if (msLeft > 0) {\n        logger.debug(`The cached application '${path.basename(fullPath)}' will expire in ${msLeft / 1000}s`);\n        return fullPath;\n      }\n      logger.debug(`The cached application '${path.basename(fullPath)}' has expired`);\n    }\n  }\n  return refresh();\n}\n\nfunction verifyAppExtension (app, supportedAppExtensions) {\n  if (supportedAppExtensions.includes(path.extname(app))) {\n    return app;\n  }\n  throw new Error(`New app path '${app}' did not have ` +\n    `${util.pluralize('extension', supportedAppExtensions.length, false)}: ` +\n    supportedAppExtensions);\n}\n\nasync function configureApp (app, supportedAppExtensions) {\n  if (!_.isString(app)) {\n    // immediately shortcircuit if not given an app\n    return;\n  }\n  if (!_.isArray(supportedAppExtensions)) {\n    supportedAppExtensions = [supportedAppExtensions];\n  }\n\n  let newApp = app;\n  let shouldUnzipApp = false;\n  let archiveHash = null;\n  const remoteAppProps = {\n    lastModified: null,\n    immutable: false,\n    maxAge: null,\n  };\n  const {protocol, pathname} = url.parse(newApp);\n  const isUrl = ['http:', 'https:'].includes(protocol);\n\n  return await APPLICATIONS_CACHE_GUARD.acquire(app, async () => {\n    if (isUrl) {\n      // Use the app from remote URL\n      logger.info(`Using downloadable app '${newApp}'`);\n      const headers = await retrieveHeaders(newApp);\n      if (!_.isEmpty(headers)) {\n        if (headers['last-modified']) {\n          remoteAppProps.lastModified = new Date(headers['last-modified']);\n        }\n        logger.debug(`Last-Modified: ${headers['last-modified']}`);\n        if (headers['cache-control']) {\n          remoteAppProps.immutable = /\\bimmutable\\b/i.test(headers['cache-control']);\n          const maxAgeMatch = /\\bmax-age=(\\d+)\\b/i.exec(headers['cache-control']);\n          if (maxAgeMatch) {\n            remoteAppProps.maxAge = parseInt(maxAgeMatch[1], 10);\n          }\n        }\n        logger.debug(`Cache-Control: ${headers['cache-control']}`);\n      }\n      const cachedPath = getCachedApplicationPath(app, remoteAppProps);\n      if (cachedPath) {\n        if (await fs.exists(cachedPath)) {\n          logger.info(`Reusing previously downloaded application at '${cachedPath}'`);\n          return verifyAppExtension(cachedPath, supportedAppExtensions);\n        }\n        logger.info(`The application at '${cachedPath}' does not exist anymore. Deleting it from the cache`);\n        APPLICATIONS_CACHE.del(app);\n      }\n\n      let fileName = null;\n      const basename = fs.sanitizeName(path.basename(decodeURIComponent(pathname)), {\n        replacement: SANITIZE_REPLACEMENT\n      });\n      const extname = path.extname(basename);\n      // to determine if we need to unzip the app, we have a number of places\n      // to look: content type, content disposition, or the file extension\n      if (ZIP_EXTS.includes(extname)) {\n        fileName = basename;\n        shouldUnzipApp = true;\n      }\n      if (headers['content-type']) {\n        const ct = headers['content-type'];\n        logger.debug(`Content-Type: ${ct}`);\n        // the filetype may not be obvious for certain urls, so check the mime type too\n        if (ZIP_MIME_TYPES.some((mimeType) => new RegExp(`\\\\b${_.escapeRegExp(mimeType)}\\\\b`).test(ct))) {\n          if (!fileName) {\n            fileName = `${DEFAULT_BASENAME}.zip`;\n          }\n          shouldUnzipApp = true;\n        }\n      }\n      if (headers['content-disposition'] && /^attachment/i.test(headers['content-disposition'])) {\n        logger.debug(`Content-Disposition: ${headers['content-disposition']}`);\n        const match = /filename=\"([^\"]+)/i.exec(headers['content-disposition']);\n        if (match) {\n          fileName = fs.sanitizeName(match[1], {\n            replacement: SANITIZE_REPLACEMENT\n          });\n          shouldUnzipApp = shouldUnzipApp || ZIP_EXTS.includes(path.extname(fileName));\n        }\n      }\n      if (!fileName) {\n        // assign the default file name and the extension if none has been detected\n        const resultingName = basename\n          ? basename.substring(0, basename.length - extname.length)\n          : DEFAULT_BASENAME;\n        let resultingExt = extname;\n        if (!supportedAppExtensions.includes(resultingExt)) {\n          logger.info(`The current file extension '${resultingExt}' is not supported. ` +\n            `Defaulting to '${_.first(supportedAppExtensions)}'`);\n          resultingExt = _.first(supportedAppExtensions);\n        }\n        fileName = `${resultingName}${resultingExt}`;\n      }\n      const targetPath = await tempDir.path({\n        prefix: fileName,\n        suffix: '',\n      });\n      newApp = await downloadApp(newApp, targetPath);\n    } else if (await fs.exists(newApp)) {\n      // Use the local app\n      logger.info(`Using local app '${newApp}'`);\n      shouldUnzipApp = ZIP_EXTS.includes(path.extname(newApp));\n    } else {\n      let errorMessage = `The application at '${newApp}' does not exist or is not accessible`;\n      // protocol value for 'C:\\\\temp' is 'c:', so we check the length as well\n      if (_.isString(protocol) && protocol.length > 2) {\n        errorMessage = `The protocol '${protocol}' used in '${newApp}' is not supported. ` +\n          `Only http: and https: protocols are supported`;\n      }\n      throw new Error(errorMessage);\n    }\n\n    if (shouldUnzipApp) {\n      const archivePath = newApp;\n      archiveHash = await fs.hash(archivePath);\n      if (APPLICATIONS_CACHE.has(app) && archiveHash === APPLICATIONS_CACHE.get(app).hash) {\n        const {fullPath} = APPLICATIONS_CACHE.get(app);\n        if (await fs.exists(fullPath)) {\n          if (archivePath !== app) {\n            await fs.rimraf(archivePath);\n          }\n          logger.info(`Will reuse previously cached application at '${fullPath}'`);\n          return verifyAppExtension(fullPath, supportedAppExtensions);\n        }\n        logger.info(`The application at '${fullPath}' does not exist anymore. Deleting it from the cache`);\n        APPLICATIONS_CACHE.del(app);\n      }\n      const tmpRoot = await tempDir.openDir();\n      try {\n        newApp = await unzipApp(archivePath, tmpRoot, supportedAppExtensions);\n      } finally {\n        if (newApp !== archivePath && archivePath !== app) {\n          await fs.rimraf(archivePath);\n        }\n      }\n      logger.info(`Unzipped local app to '${newApp}'`);\n    } else if (!path.isAbsolute(newApp)) {\n      newApp = path.resolve(process.cwd(), newApp);\n      logger.warn(`The current application path '${app}' is not absolute ` +\n        `and has been rewritten to '${newApp}'. Consider using absolute paths rather than relative`);\n      app = newApp;\n    }\n\n    verifyAppExtension(newApp, supportedAppExtensions);\n\n    if (app !== newApp && (archiveHash || _.values(remoteAppProps).some(Boolean))) {\n      if (APPLICATIONS_CACHE.has(app)) {\n        const {fullPath} = APPLICATIONS_CACHE.get(app);\n        // Clean up the obsolete entry first if needed\n        if (fullPath !== newApp && await fs.exists(fullPath)) {\n          await fs.rimraf(fullPath);\n        }\n      }\n      APPLICATIONS_CACHE.set(app, {\n        ...remoteAppProps,\n        timestamp: Date.now(),\n        hash: archiveHash,\n        fullPath: newApp,\n      });\n    }\n    return newApp;\n  });\n}\n\nasync function downloadApp (app, targetPath) {\n  const {href} = url.parse(app);\n  try {\n    await net.downloadFile(href, targetPath, {\n      timeout: APP_DOWNLOAD_TIMEOUT_MS,\n    });\n  } catch (err) {\n    throw new Error(`Unable to download the app: ${err.message}`);\n  }\n  return targetPath;\n}\n\n/**\n * Extracts the bundle from an archive into the given folder\n *\n * @param {string} zipPath Full path to the archive containing the bundle\n * @param {string} dstRoot Full path to the folder where the extracted bundle\n * should be placed\n * @param {Array<string>|string} supportedAppExtensions The list of extensions\n * the target application bundle supports, for example ['.apk', '.apks'] for\n * Android packages\n * @returns {string} Full path to the bundle in the destination folder\n * @throws {Error} If the given archive is invalid or no application bundles\n * have been found inside\n */\nasync function unzipApp (zipPath, dstRoot, supportedAppExtensions) {\n  await zip.assertValidZip(zipPath);\n\n  if (!_.isArray(supportedAppExtensions)) {\n    supportedAppExtensions = [supportedAppExtensions];\n  }\n\n  const tmpRoot = await tempDir.openDir();\n  try {\n    logger.debug(`Unzipping '${zipPath}'`);\n    const timer = new timing.Timer().start();\n    /**\n     * Attempt to use use the system `unzip` (e.g., `/usr/bin/unzip`) due\n     * to the significant performance improvement it provides over the native\n     * JS \"unzip\" implementation.\n     * @type {import('@appium/support/lib/zip').ExtractAllOptions}\n     */\n    const extractionOpts = {\n      useSystemUnzip: !system.isWindows(),\n    };\n    // https://github.com/appium/appium/issues/14100\n    if (path.extname(zipPath) === IPA_EXT) {\n      logger.debug(`Enforcing UTF-8 encoding on the extracted file names for '${path.basename(zipPath)}'`);\n      extractionOpts.fileNamesEncoding = 'utf8';\n    }\n    await zip.extractAllTo(zipPath, tmpRoot, extractionOpts);\n    const duration = timer.getDuration();\n    const allExtractedItems = await fs.glob('**', {cwd: tmpRoot});\n    logger.debug(`Extracted ${util.pluralize('item', allExtractedItems.length, true)} ` +\n      `from '${zipPath}' in ${Math.round(duration.asMilliSeconds)}ms`);\n    const allBundleItems = allExtractedItems\n      .filter((relativePath) => supportedAppExtensions.includes(path.extname(relativePath)))\n      // Get the top level match\n      .sort((a, b) => a.split(path.sep).length - b.split(path.sep).length);\n    if (_.isEmpty(allBundleItems)) {\n      throw new Error(`App zip unzipped OK, but we could not find '${supportedAppExtensions}' ` +\n        util.pluralize('bundle', supportedAppExtensions.length, false) +\n        ` in it. Make sure your archive contains at least one package having ` +\n        `'${supportedAppExtensions}' ${util.pluralize('extension', supportedAppExtensions.length, false)}`);\n    }\n    const matchedBundle = _.first(allBundleItems);\n    logger.debug(`Matched ${util.pluralize('item', allBundleItems.length, true)} in the extracted archive. ` +\n      `Assuming '${matchedBundle}' is the correct bundle`);\n    const dstPath = path.resolve(dstRoot, path.basename(matchedBundle));\n    await fs.mv(path.resolve(tmpRoot, matchedBundle), dstPath, {mkdirp: true});\n    return dstPath;\n  } finally {\n    await fs.rimraf(tmpRoot);\n  }\n}\n\nfunction isPackageOrBundle (app) {\n  return (/^([a-zA-Z0-9\\-_]+\\.[a-zA-Z0-9\\-_]+)+$/).test(app);\n}\n\n/**\n * Finds all instances 'firstKey' and create a duplicate with the key 'secondKey',\n * Do the same thing in reverse. If we find 'secondKey', create a duplicate with the key 'firstKey'.\n *\n * This will cause keys to be overwritten if the object contains 'firstKey' and 'secondKey'.\n\n * @param {*} input Any type of input\n * @param {String} firstKey The first key to duplicate\n * @param {String} secondKey The second key to duplicate\n */\nfunction duplicateKeys (input, firstKey, secondKey) {\n  // If array provided, recursively call on all elements\n  if (_.isArray(input)) {\n    return input.map((item) => duplicateKeys(item, firstKey, secondKey));\n  }\n\n  // If object, create duplicates for keys and then recursively call on values\n  if (_.isPlainObject(input)) {\n    const resultObj = {};\n    for (let [key, value] of _.toPairs(input)) {\n      const recursivelyCalledValue = duplicateKeys(value, firstKey, secondKey);\n      if (key === firstKey) {\n        resultObj[secondKey] = recursivelyCalledValue;\n      } else if (key === secondKey) {\n        resultObj[firstKey] = recursivelyCalledValue;\n      }\n      resultObj[key] = recursivelyCalledValue;\n    }\n    return resultObj;\n  }\n\n  // Base case. Return primitives without doing anything.\n  return input;\n}\n\n/**\n * Takes a desired capability and tries to JSON.parse it as an array,\n * and either returns the parsed array or a singleton array.\n *\n * @param {string|Array<String>} cap A desired capability\n */\nfunction parseCapsArray (cap) {\n  if (_.isArray(cap)) {\n    return cap;\n  }\n\n  let parsedCaps;\n  try {\n    parsedCaps = JSON.parse(cap);\n    if (_.isArray(parsedCaps)) {\n      return parsedCaps;\n    }\n  } catch (ign) {\n    logger.warn(`Failed to parse capability as JSON array`);\n  }\n  if (_.isString(cap)) {\n    return [cap];\n  }\n  throw new Error(`must provide a string or JSON Array; received ${cap}`);\n}\n\nexport {\n  configureApp, isPackageOrBundle, duplicateKeys, parseCapsArray\n};\n"],"file":"lib/basedriver/helpers.js","sourceRoot":"../../.."}
534
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/basedriver/helpers.js"],"names":["IPA_EXT","ZIP_EXTS","ZIP_MIME_TYPES","CACHED_APPS_MAX_AGE","APPLICATIONS_CACHE","LRU","maxAge","updateAgeOnGet","dispose","app","fullPath","logger","info","setTimeout","fs","rimraf","noDisposeOnSet","APPLICATIONS_CACHE_GUARD","AsyncLock","SANITIZE_REPLACEMENT","DEFAULT_BASENAME","APP_DOWNLOAD_TIMEOUT_MS","process","on","itemCount","appPaths","values","map","debug","length","util","pluralize","appPath","rimrafSync","e","warn","message","retrieveHeaders","link","url","method","timeout","headers","getCachedApplicationPath","currentAppProps","cachedAppInfo","refresh","_","isPlainObject","lastModified","currentModified","immutable","currentImmutable","currentMaxAge","timestamp","getTime","msLeft","Date","now","path","basename","verifyAppExtension","supportedAppExtensions","toLower","includes","extname","Error","calculateFolderIntegrity","folderPath","glob","cwd","strict","nosort","calculateFileIntegrity","filePath","hash","isAppIntegrityOk","currentPath","expectedIntegrity","exists","stat","isDirectory","folder","file","configureApp","options","isString","onPostProcess","isArray","supportedExtensions","isEmpty","newApp","shouldUnzipApp","packageHash","remoteAppProps","protocol","pathname","parse","isUrl","get","acquire","test","maxAgeMatch","exec","parseInt","cachedPath","integrity","del","fileName","sanitizeName","decodeURIComponent","replacement","ct","some","mimeType","RegExp","escapeRegExp","match","resultingName","substring","resultingExt","first","targetPath","tempDir","prefix","suffix","downloadApp","errorMessage","isPackageAFile","isFile","isFunction","archivePath","tmpRoot","openDir","unzipApp","isAbsolute","resolve","storeAppInCache","appPathToCache","cachedFullPath","set","result","clone","Boolean","href","net","downloadFile","err","zipPath","dstRoot","zip","assertValidZip","timer","timing","Timer","start","useSystemUnzipEnv","env","APPIUM_PREFER_SYSTEM_UNZIP","useSystemUnzip","extractionOpts","fileNamesEncoding","extractAllTo","globPattern","ext","replace","join","sortedBundleItems","sort","a","b","split","sep","errorAndThrow","Math","round","getDuration","asMilliSeconds","matchedBundle","dstPath","mv","mkdirp","isPackageOrBundle","duplicateKeys","input","firstKey","secondKey","item","resultObj","key","value","toPairs","recursivelyCalledValue","parseCapsArray","cap","parsedCaps","JSON","ign"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport path from 'path';\nimport url from 'url';\nimport logger from './logger';\nimport { tempDir, fs, util, zip, net, timing } from '@appium/support';\nimport LRU from 'lru-cache';\nimport AsyncLock from 'async-lock';\nimport axios from 'axios';\n\nconst IPA_EXT = '.ipa';\nconst ZIP_EXTS = ['.zip', IPA_EXT];\nconst ZIP_MIME_TYPES = [\n  'application/zip',\n  'application/x-zip-compressed',\n  'multipart/x-zip',\n];\nconst CACHED_APPS_MAX_AGE = 1000 * 60 * 60 * 24; // ms\nconst APPLICATIONS_CACHE = new LRU({\n  maxAge: CACHED_APPS_MAX_AGE, // expire after 24 hours\n  updateAgeOnGet: true,\n  dispose: (app, {fullPath}) => {\n    logger.info(`The application '${app}' cached at '${fullPath}' has ` +\n      `expired after ${CACHED_APPS_MAX_AGE}ms`);\n    setTimeout(async () => {\n      if (fullPath) {\n        await fs.rimraf(fullPath);\n      }\n    });\n  },\n  noDisposeOnSet: true,\n});\nconst APPLICATIONS_CACHE_GUARD = new AsyncLock();\nconst SANITIZE_REPLACEMENT = '-';\nconst DEFAULT_BASENAME = 'appium-app';\nconst APP_DOWNLOAD_TIMEOUT_MS = 120 * 1000;\n\nprocess.on('exit', () => {\n  if (APPLICATIONS_CACHE.itemCount === 0) {\n    return;\n  }\n\n  const appPaths = APPLICATIONS_CACHE.values()\n    .map(({fullPath}) => fullPath);\n  logger.debug(`Performing cleanup of ${appPaths.length} cached ` +\n    util.pluralize('application', appPaths.length));\n  for (const appPath of appPaths) {\n    try {\n      // Asynchronous calls are not supported in onExit handler\n      fs.rimrafSync(appPath);\n    } catch (e) {\n      logger.warn(e.message);\n    }\n  }\n});\n\n\nasync function retrieveHeaders (link) {\n  try {\n    return (await axios({\n      url: link,\n      method: 'HEAD',\n      timeout: 5000,\n    })).headers;\n  } catch (e) {\n    logger.info(`Cannot send HEAD request to '${link}'. Original error: ${e.message}`);\n  }\n  return {};\n}\n\nfunction getCachedApplicationPath (link, currentAppProps = {}, cachedAppInfo = {}) {\n  const refresh = () => {\n    logger.debug(`A fresh copy of the application is going to be downloaded from ${link}`);\n    return null;\n  };\n\n  if (!_.isPlainObject(cachedAppInfo) || !_.isPlainObject(currentAppProps)) {\n    // if an invalid arg is passed then assume cache miss\n    return refresh();\n  }\n\n  const {\n    lastModified: currentModified,\n    immutable: currentImmutable,\n    // maxAge is in seconds\n    maxAge: currentMaxAge,\n  } = currentAppProps;\n  const {\n    // Date instance\n    lastModified,\n    // boolean\n    immutable,\n    // Unix time in milliseconds\n    timestamp,\n    fullPath,\n  } = cachedAppInfo;\n  if (lastModified && currentModified) {\n    if (currentModified.getTime() <= lastModified.getTime()) {\n      logger.debug(`The application at ${link} has not been modified since ${lastModified}`);\n      return fullPath;\n    }\n    logger.debug(`The application at ${link} has been modified since ${lastModified}`);\n    return refresh();\n  }\n  if (immutable && currentImmutable) {\n    logger.debug(`The application at ${link} is immutable`);\n    return fullPath;\n  }\n  if (currentMaxAge && timestamp) {\n    const msLeft = timestamp + currentMaxAge * 1000 - Date.now();\n    if (msLeft > 0) {\n      logger.debug(`The cached application '${path.basename(fullPath)}' will expire in ${msLeft / 1000}s`);\n      return fullPath;\n    }\n    logger.debug(`The cached application '${path.basename(fullPath)}' has expired`);\n  }\n  return refresh();\n}\n\nfunction verifyAppExtension (app, supportedAppExtensions) {\n  if (supportedAppExtensions.map(_.toLower).includes(_.toLower(path.extname(app)))) {\n    return app;\n  }\n  throw new Error(`New app path '${app}' did not have ` +\n    `${util.pluralize('extension', supportedAppExtensions.length, false)}: ` +\n    supportedAppExtensions);\n}\n\nasync function calculateFolderIntegrity (folderPath) {\n  return (await fs.glob('**/*', {cwd: folderPath, strict: false, nosort: true})).length;\n}\n\nasync function calculateFileIntegrity (filePath) {\n  return await fs.hash(filePath);\n}\n\nasync function isAppIntegrityOk (currentPath, expectedIntegrity = {}) {\n  if (!await fs.exists(currentPath)) {\n    return false;\n  }\n\n  // Folder integrity check is simple:\n  // Verify the previous amount of files is not greater than the current one.\n  // We don't want to use equality comparison because of an assumption that the OS might\n  // create some unwanted service files/cached inside of that folder or its subfolders.\n  // Ofc, validating the hash sum of each file (or at least of file path) would be much\n  // more precise, but we don't need to be very precise here and also don't want to\n  // overuse RAM and have a performance drop.\n  return (await fs.stat(currentPath)).isDirectory()\n    ? await calculateFolderIntegrity(currentPath) >= expectedIntegrity?.folder\n    : await calculateFileIntegrity(currentPath) === expectedIntegrity?.file;\n}\n\n/**\n * @typedef {Object} PostProcessOptions\n * @property {?Object} cachedAppInfo The information about the previously cached app instance (if exists):\n *    - packageHash: SHA1 hash of the package if it is a file and not a folder\n *    - lastModified: Optional Date instance, the value of file's `Last-Modified` header\n *    - immutable: Optional boolean value. Contains true if the file has an `immutable` mark\n *                 in `Cache-control` header\n *    - maxAge: Optional integer representation of `maxAge` parameter in `Cache-control` header\n *    - timestamp: The timestamp this item has been added to the cache (measured in Unix epoch\n *                 milliseconds)\n *    - integrity: An object containing either `file` property with SHA1 hash of the file\n *                 or `folder` property with total amount of cached files and subfolders\n *    - fullPath: the full path to the cached app\n * @property {boolean} isUrl Whether the app has been downloaded from a remote URL\n * @property {?Object} headers Optional headers object. Only present if `isUrl` is true and if the server\n * responds to HEAD requests. All header names are normalized to lowercase.\n * @property {string} appPath A string containing full path to the preprocessed application package (either\n * downloaded or a local one)\n */\n\n/**\n * @typedef {Object} PostProcessResult\n * @property {string} appPath The full past to the post-processed application package on the\n * local file system (might be a file or a folder path)\n */\n\n/**\n * @typedef {Object} ConfigureAppOptions\n * @property {(obj: PostProcessOptions) => (Promise<PostProcessResult|undefined>|PostProcessResult|undefined)} onPostProcess\n * Optional function, which should be applied\n * to the application after it is downloaded/preprocessed. This function may be async\n * and is expected to accept single object parameter.\n * The function is expected to either return a falsy value, which means the app must not be\n * cached and a fresh copy of it is downloaded each time. If this function returns an object\n * containing `appPath` property then the integrity of it will be verified and stored into\n * the cache.\n * @property {string[]} supportedExtensions List of supported application extensions (\n * including starting dots). This property is mandatory and must not be empty.\n */\n\n/**\n * Prepares an app to be used in an automated test. The app gets cached automatically\n * if it is an archive or if it is downloaded from an URL.\n *\n * @param {string} app Either a full path to the app or a remote URL\n * @param {string|string[]|ConfigureAppOptions} options\n * @returns The full path to the resulting application bundle\n */\nasync function configureApp (app, options = {}) {\n  if (!_.isString(app)) {\n    // immediately shortcircuit if not given an app\n    return;\n  }\n\n  let supportedAppExtensions;\n  const {\n    onPostProcess,\n  } = _.isPlainObject(options) ? options : {};\n  if (_.isString(options)) {\n    supportedAppExtensions = [options];\n  } else if (_.isArray(options)) {\n    supportedAppExtensions = options;\n  } else if (_.isPlainObject(options)) {\n    supportedAppExtensions = options.supportedExtensions;\n  }\n  if (_.isEmpty(supportedAppExtensions)) {\n    throw new Error(`One or more supported app extensions must be provided`);\n  }\n\n  let newApp = app;\n  let shouldUnzipApp = false;\n  let packageHash = null;\n  let headers = null;\n  const remoteAppProps = {\n    lastModified: null,\n    immutable: false,\n    maxAge: null,\n  };\n  const {protocol, pathname} = url.parse(newApp);\n  const isUrl = ['http:', 'https:'].includes(protocol);\n\n  const cachedAppInfo = APPLICATIONS_CACHE.get(app);\n\n  return await APPLICATIONS_CACHE_GUARD.acquire(app, async () => {\n    if (isUrl) {\n      // Use the app from remote URL\n      logger.info(`Using downloadable app '${newApp}'`);\n      headers = await retrieveHeaders(newApp);\n      if (!_.isEmpty(headers)) {\n        if (headers['last-modified']) {\n          remoteAppProps.lastModified = new Date(headers['last-modified']);\n        }\n        logger.debug(`Last-Modified: ${headers['last-modified']}`);\n        if (headers['cache-control']) {\n          remoteAppProps.immutable = /\\bimmutable\\b/i.test(headers['cache-control']);\n          const maxAgeMatch = /\\bmax-age=(\\d+)\\b/i.exec(headers['cache-control']);\n          if (maxAgeMatch) {\n            remoteAppProps.maxAge = parseInt(maxAgeMatch[1], 10);\n          }\n        }\n        logger.debug(`Cache-Control: ${headers['cache-control']}`);\n      }\n      const cachedPath = getCachedApplicationPath(app, remoteAppProps, cachedAppInfo);\n      if (cachedPath) {\n        if (await isAppIntegrityOk(cachedPath, cachedAppInfo?.integrity)) {\n          logger.info(`Reusing previously downloaded application at '${cachedPath}'`);\n          return verifyAppExtension(cachedPath, supportedAppExtensions);\n        }\n        logger.info(`The application at '${cachedPath}' does not exist anymore ` +\n          `or its integrity has been damaged. Deleting it from the internal cache`);\n        APPLICATIONS_CACHE.del(app);\n      }\n\n      let fileName = null;\n      const basename = fs.sanitizeName(path.basename(decodeURIComponent(pathname)), {\n        replacement: SANITIZE_REPLACEMENT\n      });\n      const extname = path.extname(basename);\n      // to determine if we need to unzip the app, we have a number of places\n      // to look: content type, content disposition, or the file extension\n      if (ZIP_EXTS.includes(extname)) {\n        fileName = basename;\n        shouldUnzipApp = true;\n      }\n      if (headers['content-type']) {\n        const ct = headers['content-type'];\n        logger.debug(`Content-Type: ${ct}`);\n        // the filetype may not be obvious for certain urls, so check the mime type too\n        if (ZIP_MIME_TYPES.some((mimeType) => new RegExp(`\\\\b${_.escapeRegExp(mimeType)}\\\\b`).test(ct))) {\n          if (!fileName) {\n            fileName = `${DEFAULT_BASENAME}.zip`;\n          }\n          shouldUnzipApp = true;\n        }\n      }\n      if (headers['content-disposition'] && /^attachment/i.test(headers['content-disposition'])) {\n        logger.debug(`Content-Disposition: ${headers['content-disposition']}`);\n        const match = /filename=\"([^\"]+)/i.exec(headers['content-disposition']);\n        if (match) {\n          fileName = fs.sanitizeName(match[1], {\n            replacement: SANITIZE_REPLACEMENT\n          });\n          shouldUnzipApp = shouldUnzipApp || ZIP_EXTS.includes(path.extname(fileName));\n        }\n      }\n      if (!fileName) {\n        // assign the default file name and the extension if none has been detected\n        const resultingName = basename\n          ? basename.substring(0, basename.length - extname.length)\n          : DEFAULT_BASENAME;\n        let resultingExt = extname;\n        if (!supportedAppExtensions.includes(resultingExt)) {\n          logger.info(`The current file extension '${resultingExt}' is not supported. ` +\n            `Defaulting to '${_.first(supportedAppExtensions)}'`);\n          resultingExt = _.first(supportedAppExtensions);\n        }\n        fileName = `${resultingName}${resultingExt}`;\n      }\n      const targetPath = await tempDir.path({\n        prefix: fileName,\n        suffix: '',\n      });\n      newApp = await downloadApp(newApp, targetPath);\n    } else if (await fs.exists(newApp)) {\n      // Use the local app\n      logger.info(`Using local app '${newApp}'`);\n      shouldUnzipApp = ZIP_EXTS.includes(path.extname(newApp));\n    } else {\n      let errorMessage = `The application at '${newApp}' does not exist or is not accessible`;\n      // protocol value for 'C:\\\\temp' is 'c:', so we check the length as well\n      if (_.isString(protocol) && protocol.length > 2) {\n        errorMessage = `The protocol '${protocol}' used in '${newApp}' is not supported. ` +\n          `Only http: and https: protocols are supported`;\n      }\n      throw new Error(errorMessage);\n    }\n\n    const isPackageAFile = (await fs.stat(newApp)).isFile();\n    if (isPackageAFile) {\n      packageHash = await calculateFileIntegrity(newApp);\n    }\n\n    if (isPackageAFile && shouldUnzipApp && !_.isFunction(onPostProcess)) {\n      const archivePath = newApp;\n      if (packageHash === cachedAppInfo?.packageHash) {\n        const {fullPath} = cachedAppInfo;\n        if (await isAppIntegrityOk(fullPath, cachedAppInfo?.integrity)) {\n          if (archivePath !== app) {\n            await fs.rimraf(archivePath);\n          }\n          logger.info(`Will reuse previously cached application at '${fullPath}'`);\n          return verifyAppExtension(fullPath, supportedAppExtensions);\n        }\n        logger.info(`The application at '${fullPath}' does not exist anymore ` +\n          `or its integrity has been damaged. Deleting it from the cache`);\n        APPLICATIONS_CACHE.del(app);\n      }\n      const tmpRoot = await tempDir.openDir();\n      try {\n        newApp = await unzipApp(archivePath, tmpRoot, supportedAppExtensions);\n      } finally {\n        if (newApp !== archivePath && archivePath !== app) {\n          await fs.rimraf(archivePath);\n        }\n      }\n      logger.info(`Unzipped local app to '${newApp}'`);\n    } else if (!path.isAbsolute(newApp)) {\n      newApp = path.resolve(process.cwd(), newApp);\n      logger.warn(`The current application path '${app}' is not absolute ` +\n        `and has been rewritten to '${newApp}'. Consider using absolute paths rather than relative`);\n      app = newApp;\n    }\n\n    const storeAppInCache = async (appPathToCache) => {\n      const cachedFullPath = cachedAppInfo?.fullPath;\n      if (cachedFullPath && cachedFullPath !== appPathToCache) {\n        await fs.rimraf(cachedFullPath);\n      }\n      const integrity = {};\n      if ((await fs.stat(appPathToCache)).isDirectory()) {\n        integrity.folder = await calculateFolderIntegrity(appPathToCache);\n      } else {\n        integrity.file = await calculateFileIntegrity(appPathToCache);\n      }\n      APPLICATIONS_CACHE.set(app, {\n        ...remoteAppProps,\n        timestamp: Date.now(),\n        packageHash,\n        integrity,\n        fullPath: appPathToCache,\n      });\n      return appPathToCache;\n    };\n\n    if (_.isFunction(onPostProcess)) {\n      const result = await onPostProcess({\n        cachedAppInfo: _.clone(cachedAppInfo),\n        isUrl,\n        headers: _.clone(headers),\n        appPath: newApp,\n      });\n      return (!result?.appPath || app === result?.appPath || !await fs.exists(result?.appPath))\n        ? newApp\n        : await storeAppInCache(result.appPath);\n    }\n\n    verifyAppExtension(newApp, supportedAppExtensions);\n    return (app !== newApp && (packageHash || _.values(remoteAppProps).some(Boolean)))\n      ? await storeAppInCache(newApp)\n      : newApp;\n  });\n}\n\nasync function downloadApp (app, targetPath) {\n  const {href} = url.parse(app);\n  try {\n    await net.downloadFile(href, targetPath, {\n      timeout: APP_DOWNLOAD_TIMEOUT_MS,\n    });\n  } catch (err) {\n    throw new Error(`Unable to download the app: ${err.message}`);\n  }\n  return targetPath;\n}\n\n/**\n * Extracts the bundle from an archive into the given folder\n *\n * @param {string} zipPath Full path to the archive containing the bundle\n * @param {string} dstRoot Full path to the folder where the extracted bundle\n * should be placed\n * @param {Array<string>|string} supportedAppExtensions The list of extensions\n * the target application bundle supports, for example ['.apk', '.apks'] for\n * Android packages\n * @returns {string} Full path to the bundle in the destination folder\n * @throws {Error} If the given archive is invalid or no application bundles\n * have been found inside\n */\nasync function unzipApp (zipPath, dstRoot, supportedAppExtensions) {\n  await zip.assertValidZip(zipPath);\n\n  if (!_.isArray(supportedAppExtensions)) {\n    supportedAppExtensions = [supportedAppExtensions];\n  }\n\n  const tmpRoot = await tempDir.openDir();\n  try {\n    logger.debug(`Unzipping '${zipPath}'`);\n    const timer = new timing.Timer().start();\n    const useSystemUnzipEnv = process.env.APPIUM_PREFER_SYSTEM_UNZIP;\n    const useSystemUnzip = _.isEmpty(useSystemUnzipEnv)\n      || !['0', 'false'].includes(_.toLower(useSystemUnzipEnv));\n    /**\n     * Attempt to use use the system `unzip` (e.g., `/usr/bin/unzip`) due\n     * to the significant performance improvement it provides over the native\n     * JS \"unzip\" implementation.\n     * @type {import('@appium/support/lib/zip').ExtractAllOptions}\n     */\n    const extractionOpts = {useSystemUnzip};\n    // https://github.com/appium/appium/issues/14100\n    if (path.extname(zipPath) === IPA_EXT) {\n      logger.debug(`Enforcing UTF-8 encoding on the extracted file names for '${path.basename(zipPath)}'`);\n      extractionOpts.fileNamesEncoding = 'utf8';\n    }\n    await zip.extractAllTo(zipPath, tmpRoot, extractionOpts);\n    const globPattern = `**/*.+(${supportedAppExtensions.map((ext) => ext.replace(/^\\./, '')).join('|')})`;\n    const sortedBundleItems = (await fs.glob(globPattern, {\n      cwd: tmpRoot,\n      strict: false,\n    // Get the top level match\n    })).sort((a, b) => a.split(path.sep).length - b.split(path.sep).length);\n    if (_.isEmpty(sortedBundleItems)) {\n      logger.errorAndThrow(`App unzipped OK, but we could not find any '${supportedAppExtensions}' ` +\n        util.pluralize('bundle', supportedAppExtensions.length, false) +\n        ` in it. Make sure your archive contains at least one package having ` +\n        `'${supportedAppExtensions}' ${util.pluralize('extension', supportedAppExtensions.length, false)}`);\n    }\n    logger.debug(`Extracted ${util.pluralize('bundle item', sortedBundleItems.length, true)} ` +\n      `from '${zipPath}' in ${Math.round(timer.getDuration().asMilliSeconds)}ms: ${sortedBundleItems}`);\n    const matchedBundle = _.first(sortedBundleItems);\n    logger.info(`Assuming '${matchedBundle}' is the correct bundle`);\n    const dstPath = path.resolve(dstRoot, path.basename(matchedBundle));\n    await fs.mv(path.resolve(tmpRoot, matchedBundle), dstPath, {mkdirp: true});\n    return dstPath;\n  } finally {\n    await fs.rimraf(tmpRoot);\n  }\n}\n\nfunction isPackageOrBundle (app) {\n  return (/^([a-zA-Z0-9\\-_]+\\.[a-zA-Z0-9\\-_]+)+$/).test(app);\n}\n\n/**\n * Finds all instances 'firstKey' and create a duplicate with the key 'secondKey',\n * Do the same thing in reverse. If we find 'secondKey', create a duplicate with the key 'firstKey'.\n *\n * This will cause keys to be overwritten if the object contains 'firstKey' and 'secondKey'.\n\n * @param {*} input Any type of input\n * @param {String} firstKey The first key to duplicate\n * @param {String} secondKey The second key to duplicate\n */\nfunction duplicateKeys (input, firstKey, secondKey) {\n  // If array provided, recursively call on all elements\n  if (_.isArray(input)) {\n    return input.map((item) => duplicateKeys(item, firstKey, secondKey));\n  }\n\n  // If object, create duplicates for keys and then recursively call on values\n  if (_.isPlainObject(input)) {\n    const resultObj = {};\n    for (let [key, value] of _.toPairs(input)) {\n      const recursivelyCalledValue = duplicateKeys(value, firstKey, secondKey);\n      if (key === firstKey) {\n        resultObj[secondKey] = recursivelyCalledValue;\n      } else if (key === secondKey) {\n        resultObj[firstKey] = recursivelyCalledValue;\n      }\n      resultObj[key] = recursivelyCalledValue;\n    }\n    return resultObj;\n  }\n\n  // Base case. Return primitives without doing anything.\n  return input;\n}\n\n/**\n * Takes a desired capability and tries to JSON.parse it as an array,\n * and either returns the parsed array or a singleton array.\n *\n * @param {string|Array<String>} cap A desired capability\n */\nfunction parseCapsArray (cap) {\n  if (_.isArray(cap)) {\n    return cap;\n  }\n\n  let parsedCaps;\n  try {\n    parsedCaps = JSON.parse(cap);\n    if (_.isArray(parsedCaps)) {\n      return parsedCaps;\n    }\n  } catch (ign) {\n    logger.warn(`Failed to parse capability as JSON array`);\n  }\n  if (_.isString(cap)) {\n    return [cap];\n  }\n  throw new Error(`must provide a string or JSON Array; received ${cap}`);\n}\n\nexport {\n  configureApp, isPackageOrBundle, duplicateKeys, parseCapsArray\n};\n"],"file":"lib/basedriver/helpers.js","sourceRoot":"../../.."}
@@ -11,7 +11,7 @@ require("source-map-support/register");
11
11
 
12
12
  var _lodash = _interopRequireDefault(require("lodash"));
13
13
 
14
- require("colors");
14
+ require("@dabh/colors");
15
15
 
16
16
  var _morgan = _interopRequireDefault(require("morgan"));
17
17
 
@@ -69,4 +69,4 @@ const startLogFormatter = (0, _morgan.default)((tokens, req, res) => {
69
69
  exports.startLogFormatter = startLogFormatter;require('source-map-support').install();
70
70
 
71
71
 
72
- //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9leHByZXNzL2V4cHJlc3MtbG9nZ2luZy5qcyJdLCJuYW1lcyI6WyJjb21waWxlIiwiZm10IiwicmVwbGFjZSIsIl8iLCJuYW1lIiwiYXJnIiwianMiLCJGdW5jdGlvbiIsInJlcXVlc3RFbmRMb2dnaW5nRm9ybWF0IiwidG9rZW5zIiwicmVxIiwicmVzIiwic3RhdHVzIiwic3RhdHVzQ29kZSIsInN0YXR1c1N0ciIsInJlZCIsInllbGxvdyIsImN5YW4iLCJncmVlbiIsImZuIiwid2hpdGUiLCJncmV5IiwiZW5kTG9nRm9ybWF0dGVyIiwibG9nIiwiaW5mbyIsImpzb25SZXNwIiwicmVxdWVzdFN0YXJ0TG9nZ2luZ0Zvcm1hdCIsInN0YXJ0TG9nRm9ybWF0dGVyIiwicmVxQm9keSIsImJvZHkiLCJ0cnVuY2F0ZSIsImlzU3RyaW5nIiwiSlNPTiIsInN0cmluZ2lmeSIsImxlbmd0aCIsIk1BWF9MT0dfQk9EWV9MRU5HVEgiLCJpZ24iLCJpbW1lZGlhdGUiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBS0EsU0FBU0EsT0FBVCxDQUFrQkMsR0FBbEIsRUFBdUI7QUFFckJBLEVBQUFBLEdBQUcsR0FBR0EsR0FBRyxDQUFDQyxPQUFKLENBQVksSUFBWixFQUFrQixLQUFsQixDQUFOO0FBQ0FELEVBQUFBLEdBQUcsR0FBR0EsR0FBRyxDQUFDQyxPQUFKLENBQVksZ0NBQVosRUFDSixTQUFTQSxPQUFULENBQWtCQyxDQUFsQixFQUFxQkMsSUFBckIsRUFBMkJDLEdBQTNCLEVBQWdDO0FBQzlCLFdBQVEscUJBQW9CRCxJQUFLLGlCQUFnQkMsR0FBSSxnQkFBckQ7QUFDRCxHQUhHLENBQU47QUFJQSxNQUFJQyxFQUFFLEdBQUksYUFBWUwsR0FBSSxJQUExQjtBQUNBLFNBQU8sSUFBSU0sUUFBSixDQUFhLGtCQUFiLEVBQWlDRCxFQUFqQyxDQUFQO0FBQ0Q7O0FBRUQsU0FBU0UsdUJBQVQsQ0FBa0NDLE1BQWxDLEVBQTBDQyxHQUExQyxFQUErQ0MsR0FBL0MsRUFBb0Q7QUFDbEQsTUFBSUMsTUFBTSxHQUFHRCxHQUFHLENBQUNFLFVBQWpCO0FBQ0EsTUFBSUMsU0FBUyxHQUFHLFNBQWhCOztBQUNBLE1BQUlGLE1BQU0sSUFBSSxHQUFkLEVBQW1CO0FBQ2pCRSxJQUFBQSxTQUFTLEdBQUdBLFNBQVMsQ0FBQ0MsR0FBdEI7QUFDRCxHQUZELE1BRU8sSUFBSUgsTUFBTSxJQUFJLEdBQWQsRUFBbUI7QUFDeEJFLElBQUFBLFNBQVMsR0FBR0EsU0FBUyxDQUFDRSxNQUF0QjtBQUNELEdBRk0sTUFFQSxJQUFJSixNQUFNLElBQUksR0FBZCxFQUFtQjtBQUN4QkUsSUFBQUEsU0FBUyxHQUFHQSxTQUFTLENBQUNHLElBQXRCO0FBQ0QsR0FGTSxNQUVBO0FBQ0xILElBQUFBLFNBQVMsR0FBR0EsU0FBUyxDQUFDSSxLQUF0QjtBQUNEOztBQUNELE1BQUlDLEVBQUUsR0FBR25CLE9BQU8sQ0FBRSxHQUFFLG9CQUFvQm9CLEtBQU0sR0FBRU4sU0FBVSxJQUFHLDJDQUEyQ08sSUFBSyxFQUE3RixDQUFoQjtBQUNBLFNBQU9GLEVBQUUsQ0FBQ1YsTUFBRCxFQUFTQyxHQUFULEVBQWNDLEdBQWQsQ0FBVDtBQUNEOztBQUVELE1BQU1XLGVBQWUsR0FBRyxxQkFBTyxDQUFDYixNQUFELEVBQVNDLEdBQVQsRUFBY0MsR0FBZCxLQUFzQjtBQUNuRFksa0JBQUlDLElBQUosQ0FBU2hCLHVCQUF1QixDQUFDQyxNQUFELEVBQVNDLEdBQVQsRUFBY0MsR0FBZCxDQUFoQyxFQUNFLENBQUNBLEdBQUcsQ0FBQ2MsUUFBSixJQUFnQixFQUFqQixFQUFxQkosSUFEdkI7QUFFRCxDQUh1QixDQUF4Qjs7QUFLQSxNQUFNSyx5QkFBeUIsR0FBRzFCLE9BQU8sQ0FBRSxHQUFFLE1BQU1vQixLQUFNLElBQUcsVUFBVUEsS0FBTSxJQUFHLE9BQU9BLEtBQU0sRUFBbkQsQ0FBekM7QUFFQSxNQUFNTyxpQkFBaUIsR0FBRyxxQkFBTyxDQUFDbEIsTUFBRCxFQUFTQyxHQUFULEVBQWNDLEdBQWQsS0FBc0I7QUFFckQsTUFBSWlCLE9BQU8sR0FBRyxFQUFkOztBQUNBLE1BQUlsQixHQUFHLENBQUNtQixJQUFSLEVBQWM7QUFDWixRQUFJO0FBQ0ZELE1BQUFBLE9BQU8sR0FBR3pCLGdCQUFFMkIsUUFBRixDQUFXM0IsZ0JBQUU0QixRQUFGLENBQVdyQixHQUFHLENBQUNtQixJQUFmLElBQXVCbkIsR0FBRyxDQUFDbUIsSUFBM0IsR0FBa0NHLElBQUksQ0FBQ0MsU0FBTCxDQUFldkIsR0FBRyxDQUFDbUIsSUFBbkIsQ0FBN0MsRUFBdUU7QUFDL0VLLFFBQUFBLE1BQU0sRUFBRUM7QUFEdUUsT0FBdkUsQ0FBVjtBQUdELEtBSkQsQ0FJRSxPQUFPQyxHQUFQLEVBQVksQ0FBRTtBQUNqQjs7QUFDRGIsa0JBQUlDLElBQUosQ0FBU0UseUJBQXlCLENBQUNqQixNQUFELEVBQVNDLEdBQVQsRUFBY0MsR0FBZCxDQUFsQyxFQUFzRGlCLE9BQU8sQ0FBQ1AsSUFBOUQ7QUFDRCxDQVh5QixFQVd2QjtBQUFDZ0IsRUFBQUEsU0FBUyxFQUFFO0FBQVosQ0FYdUIsQ0FBMUIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgXyBmcm9tICdsb2Rhc2gnO1xuaW1wb3J0ICdjb2xvcnMnO1xuaW1wb3J0IG1vcmdhbiBmcm9tICdtb3JnYW4nO1xuaW1wb3J0IGxvZyBmcm9tICcuL2xvZ2dlcic7XG5pbXBvcnQgeyBNQVhfTE9HX0JPRFlfTEVOR1RIIH0gZnJvbSAnLi4vY29uc3RhbnRzJztcblxuXG4vLyBDb3BpZWQgdGhlIG1vcmdhbiBjb21waWxlIGZ1bmN0aW9uIG92ZXIgc28gdGhhdCBjb29sZXIgZm9ybWF0c1xuLy8gbWF5IGJlIGNvbmZpZ3VyZWRcbmZ1bmN0aW9uIGNvbXBpbGUgKGZtdCkge1xuICAvLyBlc2NhcGUgcXVvdGVzXG4gIGZtdCA9IGZtdC5yZXBsYWNlKC9cIi9nLCAnXFxcXFwiJyk7XG4gIGZtdCA9IGZtdC5yZXBsYWNlKC86KFstXFx3XXsyLH0pKD86XFxbKFteXFxdXSspXFxdKT8vZyxcbiAgICBmdW5jdGlvbiByZXBsYWNlIChfLCBuYW1lLCBhcmcpIHtcbiAgICAgIHJldHVybiBgXCJcXG4gICAgKyAodG9rZW5zW1wiJHtuYW1lfVwiXShyZXEsIHJlcywgXCIke2FyZ31cIikgfHwgXCItXCIpICsgXCJgO1xuICAgIH0pO1xuICBsZXQganMgPSBgICByZXR1cm4gXCIke2ZtdH1cIjtgO1xuICByZXR1cm4gbmV3IEZ1bmN0aW9uKCd0b2tlbnMsIHJlcSwgcmVzJywganMpO1xufVxuXG5mdW5jdGlvbiByZXF1ZXN0RW5kTG9nZ2luZ0Zvcm1hdCAodG9rZW5zLCByZXEsIHJlcykge1xuICBsZXQgc3RhdHVzID0gcmVzLnN0YXR1c0NvZGU7XG4gIGxldCBzdGF0dXNTdHIgPSAnOnN0YXR1cyc7XG4gIGlmIChzdGF0dXMgPj0gNTAwKSB7XG4gICAgc3RhdHVzU3RyID0gc3RhdHVzU3RyLnJlZDtcbiAgfSBlbHNlIGlmIChzdGF0dXMgPj0gNDAwKSB7XG4gICAgc3RhdHVzU3RyID0gc3RhdHVzU3RyLnllbGxvdztcbiAgfSBlbHNlIGlmIChzdGF0dXMgPj0gMzAwKSB7XG4gICAgc3RhdHVzU3RyID0gc3RhdHVzU3RyLmN5YW47XG4gIH0gZWxzZSB7XG4gICAgc3RhdHVzU3RyID0gc3RhdHVzU3RyLmdyZWVuO1xuICB9XG4gIGxldCBmbiA9IGNvbXBpbGUoYCR7JzwtLSA6bWV0aG9kIDp1cmwgJy53aGl0ZX0ke3N0YXR1c1N0cn0gJHsnOnJlc3BvbnNlLXRpbWUgbXMgLSA6cmVzW2NvbnRlbnQtbGVuZ3RoXScuZ3JleX1gKTtcbiAgcmV0dXJuIGZuKHRva2VucywgcmVxLCByZXMpO1xufVxuXG5jb25zdCBlbmRMb2dGb3JtYXR0ZXIgPSBtb3JnYW4oKHRva2VucywgcmVxLCByZXMpID0+IHtcbiAgbG9nLmluZm8ocmVxdWVzdEVuZExvZ2dpbmdGb3JtYXQodG9rZW5zLCByZXEsIHJlcyksXG4gICAgKHJlcy5qc29uUmVzcCB8fCAnJykuZ3JleSk7XG59KTtcblxuY29uc3QgcmVxdWVzdFN0YXJ0TG9nZ2luZ0Zvcm1hdCA9IGNvbXBpbGUoYCR7Jy0tPicud2hpdGV9ICR7JzptZXRob2QnLndoaXRlfSAkeyc6dXJsJy53aGl0ZX1gKTtcblxuY29uc3Qgc3RhcnRMb2dGb3JtYXR0ZXIgPSBtb3JnYW4oKHRva2VucywgcmVxLCByZXMpID0+IHtcbiAgLy8gbW9yZ2FuIG91dHB1dCBpcyByZWRpcmVjdGVkIHN0cmFpZ2h0IHRvIHdpbnN0b25cbiAgbGV0IHJlcUJvZHkgPSAnJztcbiAgaWYgKHJlcS5ib2R5KSB7XG4gICAgdHJ5IHtcbiAgICAgIHJlcUJvZHkgPSBfLnRydW5jYXRlKF8uaXNTdHJpbmcocmVxLmJvZHkpID8gcmVxLmJvZHkgOiBKU09OLnN0cmluZ2lmeShyZXEuYm9keSksIHtcbiAgICAgICAgbGVuZ3RoOiBNQVhfTE9HX0JPRFlfTEVOR1RILFxuICAgICAgfSk7XG4gICAgfSBjYXRjaCAoaWduKSB7fVxuICB9XG4gIGxvZy5pbmZvKHJlcXVlc3RTdGFydExvZ2dpbmdGb3JtYXQodG9rZW5zLCByZXEsIHJlcyksIHJlcUJvZHkuZ3JleSk7XG59LCB7aW1tZWRpYXRlOiB0cnVlfSk7XG5cbmV4cG9ydCB7IGVuZExvZ0Zvcm1hdHRlciwgc3RhcnRMb2dGb3JtYXR0ZXIgfTtcbiJdLCJmaWxlIjoibGliL2V4cHJlc3MvZXhwcmVzcy1sb2dnaW5nLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uIn0=
72
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9leHByZXNzL2V4cHJlc3MtbG9nZ2luZy5qcyJdLCJuYW1lcyI6WyJjb21waWxlIiwiZm10IiwicmVwbGFjZSIsIl8iLCJuYW1lIiwiYXJnIiwianMiLCJGdW5jdGlvbiIsInJlcXVlc3RFbmRMb2dnaW5nRm9ybWF0IiwidG9rZW5zIiwicmVxIiwicmVzIiwic3RhdHVzIiwic3RhdHVzQ29kZSIsInN0YXR1c1N0ciIsInJlZCIsInllbGxvdyIsImN5YW4iLCJncmVlbiIsImZuIiwid2hpdGUiLCJncmV5IiwiZW5kTG9nRm9ybWF0dGVyIiwibG9nIiwiaW5mbyIsImpzb25SZXNwIiwicmVxdWVzdFN0YXJ0TG9nZ2luZ0Zvcm1hdCIsInN0YXJ0TG9nRm9ybWF0dGVyIiwicmVxQm9keSIsImJvZHkiLCJ0cnVuY2F0ZSIsImlzU3RyaW5nIiwiSlNPTiIsInN0cmluZ2lmeSIsImxlbmd0aCIsIk1BWF9MT0dfQk9EWV9MRU5HVEgiLCJpZ24iLCJpbW1lZGlhdGUiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBS0EsU0FBU0EsT0FBVCxDQUFrQkMsR0FBbEIsRUFBdUI7QUFFckJBLEVBQUFBLEdBQUcsR0FBR0EsR0FBRyxDQUFDQyxPQUFKLENBQVksSUFBWixFQUFrQixLQUFsQixDQUFOO0FBQ0FELEVBQUFBLEdBQUcsR0FBR0EsR0FBRyxDQUFDQyxPQUFKLENBQVksZ0NBQVosRUFDSixTQUFTQSxPQUFULENBQWtCQyxDQUFsQixFQUFxQkMsSUFBckIsRUFBMkJDLEdBQTNCLEVBQWdDO0FBQzlCLFdBQVEscUJBQW9CRCxJQUFLLGlCQUFnQkMsR0FBSSxnQkFBckQ7QUFDRCxHQUhHLENBQU47QUFJQSxNQUFJQyxFQUFFLEdBQUksYUFBWUwsR0FBSSxJQUExQjtBQUNBLFNBQU8sSUFBSU0sUUFBSixDQUFhLGtCQUFiLEVBQWlDRCxFQUFqQyxDQUFQO0FBQ0Q7O0FBRUQsU0FBU0UsdUJBQVQsQ0FBa0NDLE1BQWxDLEVBQTBDQyxHQUExQyxFQUErQ0MsR0FBL0MsRUFBb0Q7QUFDbEQsTUFBSUMsTUFBTSxHQUFHRCxHQUFHLENBQUNFLFVBQWpCO0FBQ0EsTUFBSUMsU0FBUyxHQUFHLFNBQWhCOztBQUNBLE1BQUlGLE1BQU0sSUFBSSxHQUFkLEVBQW1CO0FBQ2pCRSxJQUFBQSxTQUFTLEdBQUdBLFNBQVMsQ0FBQ0MsR0FBdEI7QUFDRCxHQUZELE1BRU8sSUFBSUgsTUFBTSxJQUFJLEdBQWQsRUFBbUI7QUFDeEJFLElBQUFBLFNBQVMsR0FBR0EsU0FBUyxDQUFDRSxNQUF0QjtBQUNELEdBRk0sTUFFQSxJQUFJSixNQUFNLElBQUksR0FBZCxFQUFtQjtBQUN4QkUsSUFBQUEsU0FBUyxHQUFHQSxTQUFTLENBQUNHLElBQXRCO0FBQ0QsR0FGTSxNQUVBO0FBQ0xILElBQUFBLFNBQVMsR0FBR0EsU0FBUyxDQUFDSSxLQUF0QjtBQUNEOztBQUNELE1BQUlDLEVBQUUsR0FBR25CLE9BQU8sQ0FBRSxHQUFFLG9CQUFvQm9CLEtBQU0sR0FBRU4sU0FBVSxJQUFHLDJDQUEyQ08sSUFBSyxFQUE3RixDQUFoQjtBQUNBLFNBQU9GLEVBQUUsQ0FBQ1YsTUFBRCxFQUFTQyxHQUFULEVBQWNDLEdBQWQsQ0FBVDtBQUNEOztBQUVELE1BQU1XLGVBQWUsR0FBRyxxQkFBTyxDQUFDYixNQUFELEVBQVNDLEdBQVQsRUFBY0MsR0FBZCxLQUFzQjtBQUNuRFksa0JBQUlDLElBQUosQ0FBU2hCLHVCQUF1QixDQUFDQyxNQUFELEVBQVNDLEdBQVQsRUFBY0MsR0FBZCxDQUFoQyxFQUNFLENBQUNBLEdBQUcsQ0FBQ2MsUUFBSixJQUFnQixFQUFqQixFQUFxQkosSUFEdkI7QUFFRCxDQUh1QixDQUF4Qjs7QUFLQSxNQUFNSyx5QkFBeUIsR0FBRzFCLE9BQU8sQ0FBRSxHQUFFLE1BQU1vQixLQUFNLElBQUcsVUFBVUEsS0FBTSxJQUFHLE9BQU9BLEtBQU0sRUFBbkQsQ0FBekM7QUFFQSxNQUFNTyxpQkFBaUIsR0FBRyxxQkFBTyxDQUFDbEIsTUFBRCxFQUFTQyxHQUFULEVBQWNDLEdBQWQsS0FBc0I7QUFFckQsTUFBSWlCLE9BQU8sR0FBRyxFQUFkOztBQUNBLE1BQUlsQixHQUFHLENBQUNtQixJQUFSLEVBQWM7QUFDWixRQUFJO0FBQ0ZELE1BQUFBLE9BQU8sR0FBR3pCLGdCQUFFMkIsUUFBRixDQUFXM0IsZ0JBQUU0QixRQUFGLENBQVdyQixHQUFHLENBQUNtQixJQUFmLElBQXVCbkIsR0FBRyxDQUFDbUIsSUFBM0IsR0FBa0NHLElBQUksQ0FBQ0MsU0FBTCxDQUFldkIsR0FBRyxDQUFDbUIsSUFBbkIsQ0FBN0MsRUFBdUU7QUFDL0VLLFFBQUFBLE1BQU0sRUFBRUM7QUFEdUUsT0FBdkUsQ0FBVjtBQUdELEtBSkQsQ0FJRSxPQUFPQyxHQUFQLEVBQVksQ0FBRTtBQUNqQjs7QUFDRGIsa0JBQUlDLElBQUosQ0FBU0UseUJBQXlCLENBQUNqQixNQUFELEVBQVNDLEdBQVQsRUFBY0MsR0FBZCxDQUFsQyxFQUFzRGlCLE9BQU8sQ0FBQ1AsSUFBOUQ7QUFDRCxDQVh5QixFQVd2QjtBQUFDZ0IsRUFBQUEsU0FBUyxFQUFFO0FBQVosQ0FYdUIsQ0FBMUIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgXyBmcm9tICdsb2Rhc2gnO1xuaW1wb3J0ICdAZGFiaC9jb2xvcnMnO1xuaW1wb3J0IG1vcmdhbiBmcm9tICdtb3JnYW4nO1xuaW1wb3J0IGxvZyBmcm9tICcuL2xvZ2dlcic7XG5pbXBvcnQgeyBNQVhfTE9HX0JPRFlfTEVOR1RIIH0gZnJvbSAnLi4vY29uc3RhbnRzJztcblxuXG4vLyBDb3BpZWQgdGhlIG1vcmdhbiBjb21waWxlIGZ1bmN0aW9uIG92ZXIgc28gdGhhdCBjb29sZXIgZm9ybWF0c1xuLy8gbWF5IGJlIGNvbmZpZ3VyZWRcbmZ1bmN0aW9uIGNvbXBpbGUgKGZtdCkge1xuICAvLyBlc2NhcGUgcXVvdGVzXG4gIGZtdCA9IGZtdC5yZXBsYWNlKC9cIi9nLCAnXFxcXFwiJyk7XG4gIGZtdCA9IGZtdC5yZXBsYWNlKC86KFstXFx3XXsyLH0pKD86XFxbKFteXFxdXSspXFxdKT8vZyxcbiAgICBmdW5jdGlvbiByZXBsYWNlIChfLCBuYW1lLCBhcmcpIHtcbiAgICAgIHJldHVybiBgXCJcXG4gICAgKyAodG9rZW5zW1wiJHtuYW1lfVwiXShyZXEsIHJlcywgXCIke2FyZ31cIikgfHwgXCItXCIpICsgXCJgO1xuICAgIH0pO1xuICBsZXQganMgPSBgICByZXR1cm4gXCIke2ZtdH1cIjtgO1xuICByZXR1cm4gbmV3IEZ1bmN0aW9uKCd0b2tlbnMsIHJlcSwgcmVzJywganMpO1xufVxuXG5mdW5jdGlvbiByZXF1ZXN0RW5kTG9nZ2luZ0Zvcm1hdCAodG9rZW5zLCByZXEsIHJlcykge1xuICBsZXQgc3RhdHVzID0gcmVzLnN0YXR1c0NvZGU7XG4gIGxldCBzdGF0dXNTdHIgPSAnOnN0YXR1cyc7XG4gIGlmIChzdGF0dXMgPj0gNTAwKSB7XG4gICAgc3RhdHVzU3RyID0gc3RhdHVzU3RyLnJlZDtcbiAgfSBlbHNlIGlmIChzdGF0dXMgPj0gNDAwKSB7XG4gICAgc3RhdHVzU3RyID0gc3RhdHVzU3RyLnllbGxvdztcbiAgfSBlbHNlIGlmIChzdGF0dXMgPj0gMzAwKSB7XG4gICAgc3RhdHVzU3RyID0gc3RhdHVzU3RyLmN5YW47XG4gIH0gZWxzZSB7XG4gICAgc3RhdHVzU3RyID0gc3RhdHVzU3RyLmdyZWVuO1xuICB9XG4gIGxldCBmbiA9IGNvbXBpbGUoYCR7JzwtLSA6bWV0aG9kIDp1cmwgJy53aGl0ZX0ke3N0YXR1c1N0cn0gJHsnOnJlc3BvbnNlLXRpbWUgbXMgLSA6cmVzW2NvbnRlbnQtbGVuZ3RoXScuZ3JleX1gKTtcbiAgcmV0dXJuIGZuKHRva2VucywgcmVxLCByZXMpO1xufVxuXG5jb25zdCBlbmRMb2dGb3JtYXR0ZXIgPSBtb3JnYW4oKHRva2VucywgcmVxLCByZXMpID0+IHtcbiAgbG9nLmluZm8ocmVxdWVzdEVuZExvZ2dpbmdGb3JtYXQodG9rZW5zLCByZXEsIHJlcyksXG4gICAgKHJlcy5qc29uUmVzcCB8fCAnJykuZ3JleSk7XG59KTtcblxuY29uc3QgcmVxdWVzdFN0YXJ0TG9nZ2luZ0Zvcm1hdCA9IGNvbXBpbGUoYCR7Jy0tPicud2hpdGV9ICR7JzptZXRob2QnLndoaXRlfSAkeyc6dXJsJy53aGl0ZX1gKTtcblxuY29uc3Qgc3RhcnRMb2dGb3JtYXR0ZXIgPSBtb3JnYW4oKHRva2VucywgcmVxLCByZXMpID0+IHtcbiAgLy8gbW9yZ2FuIG91dHB1dCBpcyByZWRpcmVjdGVkIHN0cmFpZ2h0IHRvIHdpbnN0b25cbiAgbGV0IHJlcUJvZHkgPSAnJztcbiAgaWYgKHJlcS5ib2R5KSB7XG4gICAgdHJ5IHtcbiAgICAgIHJlcUJvZHkgPSBfLnRydW5jYXRlKF8uaXNTdHJpbmcocmVxLmJvZHkpID8gcmVxLmJvZHkgOiBKU09OLnN0cmluZ2lmeShyZXEuYm9keSksIHtcbiAgICAgICAgbGVuZ3RoOiBNQVhfTE9HX0JPRFlfTEVOR1RILFxuICAgICAgfSk7XG4gICAgfSBjYXRjaCAoaWduKSB7fVxuICB9XG4gIGxvZy5pbmZvKHJlcXVlc3RTdGFydExvZ2dpbmdGb3JtYXQodG9rZW5zLCByZXEsIHJlcyksIHJlcUJvZHkuZ3JleSk7XG59LCB7aW1tZWRpYXRlOiB0cnVlfSk7XG5cbmV4cG9ydCB7IGVuZExvZ0Zvcm1hdHRlciwgc3RhcnRMb2dGb3JtYXR0ZXIgfTtcbiJdLCJmaWxlIjoibGliL2V4cHJlc3MvZXhwcmVzcy1sb2dnaW5nLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uIn0=