@appium/base-driver 8.7.2 → 9.0.0

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 (121) hide show
  1. package/build/lib/basedriver/capabilities.d.ts +11 -163
  2. package/build/lib/basedriver/capabilities.d.ts.map +1 -1
  3. package/build/lib/basedriver/capabilities.js +354 -236
  4. package/build/lib/basedriver/capabilities.js.map +1 -1
  5. package/build/lib/basedriver/commands/event.d.ts +7 -6
  6. package/build/lib/basedriver/commands/event.d.ts.map +1 -1
  7. package/build/lib/basedriver/commands/event.js +55 -35
  8. package/build/lib/basedriver/commands/event.js.map +1 -1
  9. package/build/lib/basedriver/commands/execute.d.ts +7 -6
  10. package/build/lib/basedriver/commands/execute.d.ts.map +1 -1
  11. package/build/lib/basedriver/commands/execute.js +66 -58
  12. package/build/lib/basedriver/commands/execute.js.map +1 -1
  13. package/build/lib/basedriver/commands/find.d.ts +9 -7
  14. package/build/lib/basedriver/commands/find.d.ts.map +1 -1
  15. package/build/lib/basedriver/commands/find.js +102 -54
  16. package/build/lib/basedriver/commands/find.js.map +1 -1
  17. package/build/lib/basedriver/commands/index.d.ts +3 -7
  18. package/build/lib/basedriver/commands/index.d.ts.map +1 -1
  19. package/build/lib/basedriver/commands/index.js +30 -33
  20. package/build/lib/basedriver/commands/index.js.map +1 -1
  21. package/build/lib/basedriver/commands/log.d.ts +8 -9
  22. package/build/lib/basedriver/commands/log.d.ts.map +1 -1
  23. package/build/lib/basedriver/commands/log.js +54 -38
  24. package/build/lib/basedriver/commands/log.js.map +1 -1
  25. package/build/lib/basedriver/commands/session.d.ts +7 -6
  26. package/build/lib/basedriver/commands/session.d.ts.map +1 -1
  27. package/build/lib/basedriver/commands/session.js +46 -39
  28. package/build/lib/basedriver/commands/session.js.map +1 -1
  29. package/build/lib/basedriver/commands/settings.d.ts +7 -7
  30. package/build/lib/basedriver/commands/settings.d.ts.map +1 -1
  31. package/build/lib/basedriver/commands/settings.js +35 -28
  32. package/build/lib/basedriver/commands/settings.js.map +1 -1
  33. package/build/lib/basedriver/commands/timeout.d.ts +7 -5
  34. package/build/lib/basedriver/commands/timeout.d.ts.map +1 -1
  35. package/build/lib/basedriver/commands/timeout.js +144 -162
  36. package/build/lib/basedriver/commands/timeout.js.map +1 -1
  37. package/build/lib/basedriver/core.d.ts +6 -157
  38. package/build/lib/basedriver/core.d.ts.map +1 -1
  39. package/build/lib/basedriver/core.js +361 -230
  40. package/build/lib/basedriver/core.js.map +1 -1
  41. package/build/lib/basedriver/desired-caps.js +80 -110
  42. package/build/lib/basedriver/desired-caps.js.map +1 -1
  43. package/build/lib/basedriver/device-settings.js +57 -62
  44. package/build/lib/basedriver/device-settings.js.map +1 -1
  45. package/build/lib/basedriver/driver.d.ts +21 -267
  46. package/build/lib/basedriver/driver.d.ts.map +1 -1
  47. package/build/lib/basedriver/driver.js +362 -258
  48. package/build/lib/basedriver/driver.js.map +1 -1
  49. package/build/lib/basedriver/helpers.js +500 -495
  50. package/build/lib/basedriver/helpers.js.map +1 -1
  51. package/build/lib/basedriver/logger.d.ts +1 -1
  52. package/build/lib/basedriver/logger.d.ts.map +1 -1
  53. package/build/lib/basedriver/logger.js +5 -15
  54. package/build/lib/basedriver/logger.js.map +1 -1
  55. package/build/lib/constants.js +14 -14
  56. package/build/lib/constants.js.map +1 -1
  57. package/build/lib/express/crash.js +8 -15
  58. package/build/lib/express/crash.js.map +1 -1
  59. package/build/lib/express/express-logging.js +49 -59
  60. package/build/lib/express/express-logging.js.map +1 -1
  61. package/build/lib/express/idempotency.js +125 -177
  62. package/build/lib/express/idempotency.js.map +1 -1
  63. package/build/lib/express/logger.d.ts +1 -1
  64. package/build/lib/express/logger.d.ts.map +1 -1
  65. package/build/lib/express/logger.js +5 -15
  66. package/build/lib/express/logger.js.map +1 -1
  67. package/build/lib/express/middleware.js +82 -107
  68. package/build/lib/express/middleware.js.map +1 -1
  69. package/build/lib/express/server.d.ts +17 -5
  70. package/build/lib/express/server.d.ts.map +1 -1
  71. package/build/lib/express/server.js +259 -224
  72. package/build/lib/express/server.js.map +1 -1
  73. package/build/lib/express/static.js +64 -81
  74. package/build/lib/express/static.js.map +1 -1
  75. package/build/lib/express/websocket.js +115 -87
  76. package/build/lib/express/websocket.js.map +1 -1
  77. package/build/lib/helpers/capabilities.d.ts +1 -59
  78. package/build/lib/helpers/capabilities.d.ts.map +1 -1
  79. package/build/lib/helpers/capabilities.js +72 -69
  80. package/build/lib/helpers/capabilities.js.map +1 -1
  81. package/build/lib/index.js +64 -180
  82. package/build/lib/index.js.map +1 -1
  83. package/build/lib/jsonwp-proxy/protocol-converter.js +215 -227
  84. package/build/lib/jsonwp-proxy/protocol-converter.js.map +1 -1
  85. package/build/lib/jsonwp-proxy/proxy.d.ts.map +1 -1
  86. package/build/lib/jsonwp-proxy/proxy.js +355 -393
  87. package/build/lib/jsonwp-proxy/proxy.js.map +1 -1
  88. package/build/lib/jsonwp-status/status.js +119 -130
  89. package/build/lib/jsonwp-status/status.js.map +1 -1
  90. package/build/lib/protocol/errors.d.ts +135 -32
  91. package/build/lib/protocol/errors.d.ts.map +1 -1
  92. package/build/lib/protocol/errors.js +871 -919
  93. package/build/lib/protocol/errors.js.map +1 -1
  94. package/build/lib/protocol/helpers.js +37 -37
  95. package/build/lib/protocol/helpers.js.map +1 -1
  96. package/build/lib/protocol/index.js +22 -109
  97. package/build/lib/protocol/index.js.map +1 -1
  98. package/build/lib/protocol/protocol.js +394 -350
  99. package/build/lib/protocol/protocol.js.map +1 -1
  100. package/build/lib/protocol/routes.d.ts +1238 -4
  101. package/build/lib/protocol/routes.d.ts.map +1 -1
  102. package/build/lib/protocol/routes.js +964 -1327
  103. package/build/lib/protocol/routes.js.map +1 -1
  104. package/build/lib/protocol/validators.js +32 -39
  105. package/build/lib/protocol/validators.js.map +1 -1
  106. package/build/tsconfig.tsbuildinfo +1 -1
  107. package/lib/basedriver/capabilities.js +80 -39
  108. package/lib/basedriver/commands/event.js +10 -5
  109. package/lib/basedriver/commands/execute.js +14 -9
  110. package/lib/basedriver/commands/find.js +18 -12
  111. package/lib/basedriver/commands/index.js +21 -16
  112. package/lib/basedriver/commands/log.js +24 -18
  113. package/lib/basedriver/commands/session.js +10 -5
  114. package/lib/basedriver/commands/settings.js +9 -6
  115. package/lib/basedriver/commands/timeout.js +10 -4
  116. package/lib/basedriver/core.js +2 -3
  117. package/lib/basedriver/driver.js +29 -18
  118. package/lib/express/server.js +6 -3
  119. package/lib/protocol/errors.js +155 -44
  120. package/lib/protocol/routes.js +11 -7
  121. package/package.json +14 -16
@@ -1,546 +1,551 @@
1
1
  "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
6
- exports.configureApp = configureApp;
7
- exports.default = void 0;
8
- exports.duplicateKeys = duplicateKeys;
9
- exports.generateDriverLogPrefix = generateDriverLogPrefix;
10
- exports.isPackageOrBundle = isPackageOrBundle;
11
- exports.parseCapsArray = parseCapsArray;
12
-
13
- require("source-map-support/register");
14
-
15
- var _lodash = _interopRequireDefault(require("lodash"));
16
-
17
- var _path = _interopRequireDefault(require("path"));
18
-
19
- var _url = _interopRequireDefault(require("url"));
20
-
21
- var _logger = _interopRequireDefault(require("./logger"));
22
-
23
- var _support = require("@appium/support");
24
-
25
- var _lruCache = _interopRequireDefault(require("lru-cache"));
26
-
27
- var _asyncLock = _interopRequireDefault(require("async-lock"));
28
-
29
- var _axios = _interopRequireDefault(require("axios"));
30
-
31
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
32
-
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.generateDriverLogPrefix = exports.parseCapsArray = exports.duplicateKeys = exports.isPackageOrBundle = exports.configureApp = void 0;
7
+ const lodash_1 = __importDefault(require("lodash"));
8
+ const path_1 = __importDefault(require("path"));
9
+ const url_1 = __importDefault(require("url"));
10
+ const logger_1 = __importDefault(require("./logger"));
11
+ const support_1 = require("@appium/support");
12
+ const lru_cache_1 = __importDefault(require("lru-cache"));
13
+ const async_lock_1 = __importDefault(require("async-lock"));
14
+ const axios_1 = __importDefault(require("axios"));
33
15
  const IPA_EXT = '.ipa';
34
16
  const ZIP_EXTS = ['.zip', IPA_EXT];
35
17
  const ZIP_MIME_TYPES = ['application/zip', 'application/x-zip-compressed', 'multipart/x-zip'];
36
- const CACHED_APPS_MAX_AGE = 1000 * 60 * 60 * 24;
18
+ const CACHED_APPS_MAX_AGE = 1000 * 60 * 60 * 24; // ms
37
19
  const MAX_CACHED_APPS = 1024;
38
- const APPLICATIONS_CACHE = new _lruCache.default({
39
- max: MAX_CACHED_APPS,
40
- ttl: CACHED_APPS_MAX_AGE,
41
- updateAgeOnGet: true,
42
- dispose: (app, {
43
- fullPath
44
- }) => {
45
- _logger.default.info(`The application '${app}' cached at '${fullPath}' has ` + `expired after ${CACHED_APPS_MAX_AGE}ms`);
46
-
47
- if (fullPath) {
48
- _support.fs.rimraf(fullPath);
49
- }
50
- },
51
- noDisposeOnSet: true
20
+ const APPLICATIONS_CACHE = new lru_cache_1.default({
21
+ max: MAX_CACHED_APPS,
22
+ ttl: CACHED_APPS_MAX_AGE,
23
+ updateAgeOnGet: true,
24
+ dispose: (app, { fullPath }) => {
25
+ logger_1.default.info(`The application '${app}' cached at '${fullPath}' has ` +
26
+ `expired after ${CACHED_APPS_MAX_AGE}ms`);
27
+ if (fullPath) {
28
+ support_1.fs.rimraf(fullPath);
29
+ }
30
+ },
31
+ noDisposeOnSet: true,
52
32
  });
53
- const APPLICATIONS_CACHE_GUARD = new _asyncLock.default();
33
+ const APPLICATIONS_CACHE_GUARD = new async_lock_1.default();
54
34
  const SANITIZE_REPLACEMENT = '-';
55
35
  const DEFAULT_BASENAME = 'appium-app';
56
36
  const APP_DOWNLOAD_TIMEOUT_MS = 120 * 1000;
57
37
  process.on('exit', () => {
58
- if (APPLICATIONS_CACHE.size === 0) {
59
- return;
60
- }
61
-
62
- const appPaths = [...APPLICATIONS_CACHE.values()].map(({
63
- fullPath
64
- }) => fullPath);
65
-
66
- _logger.default.debug(`Performing cleanup of ${appPaths.length} cached ` + _support.util.pluralize('application', appPaths.length));
67
-
68
- for (const appPath of appPaths) {
69
- try {
70
- _support.fs.rimrafSync(appPath);
71
- } catch (e) {
72
- _logger.default.warn(e.message);
38
+ if (APPLICATIONS_CACHE.size === 0) {
39
+ return;
40
+ }
41
+ const appPaths = [...APPLICATIONS_CACHE.values()].map(({ fullPath }) => fullPath);
42
+ logger_1.default.debug(`Performing cleanup of ${appPaths.length} cached ` +
43
+ support_1.util.pluralize('application', appPaths.length));
44
+ for (const appPath of appPaths) {
45
+ try {
46
+ // Asynchronous calls are not supported in onExit handler
47
+ support_1.fs.rimrafSync(appPath);
48
+ }
49
+ catch (e) {
50
+ logger_1.default.warn(e.message);
51
+ }
73
52
  }
74
- }
75
53
  });
76
-
77
54
  async function retrieveHeaders(link) {
78
- try {
79
- return (await (0, _axios.default)({
80
- url: link,
81
- method: 'HEAD',
82
- timeout: 5000
83
- })).headers;
84
- } catch (e) {
85
- _logger.default.info(`Cannot send HEAD request to '${link}'. Original error: ${e.message}`);
86
- }
87
-
88
- return {};
55
+ try {
56
+ return (await (0, axios_1.default)({
57
+ url: link,
58
+ method: 'HEAD',
59
+ timeout: 5000,
60
+ })).headers;
61
+ }
62
+ catch (e) {
63
+ logger_1.default.info(`Cannot send HEAD request to '${link}'. Original error: ${e.message}`);
64
+ }
65
+ return {};
89
66
  }
90
-
91
67
  function getCachedApplicationPath(link, currentAppProps = {}, cachedAppInfo = {}) {
92
- const refresh = () => {
93
- _logger.default.debug(`A fresh copy of the application is going to be downloaded from ${link}`);
94
-
95
- return null;
96
- };
97
-
98
- if (!_lodash.default.isPlainObject(cachedAppInfo) || !_lodash.default.isPlainObject(currentAppProps)) {
99
- return refresh();
100
- }
101
-
102
- const {
103
- lastModified: currentModified,
104
- immutable: currentImmutable,
105
- maxAge: currentMaxAge
106
- } = currentAppProps;
107
- const {
108
- lastModified,
109
- immutable,
110
- timestamp,
111
- fullPath
112
- } = cachedAppInfo;
113
-
114
- if (lastModified && currentModified) {
115
- if (currentModified.getTime() <= lastModified.getTime()) {
116
- _logger.default.debug(`The application at ${link} has not been modified since ${lastModified}`);
117
-
118
- return fullPath;
68
+ const refresh = () => {
69
+ logger_1.default.debug(`A fresh copy of the application is going to be downloaded from ${link}`);
70
+ return null;
71
+ };
72
+ if (!lodash_1.default.isPlainObject(cachedAppInfo) || !lodash_1.default.isPlainObject(currentAppProps)) {
73
+ // if an invalid arg is passed then assume cache miss
74
+ return refresh();
119
75
  }
120
-
121
- _logger.default.debug(`The application at ${link} has been modified since ${lastModified}`);
122
-
123
- return refresh();
124
- }
125
-
126
- if (immutable && currentImmutable) {
127
- _logger.default.debug(`The application at ${link} is immutable`);
128
-
129
- return fullPath;
130
- }
131
-
132
- if (currentMaxAge && timestamp) {
133
- const msLeft = timestamp + currentMaxAge * 1000 - Date.now();
134
-
135
- if (msLeft > 0) {
136
- _logger.default.debug(`The cached application '${_path.default.basename(fullPath)}' will expire in ${msLeft / 1000}s`);
137
-
138
- return fullPath;
76
+ const { lastModified: currentModified, immutable: currentImmutable,
77
+ // maxAge is in seconds
78
+ maxAge: currentMaxAge, } = currentAppProps;
79
+ const {
80
+ // Date instance
81
+ lastModified,
82
+ // boolean
83
+ immutable,
84
+ // Unix time in milliseconds
85
+ timestamp, fullPath, } = cachedAppInfo;
86
+ if (lastModified && currentModified) {
87
+ if (currentModified.getTime() <= lastModified.getTime()) {
88
+ logger_1.default.debug(`The application at ${link} has not been modified since ${lastModified}`);
89
+ return fullPath;
90
+ }
91
+ logger_1.default.debug(`The application at ${link} has been modified since ${lastModified}`);
92
+ return refresh();
139
93
  }
140
-
141
- _logger.default.debug(`The cached application '${_path.default.basename(fullPath)}' has expired`);
142
- }
143
-
144
- return refresh();
94
+ if (immutable && currentImmutable) {
95
+ logger_1.default.debug(`The application at ${link} is immutable`);
96
+ return fullPath;
97
+ }
98
+ if (currentMaxAge && timestamp) {
99
+ const msLeft = timestamp + currentMaxAge * 1000 - Date.now();
100
+ if (msLeft > 0) {
101
+ logger_1.default.debug(`The cached application '${path_1.default.basename(fullPath)}' will expire in ${msLeft / 1000}s`);
102
+ return fullPath;
103
+ }
104
+ logger_1.default.debug(`The cached application '${path_1.default.basename(fullPath)}' has expired`);
105
+ }
106
+ return refresh();
145
107
  }
146
-
147
108
  function verifyAppExtension(app, supportedAppExtensions) {
148
- if (supportedAppExtensions.map(_lodash.default.toLower).includes(_lodash.default.toLower(_path.default.extname(app)))) {
149
- return app;
150
- }
151
-
152
- throw new Error(`New app path '${app}' did not have ` + `${_support.util.pluralize('extension', supportedAppExtensions.length, false)}: ` + supportedAppExtensions);
109
+ if (supportedAppExtensions.map(lodash_1.default.toLower).includes(lodash_1.default.toLower(path_1.default.extname(app)))) {
110
+ return app;
111
+ }
112
+ throw new Error(`New app path '${app}' did not have ` +
113
+ `${support_1.util.pluralize('extension', supportedAppExtensions.length, false)}: ` +
114
+ supportedAppExtensions);
153
115
  }
154
-
155
116
  async function calculateFolderIntegrity(folderPath) {
156
- return (await _support.fs.glob('**/*', {
157
- cwd: folderPath,
158
- strict: false,
159
- nosort: true
160
- })).length;
117
+ return (await support_1.fs.glob('**/*', { cwd: folderPath, strict: false, nosort: true })).length;
161
118
  }
162
-
163
119
  async function calculateFileIntegrity(filePath) {
164
- return await _support.fs.hash(filePath);
120
+ return await support_1.fs.hash(filePath);
165
121
  }
166
-
167
122
  async function isAppIntegrityOk(currentPath, expectedIntegrity = {}) {
168
- if (!(await _support.fs.exists(currentPath))) {
169
- return false;
170
- }
171
-
172
- 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);
123
+ if (!(await support_1.fs.exists(currentPath))) {
124
+ return false;
125
+ }
126
+ // Folder integrity check is simple:
127
+ // Verify the previous amount of files is not greater than the current one.
128
+ // We don't want to use equality comparison because of an assumption that the OS might
129
+ // create some unwanted service files/cached inside of that folder or its subfolders.
130
+ // Ofc, validating the hash sum of each file (or at least of file path) would be much
131
+ // more precise, but we don't need to be very precise here and also don't want to
132
+ // overuse RAM and have a performance drop.
133
+ return (await support_1.fs.stat(currentPath)).isDirectory()
134
+ ? (await calculateFolderIntegrity(currentPath)) >= expectedIntegrity?.folder
135
+ : (await calculateFileIntegrity(currentPath)) === expectedIntegrity?.file;
173
136
  }
174
-
175
- async function configureApp(app, options = {}) {
176
- if (!_lodash.default.isString(app)) {
177
- return;
178
- }
179
-
180
- let supportedAppExtensions;
181
- const onPostProcess = !_lodash.default.isString(options) && !_lodash.default.isArray(options) ? options.onPostProcess : undefined;
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`);
193
- }
194
-
195
- let newApp = app;
196
- let shouldUnzipApp = false;
197
- let packageHash = null;
198
- let headers = null;
199
- const remoteAppProps = {
200
- lastModified: null,
201
- immutable: false,
202
- maxAge: null
203
- };
204
-
205
- const {
206
- protocol,
207
- pathname
208
- } = _url.default.parse(newApp);
209
-
210
- const isUrl = protocol === null ? false : ['http:', 'https:'].includes(protocol);
211
- const cachedAppInfo = APPLICATIONS_CACHE.get(app);
212
- return await APPLICATIONS_CACHE_GUARD.acquire(app, async () => {
213
- if (isUrl) {
214
- _logger.default.info(`Using downloadable app '${newApp}'`);
215
-
216
- headers = await retrieveHeaders(newApp);
217
-
218
- if (!_lodash.default.isEmpty(headers)) {
219
- if (headers['last-modified']) {
220
- remoteAppProps.lastModified = new Date(headers['last-modified']);
137
+ /**
138
+ * @typedef PostProcessOptions
139
+ * @property {?Object} cachedAppInfo The information about the previously cached app instance (if exists):
140
+ * - packageHash: SHA1 hash of the package if it is a file and not a folder
141
+ * - lastModified: Optional Date instance, the value of file's `Last-Modified` header
142
+ * - immutable: Optional boolean value. Contains true if the file has an `immutable` mark
143
+ * in `Cache-control` header
144
+ * - maxAge: Optional integer representation of `maxAge` parameter in `Cache-control` header
145
+ * - timestamp: The timestamp this item has been added to the cache (measured in Unix epoch
146
+ * milliseconds)
147
+ * - integrity: An object containing either `file` property with SHA1 hash of the file
148
+ * or `folder` property with total amount of cached files and subfolders
149
+ * - fullPath: the full path to the cached app
150
+ * @property {boolean} isUrl Whether the app has been downloaded from a remote URL
151
+ * @property {?Object} headers Optional headers object. Only present if `isUrl` is true and if the server
152
+ * responds to HEAD requests. All header names are normalized to lowercase.
153
+ * @property {string} appPath A string containing full path to the preprocessed application package (either
154
+ * downloaded or a local one)
155
+ */
156
+ /**
157
+ * @typedef PostProcessResult
158
+ * @property {string} appPath The full past to the post-processed application package on the
159
+ * local file system (might be a file or a folder path)
160
+ */
161
+ /**
162
+ * @typedef ConfigureAppOptions
163
+ * @property {(obj: PostProcessOptions) => (Promise<PostProcessResult|undefined>|PostProcessResult|undefined)} [onPostProcess]
164
+ * Optional function, which should be applied
165
+ * to the application after it is downloaded/preprocessed. This function may be async
166
+ * and is expected to accept single object parameter.
167
+ * The function is expected to either return a falsy value, which means the app must not be
168
+ * cached and a fresh copy of it is downloaded each time. If this function returns an object
169
+ * containing `appPath` property then the integrity of it will be verified and stored into
170
+ * the cache.
171
+ * @property {string[]} supportedExtensions List of supported application extensions (
172
+ * including starting dots). This property is mandatory and must not be empty.
173
+ */
174
+ /**
175
+ * Prepares an app to be used in an automated test. The app gets cached automatically
176
+ * if it is an archive or if it is downloaded from an URL.
177
+ * If the downloaded app has `.zip` extension, this method will unzip it.
178
+ * The unzip does not work when `onPostProcess` is provided.
179
+ *
180
+ * @param {string} app Either a full path to the app or a remote URL
181
+ * @param {string|string[]|ConfigureAppOptions} options
182
+ * @returns The full path to the resulting application bundle
183
+ */
184
+ async function configureApp(app, options = /** @type {ConfigureAppOptions} */ ({})) {
185
+ if (!lodash_1.default.isString(app)) {
186
+ // immediately shortcircuit if not given an app
187
+ return;
188
+ }
189
+ let supportedAppExtensions;
190
+ const onPostProcess = !lodash_1.default.isString(options) && !lodash_1.default.isArray(options) ? options.onPostProcess : undefined;
191
+ if (lodash_1.default.isString(options)) {
192
+ supportedAppExtensions = [options];
193
+ }
194
+ else if (lodash_1.default.isArray(options)) {
195
+ supportedAppExtensions = options;
196
+ }
197
+ else if (lodash_1.default.isPlainObject(options)) {
198
+ supportedAppExtensions = options.supportedExtensions;
199
+ }
200
+ if (lodash_1.default.isEmpty(supportedAppExtensions)) {
201
+ throw new Error(`One or more supported app extensions must be provided`);
202
+ }
203
+ let newApp = app;
204
+ let shouldUnzipApp = false;
205
+ let packageHash = null;
206
+ let headers = null;
207
+ /** @type {RemoteAppProps} */
208
+ const remoteAppProps = {
209
+ lastModified: null,
210
+ immutable: false,
211
+ maxAge: null,
212
+ };
213
+ const { protocol, pathname } = url_1.default.parse(newApp);
214
+ const isUrl = protocol === null ? false : ['http:', 'https:'].includes(protocol);
215
+ const cachedAppInfo = APPLICATIONS_CACHE.get(app);
216
+ return await APPLICATIONS_CACHE_GUARD.acquire(app, async () => {
217
+ if (isUrl) {
218
+ // Use the app from remote URL
219
+ logger_1.default.info(`Using downloadable app '${newApp}'`);
220
+ headers = await retrieveHeaders(newApp);
221
+ if (!lodash_1.default.isEmpty(headers)) {
222
+ if (headers['last-modified']) {
223
+ remoteAppProps.lastModified = new Date(headers['last-modified']);
224
+ }
225
+ logger_1.default.debug(`Last-Modified: ${headers['last-modified']}`);
226
+ if (headers['cache-control']) {
227
+ remoteAppProps.immutable = /\bimmutable\b/i.test(headers['cache-control']);
228
+ const maxAgeMatch = /\bmax-age=(\d+)\b/i.exec(headers['cache-control']);
229
+ if (maxAgeMatch) {
230
+ remoteAppProps.maxAge = parseInt(maxAgeMatch[1], 10);
231
+ }
232
+ }
233
+ logger_1.default.debug(`Cache-Control: ${headers['cache-control']}`);
234
+ }
235
+ const cachedPath = getCachedApplicationPath(app, remoteAppProps, cachedAppInfo);
236
+ if (cachedPath) {
237
+ if (await isAppIntegrityOk(cachedPath, cachedAppInfo?.integrity)) {
238
+ logger_1.default.info(`Reusing previously downloaded application at '${cachedPath}'`);
239
+ return verifyAppExtension(cachedPath, supportedAppExtensions);
240
+ }
241
+ logger_1.default.info(`The application at '${cachedPath}' does not exist anymore ` +
242
+ `or its integrity has been damaged. Deleting it from the internal cache`);
243
+ APPLICATIONS_CACHE.delete(app);
244
+ }
245
+ let fileName = null;
246
+ const basename = support_1.fs.sanitizeName(path_1.default.basename(decodeURIComponent(pathname ?? '')), {
247
+ replacement: SANITIZE_REPLACEMENT,
248
+ });
249
+ const extname = path_1.default.extname(basename);
250
+ // to determine if we need to unzip the app, we have a number of places
251
+ // to look: content type, content disposition, or the file extension
252
+ if (ZIP_EXTS.includes(extname)) {
253
+ fileName = basename;
254
+ shouldUnzipApp = true;
255
+ }
256
+ if (headers['content-type']) {
257
+ const ct = headers['content-type'];
258
+ logger_1.default.debug(`Content-Type: ${ct}`);
259
+ // the filetype may not be obvious for certain urls, so check the mime type too
260
+ if (ZIP_MIME_TYPES.some((mimeType) => new RegExp(`\\b${lodash_1.default.escapeRegExp(mimeType)}\\b`).test(ct))) {
261
+ if (!fileName) {
262
+ fileName = `${DEFAULT_BASENAME}.zip`;
263
+ }
264
+ shouldUnzipApp = true;
265
+ }
266
+ }
267
+ if (headers['content-disposition'] && /^attachment/i.test(headers['content-disposition'])) {
268
+ logger_1.default.debug(`Content-Disposition: ${headers['content-disposition']}`);
269
+ const match = /filename="([^"]+)/i.exec(headers['content-disposition']);
270
+ if (match) {
271
+ fileName = support_1.fs.sanitizeName(match[1], {
272
+ replacement: SANITIZE_REPLACEMENT,
273
+ });
274
+ shouldUnzipApp = shouldUnzipApp || ZIP_EXTS.includes(path_1.default.extname(fileName));
275
+ }
276
+ }
277
+ if (!fileName) {
278
+ // assign the default file name and the extension if none has been detected
279
+ const resultingName = basename
280
+ ? basename.substring(0, basename.length - extname.length)
281
+ : DEFAULT_BASENAME;
282
+ let resultingExt = extname;
283
+ if (!supportedAppExtensions.includes(resultingExt)) {
284
+ logger_1.default.info(`The current file extension '${resultingExt}' is not supported. ` +
285
+ `Defaulting to '${lodash_1.default.first(supportedAppExtensions)}'`);
286
+ resultingExt = /** @type {string} */ (lodash_1.default.first(supportedAppExtensions));
287
+ }
288
+ fileName = `${resultingName}${resultingExt}`;
289
+ }
290
+ const targetPath = await support_1.tempDir.path({
291
+ prefix: fileName,
292
+ suffix: '',
293
+ });
294
+ newApp = await downloadApp(newApp, targetPath);
221
295
  }
222
-
223
- _logger.default.debug(`Last-Modified: ${headers['last-modified']}`);
224
-
225
- if (headers['cache-control']) {
226
- remoteAppProps.immutable = /\bimmutable\b/i.test(headers['cache-control']);
227
- const maxAgeMatch = /\bmax-age=(\d+)\b/i.exec(headers['cache-control']);
228
-
229
- if (maxAgeMatch) {
230
- remoteAppProps.maxAge = parseInt(maxAgeMatch[1], 10);
231
- }
296
+ else if (await support_1.fs.exists(newApp)) {
297
+ // Use the local app
298
+ logger_1.default.info(`Using local app '${newApp}'`);
299
+ shouldUnzipApp = ZIP_EXTS.includes(path_1.default.extname(newApp));
232
300
  }
233
-
234
- _logger.default.debug(`Cache-Control: ${headers['cache-control']}`);
235
- }
236
-
237
- const cachedPath = getCachedApplicationPath(app, remoteAppProps, cachedAppInfo);
238
-
239
- if (cachedPath) {
240
- if (await isAppIntegrityOk(cachedPath, cachedAppInfo === null || cachedAppInfo === void 0 ? void 0 : cachedAppInfo.integrity)) {
241
- _logger.default.info(`Reusing previously downloaded application at '${cachedPath}'`);
242
-
243
- return verifyAppExtension(cachedPath, supportedAppExtensions);
301
+ else {
302
+ let errorMessage = `The application at '${newApp}' does not exist or is not accessible`;
303
+ // protocol value for 'C:\\temp' is 'c:', so we check the length as well
304
+ if (lodash_1.default.isString(protocol) && protocol.length > 2) {
305
+ errorMessage =
306
+ `The protocol '${protocol}' used in '${newApp}' is not supported. ` +
307
+ `Only http: and https: protocols are supported`;
308
+ }
309
+ throw new Error(errorMessage);
244
310
  }
245
-
246
- _logger.default.info(`The application at '${cachedPath}' does not exist anymore ` + `or its integrity has been damaged. Deleting it from the internal cache`);
247
-
248
- APPLICATIONS_CACHE.delete(app);
249
- }
250
-
251
- let fileName = null;
252
-
253
- const basename = _support.fs.sanitizeName(_path.default.basename(decodeURIComponent(pathname ?? '')), {
254
- replacement: SANITIZE_REPLACEMENT
255
- });
256
-
257
- const extname = _path.default.extname(basename);
258
-
259
- if (ZIP_EXTS.includes(extname)) {
260
- fileName = basename;
261
- shouldUnzipApp = true;
262
- }
263
-
264
- if (headers['content-type']) {
265
- const ct = headers['content-type'];
266
-
267
- _logger.default.debug(`Content-Type: ${ct}`);
268
-
269
- if (ZIP_MIME_TYPES.some(mimeType => new RegExp(`\\b${_lodash.default.escapeRegExp(mimeType)}\\b`).test(ct))) {
270
- if (!fileName) {
271
- fileName = `${DEFAULT_BASENAME}.zip`;
272
- }
273
-
274
- shouldUnzipApp = true;
275
- }
276
- }
277
-
278
- if (headers['content-disposition'] && /^attachment/i.test(headers['content-disposition'])) {
279
- _logger.default.debug(`Content-Disposition: ${headers['content-disposition']}`);
280
-
281
- const match = /filename="([^"]+)/i.exec(headers['content-disposition']);
282
-
283
- if (match) {
284
- fileName = _support.fs.sanitizeName(match[1], {
285
- replacement: SANITIZE_REPLACEMENT
286
- });
287
- shouldUnzipApp = shouldUnzipApp || ZIP_EXTS.includes(_path.default.extname(fileName));
311
+ const isPackageAFile = (await support_1.fs.stat(newApp)).isFile();
312
+ if (isPackageAFile) {
313
+ packageHash = await calculateFileIntegrity(newApp);
288
314
  }
289
- }
290
-
291
- if (!fileName) {
292
- const resultingName = basename ? basename.substring(0, basename.length - extname.length) : DEFAULT_BASENAME;
293
- let resultingExt = extname;
294
-
295
- if (!supportedAppExtensions.includes(resultingExt)) {
296
- _logger.default.info(`The current file extension '${resultingExt}' is not supported. ` + `Defaulting to '${_lodash.default.first(supportedAppExtensions)}'`);
297
-
298
- resultingExt = _lodash.default.first(supportedAppExtensions);
315
+ if (isPackageAFile && shouldUnzipApp && !lodash_1.default.isFunction(onPostProcess)) {
316
+ const archivePath = newApp;
317
+ if (packageHash === cachedAppInfo?.packageHash) {
318
+ const { fullPath } = cachedAppInfo;
319
+ if (await isAppIntegrityOk(fullPath, cachedAppInfo?.integrity)) {
320
+ if (archivePath !== app) {
321
+ await support_1.fs.rimraf(archivePath);
322
+ }
323
+ logger_1.default.info(`Will reuse previously cached application at '${fullPath}'`);
324
+ return verifyAppExtension(fullPath, supportedAppExtensions);
325
+ }
326
+ logger_1.default.info(`The application at '${fullPath}' does not exist anymore ` +
327
+ `or its integrity has been damaged. Deleting it from the cache`);
328
+ APPLICATIONS_CACHE.delete(app);
329
+ }
330
+ const tmpRoot = await support_1.tempDir.openDir();
331
+ try {
332
+ newApp = await unzipApp(archivePath, tmpRoot, supportedAppExtensions);
333
+ }
334
+ finally {
335
+ if (newApp !== archivePath && archivePath !== app) {
336
+ await support_1.fs.rimraf(archivePath);
337
+ }
338
+ }
339
+ logger_1.default.info(`Unzipped local app to '${newApp}'`);
299
340
  }
300
-
301
- fileName = `${resultingName}${resultingExt}`;
302
- }
303
-
304
- const targetPath = await _support.tempDir.path({
305
- prefix: fileName,
306
- suffix: ''
307
- });
308
- newApp = await downloadApp(newApp, targetPath);
309
- } else if (await _support.fs.exists(newApp)) {
310
- _logger.default.info(`Using local app '${newApp}'`);
311
-
312
- shouldUnzipApp = ZIP_EXTS.includes(_path.default.extname(newApp));
313
- } else {
314
- let errorMessage = `The application at '${newApp}' does not exist or is not accessible`;
315
-
316
- if (_lodash.default.isString(protocol) && protocol.length > 2) {
317
- errorMessage = `The protocol '${protocol}' used in '${newApp}' is not supported. ` + `Only http: and https: protocols are supported`;
318
- }
319
-
320
- throw new Error(errorMessage);
321
- }
322
-
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)) {
330
- const archivePath = newApp;
331
-
332
- if (packageHash === (cachedAppInfo === null || cachedAppInfo === void 0 ? void 0 : cachedAppInfo.packageHash)) {
333
- const {
334
- fullPath
335
- } = cachedAppInfo;
336
-
337
- if (await isAppIntegrityOk(fullPath, cachedAppInfo === null || cachedAppInfo === void 0 ? void 0 : cachedAppInfo.integrity)) {
338
- if (archivePath !== app) {
339
- await _support.fs.rimraf(archivePath);
340
- }
341
-
342
- _logger.default.info(`Will reuse previously cached application at '${fullPath}'`);
343
-
344
- return verifyAppExtension(fullPath, supportedAppExtensions);
341
+ else if (!path_1.default.isAbsolute(newApp)) {
342
+ newApp = path_1.default.resolve(process.cwd(), newApp);
343
+ logger_1.default.warn(`The current application path '${app}' is not absolute ` +
344
+ `and has been rewritten to '${newApp}'. Consider using absolute paths rather than relative`);
345
+ app = newApp;
345
346
  }
346
-
347
- _logger.default.info(`The application at '${fullPath}' does not exist anymore ` + `or its integrity has been damaged. Deleting it from the cache`);
348
-
349
- APPLICATIONS_CACHE.delete(app);
350
- }
351
-
352
- const tmpRoot = await _support.tempDir.openDir();
353
-
354
- try {
355
- newApp = await unzipApp(archivePath, tmpRoot, supportedAppExtensions);
356
- } finally {
357
- if (newApp !== archivePath && archivePath !== app) {
358
- await _support.fs.rimraf(archivePath);
347
+ const storeAppInCache = async (appPathToCache) => {
348
+ const cachedFullPath = cachedAppInfo?.fullPath;
349
+ if (cachedFullPath && cachedFullPath !== appPathToCache) {
350
+ await support_1.fs.rimraf(cachedFullPath);
351
+ }
352
+ const integrity = {};
353
+ if ((await support_1.fs.stat(appPathToCache)).isDirectory()) {
354
+ integrity.folder = await calculateFolderIntegrity(appPathToCache);
355
+ }
356
+ else {
357
+ integrity.file = await calculateFileIntegrity(appPathToCache);
358
+ }
359
+ APPLICATIONS_CACHE.set(app, {
360
+ ...remoteAppProps,
361
+ timestamp: Date.now(),
362
+ packageHash,
363
+ integrity,
364
+ fullPath: appPathToCache,
365
+ });
366
+ return appPathToCache;
367
+ };
368
+ if (lodash_1.default.isFunction(onPostProcess)) {
369
+ const result = await onPostProcess({
370
+ cachedAppInfo: lodash_1.default.clone(cachedAppInfo),
371
+ isUrl,
372
+ headers: lodash_1.default.clone(headers),
373
+ appPath: newApp,
374
+ });
375
+ return !result?.appPath || app === result?.appPath || !(await support_1.fs.exists(result?.appPath))
376
+ ? newApp
377
+ : await storeAppInCache(result.appPath);
359
378
  }
360
- }
361
-
362
- _logger.default.info(`Unzipped local app to '${newApp}'`);
363
- } else if (!_path.default.isAbsolute(newApp)) {
364
- newApp = _path.default.resolve(process.cwd(), newApp);
365
-
366
- _logger.default.warn(`The current application path '${app}' is not absolute ` + `and has been rewritten to '${newApp}'. Consider using absolute paths rather than relative`);
367
-
368
- app = newApp;
369
- }
370
-
371
- const storeAppInCache = async appPathToCache => {
372
- const cachedFullPath = cachedAppInfo === null || cachedAppInfo === void 0 ? void 0 : cachedAppInfo.fullPath;
373
-
374
- if (cachedFullPath && cachedFullPath !== appPathToCache) {
375
- await _support.fs.rimraf(cachedFullPath);
376
- }
377
-
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);
384
- }
385
-
386
- APPLICATIONS_CACHE.set(app, { ...remoteAppProps,
387
- timestamp: Date.now(),
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
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);
403
- }
404
-
405
- verifyAppExtension(newApp, supportedAppExtensions);
406
- return app !== newApp && (packageHash || _lodash.default.values(remoteAppProps).some(Boolean)) ? await storeAppInCache(newApp) : newApp;
407
- });
379
+ verifyAppExtension(newApp, supportedAppExtensions);
380
+ return app !== newApp && (packageHash || lodash_1.default.values(remoteAppProps).some(Boolean))
381
+ ? await storeAppInCache(newApp)
382
+ : newApp;
383
+ });
408
384
  }
409
-
385
+ exports.configureApp = configureApp;
410
386
  async function downloadApp(app, targetPath) {
411
- const {
412
- href
413
- } = _url.default.parse(app);
414
-
415
- try {
416
- await _support.net.downloadFile(href, targetPath, {
417
- timeout: APP_DOWNLOAD_TIMEOUT_MS
418
- });
419
- } catch (err) {
420
- throw new Error(`Unable to download the app: ${err.message}`);
421
- }
422
-
423
- return targetPath;
387
+ const { href } = url_1.default.parse(app);
388
+ try {
389
+ await support_1.net.downloadFile(href, targetPath, {
390
+ timeout: APP_DOWNLOAD_TIMEOUT_MS,
391
+ });
392
+ }
393
+ catch (err) {
394
+ throw new Error(`Unable to download the app: ${err.message}`);
395
+ }
396
+ return targetPath;
424
397
  }
425
-
398
+ /**
399
+ * Extracts the bundle from an archive into the given folder
400
+ *
401
+ * @param {string} zipPath Full path to the archive containing the bundle
402
+ * @param {string} dstRoot Full path to the folder where the extracted bundle
403
+ * should be placed
404
+ * @param {Array<string>|string} supportedAppExtensions The list of extensions
405
+ * the target application bundle supports, for example ['.apk', '.apks'] for
406
+ * Android packages
407
+ * @returns {Promise<string>} Full path to the bundle in the destination folder
408
+ * @throws {Error} If the given archive is invalid or no application bundles
409
+ * have been found inside
410
+ */
426
411
  async function unzipApp(zipPath, dstRoot, supportedAppExtensions) {
427
- await _support.zip.assertValidZip(zipPath);
428
-
429
- if (!_lodash.default.isArray(supportedAppExtensions)) {
430
- supportedAppExtensions = [supportedAppExtensions];
431
- }
432
-
433
- const tmpRoot = await _support.tempDir.openDir();
434
-
435
- try {
436
- _logger.default.debug(`Unzipping '${zipPath}'`);
437
-
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));
441
- const extractionOpts = {
442
- useSystemUnzip
443
- };
444
-
445
- if (_path.default.extname(zipPath) === IPA_EXT) {
446
- _logger.default.debug(`Enforcing UTF-8 encoding on the extracted file names for '${_path.default.basename(zipPath)}'`);
447
-
448
- extractionOpts.fileNamesEncoding = 'utf8';
412
+ await support_1.zip.assertValidZip(zipPath);
413
+ if (!lodash_1.default.isArray(supportedAppExtensions)) {
414
+ supportedAppExtensions = [supportedAppExtensions];
449
415
  }
450
-
451
- await _support.zip.extractAllTo(zipPath, tmpRoot, extractionOpts);
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)}`);
416
+ const tmpRoot = await support_1.tempDir.openDir();
417
+ try {
418
+ logger_1.default.debug(`Unzipping '${zipPath}'`);
419
+ const timer = new support_1.timing.Timer().start();
420
+ const useSystemUnzipEnv = process.env.APPIUM_PREFER_SYSTEM_UNZIP;
421
+ const useSystemUnzip = lodash_1.default.isEmpty(useSystemUnzipEnv) || !['0', 'false'].includes(lodash_1.default.toLower(useSystemUnzipEnv));
422
+ /**
423
+ * Attempt to use use the system `unzip` (e.g., `/usr/bin/unzip`) due
424
+ * to the significant performance improvement it provides over the native
425
+ * JS "unzip" implementation.
426
+ * @type {import('@appium/support/lib/zip').ExtractAllOptions}
427
+ */
428
+ const extractionOpts = { useSystemUnzip };
429
+ // https://github.com/appium/appium/issues/14100
430
+ if (path_1.default.extname(zipPath) === IPA_EXT) {
431
+ logger_1.default.debug(`Enforcing UTF-8 encoding on the extracted file names for '${path_1.default.basename(zipPath)}'`);
432
+ extractionOpts.fileNamesEncoding = 'utf8';
433
+ }
434
+ await support_1.zip.extractAllTo(zipPath, tmpRoot, extractionOpts);
435
+ const globPattern = `**/*.+(${supportedAppExtensions
436
+ .map((ext) => ext.replace(/^\./, ''))
437
+ .join('|')})`;
438
+ const sortedBundleItems = (await support_1.fs.glob(globPattern, {
439
+ cwd: tmpRoot,
440
+ strict: false,
441
+ // Get the top level match
442
+ })).sort((a, b) => a.split(path_1.default.sep).length - b.split(path_1.default.sep).length);
443
+ if (lodash_1.default.isEmpty(sortedBundleItems)) {
444
+ logger_1.default.errorAndThrow(`App unzipped OK, but we could not find any '${supportedAppExtensions}' ` +
445
+ support_1.util.pluralize('bundle', supportedAppExtensions.length, false) +
446
+ ` in it. Make sure your archive contains at least one package having ` +
447
+ `'${supportedAppExtensions}' ${support_1.util.pluralize('extension', supportedAppExtensions.length, false)}`);
448
+ }
449
+ logger_1.default.debug(`Extracted ${support_1.util.pluralize('bundle item', sortedBundleItems.length, true)} ` +
450
+ `from '${zipPath}' in ${Math.round(timer.getDuration().asMilliSeconds)}ms: ${sortedBundleItems}`);
451
+ const matchedBundle = /** @type {string} */ (lodash_1.default.first(sortedBundleItems));
452
+ logger_1.default.info(`Assuming '${matchedBundle}' is the correct bundle`);
453
+ const dstPath = path_1.default.resolve(dstRoot, path_1.default.basename(matchedBundle));
454
+ await support_1.fs.mv(path_1.default.resolve(tmpRoot, matchedBundle), dstPath, { mkdirp: true });
455
+ return dstPath;
456
+ }
457
+ finally {
458
+ await support_1.fs.rimraf(tmpRoot);
460
459
  }
461
-
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);
465
-
466
- _logger.default.info(`Assuming '${matchedBundle}' is the correct bundle`);
467
-
468
- const dstPath = _path.default.resolve(dstRoot, _path.default.basename(matchedBundle));
469
-
470
- await _support.fs.mv(_path.default.resolve(tmpRoot, matchedBundle), dstPath, {
471
- mkdirp: true
472
- });
473
- return dstPath;
474
- } finally {
475
- await _support.fs.rimraf(tmpRoot);
476
- }
477
460
  }
478
-
479
461
  function isPackageOrBundle(app) {
480
- return /^([a-zA-Z0-9\-_]+\.[a-zA-Z0-9\-_]+)+$/.test(app);
462
+ return /^([a-zA-Z0-9\-_]+\.[a-zA-Z0-9\-_]+)+$/.test(app);
481
463
  }
482
-
464
+ exports.isPackageOrBundle = isPackageOrBundle;
465
+ /**
466
+ * Finds all instances 'firstKey' and create a duplicate with the key 'secondKey',
467
+ * Do the same thing in reverse. If we find 'secondKey', create a duplicate with the key 'firstKey'.
468
+ *
469
+ * This will cause keys to be overwritten if the object contains 'firstKey' and 'secondKey'.
470
+
471
+ * @param {*} input Any type of input
472
+ * @param {String} firstKey The first key to duplicate
473
+ * @param {String} secondKey The second key to duplicate
474
+ */
483
475
  function duplicateKeys(input, firstKey, secondKey) {
484
- if (_lodash.default.isArray(input)) {
485
- return input.map(item => duplicateKeys(item, firstKey, secondKey));
486
- }
487
-
488
- if (_lodash.default.isPlainObject(input)) {
489
- const resultObj = {};
490
-
491
- for (let [key, value] of _lodash.default.toPairs(input)) {
492
- const recursivelyCalledValue = duplicateKeys(value, firstKey, secondKey);
493
-
494
- if (key === firstKey) {
495
- resultObj[secondKey] = recursivelyCalledValue;
496
- } else if (key === secondKey) {
497
- resultObj[firstKey] = recursivelyCalledValue;
498
- }
499
-
500
- resultObj[key] = recursivelyCalledValue;
476
+ // If array provided, recursively call on all elements
477
+ if (lodash_1.default.isArray(input)) {
478
+ return input.map((item) => duplicateKeys(item, firstKey, secondKey));
501
479
  }
502
-
503
- return resultObj;
504
- }
505
-
506
- return input;
480
+ // If object, create duplicates for keys and then recursively call on values
481
+ if (lodash_1.default.isPlainObject(input)) {
482
+ const resultObj = {};
483
+ for (let [key, value] of lodash_1.default.toPairs(input)) {
484
+ const recursivelyCalledValue = duplicateKeys(value, firstKey, secondKey);
485
+ if (key === firstKey) {
486
+ resultObj[secondKey] = recursivelyCalledValue;
487
+ }
488
+ else if (key === secondKey) {
489
+ resultObj[firstKey] = recursivelyCalledValue;
490
+ }
491
+ resultObj[key] = recursivelyCalledValue;
492
+ }
493
+ return resultObj;
494
+ }
495
+ // Base case. Return primitives without doing anything.
496
+ return input;
507
497
  }
508
-
498
+ exports.duplicateKeys = duplicateKeys;
499
+ /**
500
+ * Takes a desired capability and tries to JSON.parse it as an array,
501
+ * and either returns the parsed array or a singleton array.
502
+ *
503
+ * @param {string|Array<String>} cap A desired capability
504
+ */
509
505
  function parseCapsArray(cap) {
510
- if (_lodash.default.isArray(cap)) {
511
- return cap;
512
- }
513
-
514
- let parsedCaps;
515
-
516
- try {
517
- parsedCaps = JSON.parse(cap);
518
-
519
- if (_lodash.default.isArray(parsedCaps)) {
520
- return parsedCaps;
506
+ if (lodash_1.default.isArray(cap)) {
507
+ return cap;
521
508
  }
522
- } catch (ign) {
523
- _logger.default.warn(`Failed to parse capability as JSON array`);
524
- }
525
-
526
- if (_lodash.default.isString(cap)) {
527
- return [cap];
528
- }
529
-
530
- throw new Error(`must provide a string or JSON Array; received ${cap}`);
509
+ let parsedCaps;
510
+ try {
511
+ parsedCaps = JSON.parse(cap);
512
+ if (lodash_1.default.isArray(parsedCaps)) {
513
+ return parsedCaps;
514
+ }
515
+ }
516
+ catch (ign) {
517
+ logger_1.default.warn(`Failed to parse capability as JSON array`);
518
+ }
519
+ if (lodash_1.default.isString(cap)) {
520
+ return [cap];
521
+ }
522
+ throw new Error(`must provide a string or JSON Array; received ${cap}`);
531
523
  }
532
-
524
+ exports.parseCapsArray = parseCapsArray;
525
+ /**
526
+ * Generate a string that uniquely describes driver instance
527
+ *
528
+ * @param {import('@appium/types').Core} obj driver instance
529
+ * @param {string?} sessionId session identifier (if exists)
530
+ * @returns {string}
531
+ */
533
532
  function generateDriverLogPrefix(obj, sessionId = null) {
534
- const instanceName = `${obj.constructor.name}@${_support.node.getObjectId(obj).substring(0, 4)}`;
535
- return sessionId ? `${instanceName} (${sessionId.substring(0, 8)})` : instanceName;
533
+ const instanceName = `${obj.constructor.name}@${support_1.node.getObjectId(obj).substring(0, 4)}`;
534
+ return sessionId ? `${instanceName} (${sessionId.substring(0, 8)})` : instanceName;
536
535
  }
537
-
538
- var _default = {
539
- configureApp,
540
- isPackageOrBundle,
541
- duplicateKeys,
542
- parseCapsArray,
543
- generateDriverLogPrefix
536
+ exports.generateDriverLogPrefix = generateDriverLogPrefix;
537
+ /** @type {import('@appium/types').DriverHelpers} */
538
+ exports.default = {
539
+ configureApp,
540
+ isPackageOrBundle,
541
+ duplicateKeys,
542
+ parseCapsArray,
543
+ generateDriverLogPrefix,
544
544
  };
545
- exports.default = _default;
546
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["IPA_EXT","ZIP_EXTS","ZIP_MIME_TYPES","CACHED_APPS_MAX_AGE","MAX_CACHED_APPS","APPLICATIONS_CACHE","LRU","max","ttl","updateAgeOnGet","dispose","app","fullPath","logger","info","fs","rimraf","noDisposeOnSet","APPLICATIONS_CACHE_GUARD","AsyncLock","SANITIZE_REPLACEMENT","DEFAULT_BASENAME","APP_DOWNLOAD_TIMEOUT_MS","process","on","size","appPaths","values","map","debug","length","util","pluralize","appPath","rimrafSync","e","warn","message","retrieveHeaders","link","axios","url","method","timeout","headers","getCachedApplicationPath","currentAppProps","cachedAppInfo","refresh","_","isPlainObject","lastModified","currentModified","immutable","currentImmutable","maxAge","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","undefined","supportedExtensions","isEmpty","newApp","shouldUnzipApp","packageHash","remoteAppProps","protocol","pathname","parse","isUrl","get","acquire","test","maxAgeMatch","exec","parseInt","cachedPath","integrity","delete","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","generateDriverLogPrefix","obj","sessionId","instanceName","constructor","name","node","getObjectId"],"sources":["../../../lib/basedriver/helpers.js"],"sourcesContent":["import _ from 'lodash';\nimport path from 'path';\nimport url from 'url';\nimport logger from './logger';\nimport {tempDir, fs, util, zip, net, timing, node} 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 = ['application/zip', 'application/x-zip-compressed', 'multipart/x-zip'];\nconst CACHED_APPS_MAX_AGE = 1000 * 60 * 60 * 24; // ms\nconst MAX_CACHED_APPS = 1024;\nconst APPLICATIONS_CACHE = new LRU({\n  max: MAX_CACHED_APPS,\n  ttl: CACHED_APPS_MAX_AGE, // expire after 24 hours\n  updateAgeOnGet: true,\n  dispose: (app, {fullPath}) => {\n    logger.info(\n      `The application '${app}' cached at '${fullPath}' has ` +\n        `expired after ${CACHED_APPS_MAX_AGE}ms`\n    );\n    if (fullPath) {\n      fs.rimraf(fullPath);\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.size === 0) {\n    return;\n  }\n\n  const appPaths = [...APPLICATIONS_CACHE.values()].map(({fullPath}) => fullPath);\n  logger.debug(\n    `Performing cleanup of ${appPaths.length} cached ` +\n      util.pluralize('application', appPaths.length)\n  );\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\nasync function retrieveHeaders(link) {\n  try {\n    return (\n      await axios({\n        url: link,\n        method: 'HEAD',\n        timeout: 5000,\n      })\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(\n        `The cached application '${path.basename(fullPath)}' will expire in ${msLeft / 1000}s`\n      );\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(\n    `New app path '${app}' did not have ` +\n      `${util.pluralize('extension', supportedAppExtensions.length, false)}: ` +\n      supportedAppExtensions\n  );\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 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 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 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 * If the downloaded app has `.zip` extension, this method will unzip it.\n * The unzip does not work when `onPostProcess` is provided.\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 = /** @type {ConfigureAppOptions} */ ({})) {\n  if (!_.isString(app)) {\n    // immediately shortcircuit if not given an app\n    return;\n  }\n\n  let supportedAppExtensions;\n  const onPostProcess =\n    !_.isString(options) && !_.isArray(options) ? options.onPostProcess : undefined;\n\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  /** @type {RemoteAppProps} */\n  const remoteAppProps = {\n    lastModified: null,\n    immutable: false,\n    maxAge: null,\n  };\n  const {protocol, pathname} = url.parse(newApp);\n  const isUrl = protocol === null ? false : ['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(\n          `The application at '${cachedPath}' does not exist anymore ` +\n            `or its integrity has been damaged. Deleting it from the internal cache`\n        );\n        APPLICATIONS_CACHE.delete(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 (\n          ZIP_MIME_TYPES.some((mimeType) =>\n            new RegExp(`\\\\b${_.escapeRegExp(mimeType)}\\\\b`).test(ct)\n          )\n        ) {\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(\n            `The current file extension '${resultingExt}' is not supported. ` +\n              `Defaulting to '${_.first(supportedAppExtensions)}'`\n          );\n          resultingExt = /** @type {string} */ (_.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 =\n          `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(\n          `The application at '${fullPath}' does not exist anymore ` +\n            `or its integrity has been damaged. Deleting it from the cache`\n        );\n        APPLICATIONS_CACHE.delete(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(\n        `The current application path '${app}' is not absolute ` +\n          `and has been rewritten to '${newApp}'. Consider using absolute paths rather than relative`\n      );\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 {Promise<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 =\n      _.isEmpty(useSystemUnzipEnv) || !['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(\n        `Enforcing UTF-8 encoding on the extracted file names for '${path.basename(zipPath)}'`\n      );\n      extractionOpts.fileNamesEncoding = 'utf8';\n    }\n    await zip.extractAllTo(zipPath, tmpRoot, extractionOpts);\n    const globPattern = `**/*.+(${supportedAppExtensions\n      .map((ext) => ext.replace(/^\\./, ''))\n      .join('|')})`;\n    const sortedBundleItems = (\n      await fs.glob(globPattern, {\n        cwd: tmpRoot,\n        strict: false,\n        // Get the top level match\n      })\n    ).sort((a, b) => a.split(path.sep).length - b.split(path.sep).length);\n    if (_.isEmpty(sortedBundleItems)) {\n      logger.errorAndThrow(\n        `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(\n            'extension',\n            supportedAppExtensions.length,\n            false\n          )}`\n      );\n    }\n    logger.debug(\n      `Extracted ${util.pluralize('bundle item', sortedBundleItems.length, true)} ` +\n        `from '${zipPath}' in ${Math.round(\n          timer.getDuration().asMilliSeconds\n        )}ms: ${sortedBundleItems}`\n    );\n    const matchedBundle = /** @type {string} */ (_.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\n/**\n * Generate a string that uniquely describes driver instance\n *\n * @param {import('@appium/types').Core} obj driver instance\n * @param {string?} sessionId session identifier (if exists)\n * @returns {string}\n */\nfunction generateDriverLogPrefix(obj, sessionId = null) {\n  const instanceName = `${obj.constructor.name}@${node.getObjectId(obj).substring(0, 4)}`;\n  return sessionId ? `${instanceName} (${sessionId.substring(0, 8)})` : instanceName;\n}\n\n/** @type {import('@appium/types').DriverHelpers} */\nexport default {\n  configureApp,\n  isPackageOrBundle,\n  duplicateKeys,\n  parseCapsArray,\n  generateDriverLogPrefix,\n};\nexport {configureApp, isPackageOrBundle, duplicateKeys, parseCapsArray, generateDriverLogPrefix};\n\n/**\n * @typedef RemoteAppProps\n * @property {Date?} lastModified\n * @property {boolean} immutable\n * @property {number?} maxAge\n */\n"],"mappings":"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"}
545
+ /**
546
+ * @typedef RemoteAppProps
547
+ * @property {Date?} lastModified
548
+ * @property {boolean} immutable
549
+ * @property {number?} maxAge
550
+ */
551
+ //# sourceMappingURL=helpers.js.map