appium-uiautomator2-driver 2.0.3 → 2.0.6
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.
- package/README.md +7 -0
- package/build/lib/commands/element.js +2 -6
- package/build/lib/commands/general.js +8 -12
- package/build/lib/commands/screenshot.js +3 -7
- package/build/lib/commands/touch.js +4 -11
- package/build/lib/driver.js +52 -73
- package/build/lib/uiautomator2.js +29 -39
- package/lib/commands/element.js +1 -3
- package/lib/commands/general.js +7 -8
- package/lib/commands/screenshot.js +3 -5
- package/lib/commands/touch.js +3 -5
- package/lib/driver.js +53 -52
- package/lib/uiautomator2.js +28 -27
- package/npm-shrinkwrap.json +496 -280
- package/package.json +5 -5
|
@@ -15,8 +15,6 @@ var _baseDriver = require("@appium/base-driver");
|
|
|
15
15
|
|
|
16
16
|
var _asyncbox = require("asyncbox");
|
|
17
17
|
|
|
18
|
-
var _logger = _interopRequireDefault(require("./logger"));
|
|
19
|
-
|
|
20
18
|
var _appiumUiautomator2Server = require("appium-uiautomator2-server");
|
|
21
19
|
|
|
22
20
|
var _support = require("@appium/support");
|
|
@@ -54,7 +52,7 @@ class UIA2Proxy extends _baseDriver.JWProxy {
|
|
|
54
52
|
}
|
|
55
53
|
|
|
56
54
|
class UiAutomator2Server {
|
|
57
|
-
constructor(opts = {}) {
|
|
55
|
+
constructor(log, opts = {}) {
|
|
58
56
|
for (let req of REQD_PARAMS) {
|
|
59
57
|
if (!opts || !_support.util.hasValue(opts[req])) {
|
|
60
58
|
throw new Error(`Option '${req}' is required!`);
|
|
@@ -63,8 +61,10 @@ class UiAutomator2Server {
|
|
|
63
61
|
this[req] = opts[req];
|
|
64
62
|
}
|
|
65
63
|
|
|
64
|
+
this.log = log;
|
|
66
65
|
this.disableSuppressAccessibilityService = opts.disableSuppressAccessibilityService;
|
|
67
66
|
const proxyOpts = {
|
|
67
|
+
log,
|
|
68
68
|
server: this.host,
|
|
69
69
|
port: this.systemPort,
|
|
70
70
|
keepAlive: true
|
|
@@ -94,7 +94,7 @@ class UiAutomator2Server {
|
|
|
94
94
|
};
|
|
95
95
|
}
|
|
96
96
|
|
|
97
|
-
|
|
97
|
+
this.log.info(`Server package at '${appPath}' is not writeable. ` + `Will copy it into the temporary location at '${tmpRoot}' as a workaround. ` + `Consider making this file writeable manually in order to improve the performance of session startup.`);
|
|
98
98
|
|
|
99
99
|
const dstPath = _path.default.resolve(tmpRoot, _path.default.basename(appPath));
|
|
100
100
|
|
|
@@ -137,8 +137,7 @@ class UiAutomator2Server {
|
|
|
137
137
|
}
|
|
138
138
|
|
|
139
139
|
const appState = await this.adb.getApplicationInstallState(appPath, appId);
|
|
140
|
-
|
|
141
|
-
_logger.default.debug(`${appId} installation state: ${appState}`);
|
|
140
|
+
this.log.debug(`${appId} installation state: ${appState}`);
|
|
142
141
|
|
|
143
142
|
if (await this.adb.checkApkCert(appPath, appId)) {
|
|
144
143
|
shouldUninstallServerPackages = shouldUninstallServerPackages || [this.adb.APP_INSTALL_STATE.OLDER_VERSION_INSTALLED, this.adb.APP_INSTALL_STATE.NEWER_VERSION_INSTALLED].includes(appState);
|
|
@@ -150,10 +149,10 @@ class UiAutomator2Server {
|
|
|
150
149
|
shouldInstallServerPackages = shouldInstallServerPackages || shouldUninstallServerPackages || [this.adb.APP_INSTALL_STATE.NOT_INSTALLED].includes(appState);
|
|
151
150
|
}
|
|
152
151
|
|
|
153
|
-
|
|
152
|
+
this.log.info(`Server packages are ${shouldInstallServerPackages ? '' : 'not '}going to be (re)installed`);
|
|
154
153
|
|
|
155
154
|
if (shouldInstallServerPackages && shouldUninstallServerPackages) {
|
|
156
|
-
|
|
155
|
+
this.log.info('Full packages reinstall is going to be performed');
|
|
157
156
|
}
|
|
158
157
|
|
|
159
158
|
for (const {
|
|
@@ -164,7 +163,7 @@ class UiAutomator2Server {
|
|
|
164
163
|
try {
|
|
165
164
|
await this.adb.uninstallApk(appId);
|
|
166
165
|
} catch (err) {
|
|
167
|
-
|
|
166
|
+
this.log.warn(`Error uninstalling '${appId}': ${err.message}`);
|
|
168
167
|
}
|
|
169
168
|
}
|
|
170
169
|
|
|
@@ -185,8 +184,7 @@ class UiAutomator2Server {
|
|
|
185
184
|
}
|
|
186
185
|
|
|
187
186
|
async verifyServicesAvailability() {
|
|
188
|
-
|
|
189
|
-
|
|
187
|
+
this.log.debug(`Waiting up to ${SERVICES_LAUNCH_TIMEOUT}ms for services to be available`);
|
|
190
188
|
let isPmServiceAvailable = false;
|
|
191
189
|
let pmOutput = '';
|
|
192
190
|
let pmError = null;
|
|
@@ -208,9 +206,7 @@ class UiAutomator2Server {
|
|
|
208
206
|
pmOutput = '';
|
|
209
207
|
} else if (pmOutput.includes(INSTRUMENTATION_TARGET)) {
|
|
210
208
|
pmOutput = '';
|
|
211
|
-
|
|
212
|
-
_logger.default.debug(`Instrumentation target '${INSTRUMENTATION_TARGET}' is available`);
|
|
213
|
-
|
|
209
|
+
this.log.debug(`Instrumentation target '${INSTRUMENTATION_TARGET}' is available`);
|
|
214
210
|
isPmServiceAvailable = true;
|
|
215
211
|
} else if (!pmError) {
|
|
216
212
|
pmError = new Error('The instrumentation target is not listed by Package Manager');
|
|
@@ -223,13 +219,13 @@ class UiAutomator2Server {
|
|
|
223
219
|
intervalMs: 1000
|
|
224
220
|
});
|
|
225
221
|
} catch (err) {
|
|
226
|
-
|
|
222
|
+
this.log.error(`Unable to find instrumentation target '${INSTRUMENTATION_TARGET}': ${(pmError || {}).message}`);
|
|
227
223
|
|
|
228
224
|
if (pmOutput) {
|
|
229
|
-
|
|
225
|
+
this.log.debug('Available targets:');
|
|
230
226
|
|
|
231
227
|
for (const line of pmOutput.split('\n')) {
|
|
232
|
-
|
|
228
|
+
this.log.debug(` ${line.replace('instrumentation:', '')}`);
|
|
233
229
|
}
|
|
234
230
|
}
|
|
235
231
|
}
|
|
@@ -239,11 +235,10 @@ class UiAutomator2Server {
|
|
|
239
235
|
await this.cleanupAutomationLeftovers();
|
|
240
236
|
|
|
241
237
|
if (caps.skipServerInstallation) {
|
|
242
|
-
|
|
238
|
+
this.log.info(`'skipServerInstallation' is set. Attempting to use UIAutomator2 server from the device`);
|
|
243
239
|
} else {
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
_logger.default.info(`Using UIAutomator2 server from '${_appiumUiautomator2Server.SERVER_APK_PATH}' and test from '${_appiumUiautomator2Server.TEST_APK_PATH}'`);
|
|
240
|
+
this.log.info(`Starting UIAutomator2 server ${_appiumUiautomator2Server.version}`);
|
|
241
|
+
this.log.info(`Using UIAutomator2 server from '${_appiumUiautomator2Server.SERVER_APK_PATH}' and test from '${_appiumUiautomator2Server.TEST_APK_PATH}'`);
|
|
247
242
|
}
|
|
248
243
|
|
|
249
244
|
const timeout = caps.uiautomator2ServerLaunchTimeout || SERVER_LAUNCH_TIMEOUT;
|
|
@@ -253,8 +248,7 @@ class UiAutomator2Server {
|
|
|
253
248
|
const delayBetweenRetries = 3000;
|
|
254
249
|
|
|
255
250
|
while (retries < maxRetries) {
|
|
256
|
-
|
|
257
|
-
|
|
251
|
+
this.log.info(`Waiting up to ${timeout}ms for UiAutomator2 to be online...`);
|
|
258
252
|
this.jwproxy.didInstrumentationExit = false;
|
|
259
253
|
await this.startInstrumentationProcess();
|
|
260
254
|
|
|
@@ -272,7 +266,7 @@ class UiAutomator2Server {
|
|
|
272
266
|
intervalMs: 1000
|
|
273
267
|
});
|
|
274
268
|
} catch (err) {
|
|
275
|
-
|
|
269
|
+
this.log.errorAndThrow(`The instrumentation process cannot be initialized within ${timeout}ms timeout. ` + 'Make sure the application under test does not crash and investigate the logcat output. ' + `You could also try to increase the value of 'uiautomator2ServerLaunchTimeout' capability`);
|
|
276
270
|
}
|
|
277
271
|
}
|
|
278
272
|
|
|
@@ -283,17 +277,15 @@ class UiAutomator2Server {
|
|
|
283
277
|
retries++;
|
|
284
278
|
|
|
285
279
|
if (retries >= maxRetries) {
|
|
286
|
-
|
|
280
|
+
this.log.errorAndThrow('The instrumentation process cannot be initialized. ' + 'Make sure the application under test does not crash and investigate the logcat output.');
|
|
287
281
|
}
|
|
288
282
|
|
|
289
|
-
|
|
290
|
-
|
|
283
|
+
this.log.warn(`The instrumentation process has been unexpectedly terminated. ` + `Retrying UiAutomator2 startup (#${retries} of ${maxRetries - 1})`);
|
|
291
284
|
await this.cleanupAutomationLeftovers(true);
|
|
292
285
|
await _bluebird.default.delay(delayBetweenRetries);
|
|
293
286
|
}
|
|
294
287
|
|
|
295
|
-
|
|
296
|
-
|
|
288
|
+
this.log.debug(`The initialization of the instrumentation process took ` + `${timer.getDuration().asMilliSeconds.toFixed(0)}ms`);
|
|
297
289
|
await this.jwproxy.command('/session', 'POST', {
|
|
298
290
|
capabilities: {
|
|
299
291
|
firstMatch: [caps],
|
|
@@ -331,17 +323,17 @@ class UiAutomator2Server {
|
|
|
331
323
|
}
|
|
332
324
|
|
|
333
325
|
async deleteSession() {
|
|
334
|
-
|
|
326
|
+
this.log.debug('Deleting UiAutomator2 server session');
|
|
335
327
|
|
|
336
328
|
try {
|
|
337
329
|
await this.jwproxy.command('/', 'DELETE');
|
|
338
330
|
} catch (err) {
|
|
339
|
-
|
|
331
|
+
this.log.warn(`Did not get confirmation UiAutomator2 deleteSession worked; ` + `Error was: ${err}`);
|
|
340
332
|
}
|
|
341
333
|
}
|
|
342
334
|
|
|
343
335
|
async cleanupAutomationLeftovers(strictCleanup = false) {
|
|
344
|
-
|
|
336
|
+
this.log.debug(`Performing ${strictCleanup ? 'strict' : 'shallow'} cleanup of automation leftovers`);
|
|
345
337
|
|
|
346
338
|
try {
|
|
347
339
|
const {
|
|
@@ -355,17 +347,15 @@ class UiAutomator2Server {
|
|
|
355
347
|
}) => id).filter(Boolean);
|
|
356
348
|
|
|
357
349
|
if (activeSessionIds.length) {
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
_logger.default.debug(`Cleaning up ${_support.util.pluralize('obsolete session', activeSessionIds.length, true)}`);
|
|
361
|
-
|
|
350
|
+
this.log.debug(`The following obsolete sessions are still running: ${JSON.stringify(activeSessionIds)}`);
|
|
351
|
+
this.log.debug(`Cleaning up ${_support.util.pluralize('obsolete session', activeSessionIds.length, true)}`);
|
|
362
352
|
await _bluebird.default.all(activeSessionIds.map(id => _axios.default.delete(`http://${this.host}:${this.systemPort}/session/${id}`)));
|
|
363
353
|
await _bluebird.default.delay(1000);
|
|
364
354
|
} else {
|
|
365
|
-
|
|
355
|
+
this.log.debug('No obsolete sessions have been detected');
|
|
366
356
|
}
|
|
367
357
|
} catch (e) {
|
|
368
|
-
|
|
358
|
+
this.log.debug(`No obsolete sessions have been detected (${e.message})`);
|
|
369
359
|
}
|
|
370
360
|
|
|
371
361
|
try {
|
|
@@ -388,4 +378,4 @@ var _default = UiAutomator2Server;
|
|
|
388
378
|
exports.default = _default;require('source-map-support').install();
|
|
389
379
|
|
|
390
380
|
|
|
391
|
-
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/uiautomator2.js"],"names":["REQD_PARAMS","SERVER_LAUNCH_TIMEOUT","SERVER_INSTALL_RETRIES","SERVICES_LAUNCH_TIMEOUT","SERVER_PACKAGE_ID","SERVER_TEST_PACKAGE_ID","INSTRUMENTATION_TARGET","instrumentationLogger","logger","getLogger","UIA2Proxy","JWProxy","proxyCommand","url","method","body","didInstrumentationExit","errors","InvalidContextError","UiAutomator2Server","constructor","opts","req","util","hasValue","Error","disableSuppressAccessibilityService","proxyOpts","server","host","port","systemPort","keepAlive","readTimeout","timeout","jwproxy","proxyReqRes","bind","command","installServerApk","installTimeout","tmpRoot","tempDir","openDir","packageInfosMapper","appPath","appId","helpers","isWriteable","log","info","dstPath","path","resolve","basename","fs","copyFile","packagesInfo","B","all","map","apkPath","testApkPath","shouldUninstallServerPackages","shouldInstallServerPackages","isAppInstalled","adb","checkApkCert","signApp","appState","getApplicationInstallState","debug","APP_INSTALL_STATE","OLDER_VERSION_INSTALLED","NEWER_VERSION_INSTALLED","includes","NOT_INSTALLED","uninstallApk","err","warn","message","install","noIncremental","replace","timeoutCapName","rimraf","verifyServicesAvailability","isPmServiceAvailable","pmOutput","pmError","shell","e","waitMs","intervalMs","error","line","split","startSession","caps","cleanupAutomationLeftovers","skipServerInstallation","serverVersion","uiautomator2ServerLaunchTimeout","timer","timing","Timer","start","retries","maxRetries","delayBetweenRetries","startInstrumentationProcess","errorAndThrow","delay","getDuration","asMilliSeconds","toFixed","capabilities","firstMatch","alwaysMatch","cmd","disableWindowAnimation","push","_","isBoolean","instrumentationProcess","createSubProcess","on","stdout","stderr","output","trim","code","deleteSession","strictCleanup","value","data","activeSessionIds","id","filter","Boolean","length","JSON","stringify","pluralize","axios","delete","forceStop","ignore","killProcessesByName"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport { JWProxy, errors } from '@appium/base-driver';\nimport { waitForCondition } from 'asyncbox';\nimport log from './logger';\nimport {\n  SERVER_APK_PATH as apkPath,\n  TEST_APK_PATH as testApkPath,\n  version as serverVersion\n} from 'appium-uiautomator2-server';\nimport {\n  util, logger, tempDir, fs, timing\n} from '@appium/support';\nimport B from 'bluebird';\nimport helpers from './helpers';\nimport axios from 'axios';\nimport path from 'path';\n\nconst REQD_PARAMS = ['adb', 'tmpDir', 'host', 'systemPort', 'devicePort', 'disableWindowAnimation'];\nconst SERVER_LAUNCH_TIMEOUT = 30000;\nconst SERVER_INSTALL_RETRIES = 20;\nconst SERVICES_LAUNCH_TIMEOUT = 30000;\nconst SERVER_PACKAGE_ID = 'io.appium.uiautomator2.server';\nconst SERVER_TEST_PACKAGE_ID = `${SERVER_PACKAGE_ID}.test`;\nconst INSTRUMENTATION_TARGET = `${SERVER_TEST_PACKAGE_ID}/androidx.test.runner.AndroidJUnitRunner`;\nconst instrumentationLogger = logger.getLogger('Instrumentation');\n\nclass UIA2Proxy extends JWProxy {\n  async proxyCommand (url, method, body = null) {\n    if (this.didInstrumentationExit) {\n      throw new errors.InvalidContextError(\n        `'${method} ${url}' cannot be proxied to UiAutomator2 server because ` +\n        'the instrumentation process is not running (probably crashed). ' +\n        'Check the server log and/or the logcat output for more details');\n    }\n    return await super.proxyCommand(url, method, body);\n  }\n}\n\nclass UiAutomator2Server {\n  constructor (opts = {}) {\n    for (let req of REQD_PARAMS) {\n      if (!opts || !util.hasValue(opts[req])) {\n        throw new Error(`Option '${req}' is required!`);\n      }\n      this[req] = opts[req];\n    }\n    this.disableSuppressAccessibilityService = opts.disableSuppressAccessibilityService;\n    const proxyOpts = {\n      server: this.host,\n      port: this.systemPort,\n      keepAlive: true,\n    };\n    if (opts.readTimeout && opts.readTimeout > 0) {\n      proxyOpts.timeout = opts.readTimeout;\n    }\n    this.jwproxy = new UIA2Proxy(proxyOpts);\n    this.proxyReqRes = this.jwproxy.proxyReqRes.bind(this.jwproxy);\n    this.proxyCommand = this.jwproxy.command.bind(this.jwproxy);\n    this.jwproxy.didInstrumentationExit = false;\n  }\n\n  /**\n   * Installs the apks on to the device or emulator.\n   *\n   * @param {number} installTimeout - Installation timeout\n   */\n  async installServerApk (installTimeout = SERVER_INSTALL_RETRIES * 1000) {\n    const tmpRoot = await tempDir.openDir();\n    const packageInfosMapper = async ({appPath, appId}) => {\n      if (await helpers.isWriteable(appPath)) {\n        return { appPath, appId };\n      }\n\n      log.info(`Server package at '${appPath}' is not writeable. ` +\n        `Will copy it into the temporary location at '${tmpRoot}' as a workaround. ` +\n        `Consider making this file writeable manually in order to improve the performance of session startup.`);\n      const dstPath = path.resolve(tmpRoot, path.basename(appPath));\n      await fs.copyFile(appPath, dstPath);\n      return {\n        appPath: dstPath,\n        appId,\n      };\n    };\n\n    try {\n      const packagesInfo = await B.all(B.map([\n        {\n          appPath: apkPath,\n          appId: SERVER_PACKAGE_ID,\n        }, {\n          appPath: testApkPath,\n          appId: SERVER_TEST_PACKAGE_ID,\n        },\n      ], packageInfosMapper));\n\n      let shouldUninstallServerPackages = false;\n      let shouldInstallServerPackages = false;\n      for (const {appId, appPath} of packagesInfo) {\n        if (appId === SERVER_TEST_PACKAGE_ID) {\n          const isAppInstalled = await this.adb.isAppInstalled(appId);\n\n          // There is no point in getting the state for test server,\n          // since it does not contain version info\n          if (!await this.adb.checkApkCert(appPath, appId)) {\n            await helpers.signApp(this.adb, appPath);\n            shouldUninstallServerPackages = shouldUninstallServerPackages || isAppInstalled;\n            shouldInstallServerPackages = true;\n          }\n\n          if (!isAppInstalled) {\n            shouldInstallServerPackages = true;\n          }\n          continue;\n        }\n\n        const appState = await this.adb.getApplicationInstallState(appPath, appId);\n        log.debug(`${appId} installation state: ${appState}`);\n        if (await this.adb.checkApkCert(appPath, appId)) {\n          shouldUninstallServerPackages = shouldUninstallServerPackages || [\n            this.adb.APP_INSTALL_STATE.OLDER_VERSION_INSTALLED,\n            this.adb.APP_INSTALL_STATE.NEWER_VERSION_INSTALLED,\n          ].includes(appState);\n        } else {\n          await helpers.signApp(this.adb, appPath);\n          shouldUninstallServerPackages = shouldUninstallServerPackages || ![\n            this.adb.APP_INSTALL_STATE.NOT_INSTALLED,\n          ].includes(appState);\n        }\n        shouldInstallServerPackages = shouldInstallServerPackages || shouldUninstallServerPackages || [\n          this.adb.APP_INSTALL_STATE.NOT_INSTALLED,\n        ].includes(appState);\n      }\n      log.info(`Server packages are ${shouldInstallServerPackages ? '' : 'not '}going to be (re)installed`);\n      if (shouldInstallServerPackages && shouldUninstallServerPackages) {\n        log.info('Full packages reinstall is going to be performed');\n      }\n      for (const {appId, appPath} of packagesInfo) {\n        if (shouldUninstallServerPackages) {\n          try {\n            await this.adb.uninstallApk(appId);\n          } catch (err) {\n            log.warn(`Error uninstalling '${appId}': ${err.message}`);\n          }\n        }\n        if (shouldInstallServerPackages) {\n          await this.adb.install(appPath, {\n            noIncremental: true,\n            replace: true,\n            timeout: installTimeout,\n            timeoutCapName: 'uiautomator2ServerInstallTimeout'\n          });\n        }\n      }\n    } finally {\n      await fs.rimraf(tmpRoot);\n    }\n\n    await this.verifyServicesAvailability();\n  }\n\n  async verifyServicesAvailability () {\n    log.debug(`Waiting up to ${SERVICES_LAUNCH_TIMEOUT}ms for services to be available`);\n    let isPmServiceAvailable = false;\n    let pmOutput = '';\n    let pmError = null;\n    try {\n      await waitForCondition(async () => {\n        if (!isPmServiceAvailable) {\n          pmError = null;\n          pmOutput = '';\n          try {\n            pmOutput = await this.adb.shell(['pm', 'list', 'instrumentation']);\n          } catch (e) {\n            pmError = e;\n          }\n          if (pmOutput.includes('Could not access the Package Manager')) {\n            pmError = new Error(`Problem running Package Manager: ${pmOutput}`);\n            pmOutput = ''; // remove output, so it is not printed below\n          } else if (pmOutput.includes(INSTRUMENTATION_TARGET)) {\n            pmOutput = ''; // remove output, so it is not printed below\n            log.debug(`Instrumentation target '${INSTRUMENTATION_TARGET}' is available`);\n            // eslint-disable-next-line require-atomic-updates\n            isPmServiceAvailable = true;\n          } else if (!pmError) {\n            pmError = new Error('The instrumentation target is not listed by Package Manager');\n          }\n        }\n        return isPmServiceAvailable;\n      }, {\n        waitMs: SERVICES_LAUNCH_TIMEOUT,\n        intervalMs: 1000,\n      });\n    } catch (err) {\n      log.error(`Unable to find instrumentation target '${INSTRUMENTATION_TARGET}': ${(pmError || {}).message}`);\n      if (pmOutput) {\n        log.debug('Available targets:');\n        for (const line of pmOutput.split('\\n')) {\n          log.debug(`    ${line.replace('instrumentation:', '')}`);\n        }\n      }\n    }\n  }\n\n  async startSession (caps) {\n    await this.cleanupAutomationLeftovers();\n    if (caps.skipServerInstallation) {\n      log.info(`'skipServerInstallation' is set. Attempting to use UIAutomator2 server from the device`);\n    } else {\n      log.info(`Starting UIAutomator2 server ${serverVersion}`);\n      log.info(`Using UIAutomator2 server from '${apkPath}' and test from '${testApkPath}'`);\n    }\n\n    const timeout = caps.uiautomator2ServerLaunchTimeout || SERVER_LAUNCH_TIMEOUT;\n    const timer = new timing.Timer().start();\n    let retries = 0;\n    const maxRetries = 2;\n    const delayBetweenRetries = 3000;\n    while (retries < maxRetries) {\n      log.info(`Waiting up to ${timeout}ms for UiAutomator2 to be online...`);\n      this.jwproxy.didInstrumentationExit = false;\n      await this.startInstrumentationProcess();\n      if (!this.jwproxy.didInstrumentationExit) {\n        try {\n          await waitForCondition(async () => {\n            try {\n              await this.jwproxy.command('/status', 'GET');\n              return true;\n            } catch (err) {\n              // short circuit to retry or fail fast\n              return this.jwproxy.didInstrumentationExit;\n            }\n          }, {\n            waitMs: timeout,\n            intervalMs: 1000,\n          });\n        } catch (err) {\n          log.errorAndThrow(`The instrumentation process cannot be initialized within ${timeout}ms timeout. `\n            + 'Make sure the application under test does not crash and investigate the logcat output. '\n            + `You could also try to increase the value of 'uiautomator2ServerLaunchTimeout' capability`);\n        }\n      }\n      if (!this.jwproxy.didInstrumentationExit) {\n        break;\n      }\n\n      retries++;\n      if (retries >= maxRetries) {\n        log.errorAndThrow('The instrumentation process cannot be initialized. '\n          + 'Make sure the application under test does not crash and investigate the logcat output.');\n      }\n      log.warn(`The instrumentation process has been unexpectedly terminated. `\n        + `Retrying UiAutomator2 startup (#${retries} of ${maxRetries - 1})`);\n      await this.cleanupAutomationLeftovers(true);\n      await B.delay(delayBetweenRetries);\n    }\n\n    log.debug(`The initialization of the instrumentation process took `\n      + `${timer.getDuration().asMilliSeconds.toFixed(0)}ms`);\n    await this.jwproxy.command('/session', 'POST', {\n      capabilities: {\n        firstMatch: [caps],\n        alwaysMatch: {},\n      }\n    });\n  }\n\n  async startInstrumentationProcess () {\n    const cmd = ['am', 'instrument', '-w'];\n    if (this.disableWindowAnimation) {\n      cmd.push('--no-window-animation');\n    }\n    if (_.isBoolean(this.disableSuppressAccessibilityService)) {\n      cmd.push('-e', 'DISABLE_SUPPRESS_ACCESSIBILITY_SERVICES', this.disableSuppressAccessibilityService);\n    }\n    // Disable Google analytics to prevent possible fatal exception\n    cmd.push('-e', 'disableAnalytics', true);\n    cmd.push(INSTRUMENTATION_TARGET);\n    const instrumentationProcess = this.adb.createSubProcess(['shell', ...cmd]);\n    instrumentationProcess.on('output', (stdout, stderr) => {\n      const output = _.trim(stdout || stderr);\n      if (output) {\n        instrumentationLogger.debug(output);\n      }\n    });\n    instrumentationProcess.on('exit', (code) => {\n      instrumentationLogger.debug(`The process has exited with code ${code}`);\n      this.jwproxy.didInstrumentationExit = true;\n    });\n    await instrumentationProcess.start(0);\n  }\n\n  async deleteSession () {\n    log.debug('Deleting UiAutomator2 server session');\n    // rely on jwproxy's intelligence to know what we're talking about and\n    // delete the current session\n    try {\n      await this.jwproxy.command('/', 'DELETE');\n    } catch (err) {\n      log.warn(`Did not get confirmation UiAutomator2 deleteSession worked; ` +\n          `Error was: ${err}`);\n    }\n  }\n\n  async cleanupAutomationLeftovers (strictCleanup = false) {\n    log.debug(`Performing ${strictCleanup ? 'strict' : 'shallow'} cleanup of automation leftovers`);\n\n    try {\n      const {value} = (await axios({\n        url: `http://${this.host}:${this.systemPort}/sessions`,\n        timeout: 500,\n      })).data;\n      const activeSessionIds = value.map(({id}) => id).filter(Boolean);\n      if (activeSessionIds.length) {\n        log.debug(`The following obsolete sessions are still running: ${JSON.stringify(activeSessionIds)}`);\n        log.debug(`Cleaning up ${util.pluralize('obsolete session', activeSessionIds.length, true)}`);\n        await B.all(activeSessionIds\n          .map((id) => axios.delete(`http://${this.host}:${this.systemPort}/session/${id}`))\n        );\n        // Let all sessions to be properly terminated before continuing\n        await B.delay(1000);\n      } else {\n        log.debug('No obsolete sessions have been detected');\n      }\n    } catch (e) {\n      log.debug(`No obsolete sessions have been detected (${e.message})`);\n    }\n\n    try {\n      await this.adb.forceStop(SERVER_TEST_PACKAGE_ID);\n    } catch (ignore) {}\n    if (!strictCleanup) {\n      return;\n    }\n    // https://github.com/appium/appium/issues/10749\n    try {\n      await this.adb.killProcessesByName('uiautomator');\n    } catch (ignore) {}\n  }\n}\n\nexport { UiAutomator2Server, INSTRUMENTATION_TARGET, SERVER_PACKAGE_ID, SERVER_TEST_PACKAGE_ID };\nexport default UiAutomator2Server;\n"],"file":"lib/uiautomator2.js","sourceRoot":"../.."}
|
|
381
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/uiautomator2.js"],"names":["REQD_PARAMS","SERVER_LAUNCH_TIMEOUT","SERVER_INSTALL_RETRIES","SERVICES_LAUNCH_TIMEOUT","SERVER_PACKAGE_ID","SERVER_TEST_PACKAGE_ID","INSTRUMENTATION_TARGET","instrumentationLogger","logger","getLogger","UIA2Proxy","JWProxy","proxyCommand","url","method","body","didInstrumentationExit","errors","InvalidContextError","UiAutomator2Server","constructor","log","opts","req","util","hasValue","Error","disableSuppressAccessibilityService","proxyOpts","server","host","port","systemPort","keepAlive","readTimeout","timeout","jwproxy","proxyReqRes","bind","command","installServerApk","installTimeout","tmpRoot","tempDir","openDir","packageInfosMapper","appPath","appId","helpers","isWriteable","info","dstPath","path","resolve","basename","fs","copyFile","packagesInfo","B","all","map","apkPath","testApkPath","shouldUninstallServerPackages","shouldInstallServerPackages","isAppInstalled","adb","checkApkCert","signApp","appState","getApplicationInstallState","debug","APP_INSTALL_STATE","OLDER_VERSION_INSTALLED","NEWER_VERSION_INSTALLED","includes","NOT_INSTALLED","uninstallApk","err","warn","message","install","noIncremental","replace","timeoutCapName","rimraf","verifyServicesAvailability","isPmServiceAvailable","pmOutput","pmError","shell","e","waitMs","intervalMs","error","line","split","startSession","caps","cleanupAutomationLeftovers","skipServerInstallation","serverVersion","uiautomator2ServerLaunchTimeout","timer","timing","Timer","start","retries","maxRetries","delayBetweenRetries","startInstrumentationProcess","errorAndThrow","delay","getDuration","asMilliSeconds","toFixed","capabilities","firstMatch","alwaysMatch","cmd","disableWindowAnimation","push","_","isBoolean","instrumentationProcess","createSubProcess","on","stdout","stderr","output","trim","code","deleteSession","strictCleanup","value","data","activeSessionIds","id","filter","Boolean","length","JSON","stringify","pluralize","axios","delete","forceStop","ignore","killProcessesByName"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport { JWProxy, errors } from '@appium/base-driver';\nimport { waitForCondition } from 'asyncbox';\nimport {\n  SERVER_APK_PATH as apkPath,\n  TEST_APK_PATH as testApkPath,\n  version as serverVersion\n} from 'appium-uiautomator2-server';\nimport {\n  util, logger, tempDir, fs, timing\n} from '@appium/support';\nimport B from 'bluebird';\nimport helpers from './helpers';\nimport axios from 'axios';\nimport path from 'path';\n\nconst REQD_PARAMS = ['adb', 'tmpDir', 'host', 'systemPort', 'devicePort', 'disableWindowAnimation'];\nconst SERVER_LAUNCH_TIMEOUT = 30000;\nconst SERVER_INSTALL_RETRIES = 20;\nconst SERVICES_LAUNCH_TIMEOUT = 30000;\nconst SERVER_PACKAGE_ID = 'io.appium.uiautomator2.server';\nconst SERVER_TEST_PACKAGE_ID = `${SERVER_PACKAGE_ID}.test`;\nconst INSTRUMENTATION_TARGET = `${SERVER_TEST_PACKAGE_ID}/androidx.test.runner.AndroidJUnitRunner`;\nconst instrumentationLogger = logger.getLogger('Instrumentation');\n\nclass UIA2Proxy extends JWProxy {\n  async proxyCommand (url, method, body = null) {\n    if (this.didInstrumentationExit) {\n      throw new errors.InvalidContextError(\n        `'${method} ${url}' cannot be proxied to UiAutomator2 server because ` +\n        'the instrumentation process is not running (probably crashed). ' +\n        'Check the server log and/or the logcat output for more details');\n    }\n    return await super.proxyCommand(url, method, body);\n  }\n}\n\nclass UiAutomator2Server {\n  constructor (log, opts = {}) {\n    for (let req of REQD_PARAMS) {\n      if (!opts || !util.hasValue(opts[req])) {\n        throw new Error(`Option '${req}' is required!`);\n      }\n      this[req] = opts[req];\n    }\n    this.log = log;\n    this.disableSuppressAccessibilityService = opts.disableSuppressAccessibilityService;\n    const proxyOpts = {\n      log,\n      server: this.host,\n      port: this.systemPort,\n      keepAlive: true,\n    };\n    if (opts.readTimeout && opts.readTimeout > 0) {\n      proxyOpts.timeout = opts.readTimeout;\n    }\n    this.jwproxy = new UIA2Proxy(proxyOpts);\n    this.proxyReqRes = this.jwproxy.proxyReqRes.bind(this.jwproxy);\n    this.proxyCommand = this.jwproxy.command.bind(this.jwproxy);\n    this.jwproxy.didInstrumentationExit = false;\n  }\n\n  /**\n   * Installs the apks on to the device or emulator.\n   *\n   * @param {number} installTimeout - Installation timeout\n   */\n  async installServerApk (installTimeout = SERVER_INSTALL_RETRIES * 1000) {\n    const tmpRoot = await tempDir.openDir();\n    const packageInfosMapper = async ({appPath, appId}) => {\n      if (await helpers.isWriteable(appPath)) {\n        return { appPath, appId };\n      }\n\n      this.log.info(`Server package at '${appPath}' is not writeable. ` +\n        `Will copy it into the temporary location at '${tmpRoot}' as a workaround. ` +\n        `Consider making this file writeable manually in order to improve the performance of session startup.`);\n      const dstPath = path.resolve(tmpRoot, path.basename(appPath));\n      await fs.copyFile(appPath, dstPath);\n      return {\n        appPath: dstPath,\n        appId,\n      };\n    };\n\n    try {\n      const packagesInfo = await B.all(B.map([\n        {\n          appPath: apkPath,\n          appId: SERVER_PACKAGE_ID,\n        }, {\n          appPath: testApkPath,\n          appId: SERVER_TEST_PACKAGE_ID,\n        },\n      ], packageInfosMapper));\n\n      let shouldUninstallServerPackages = false;\n      let shouldInstallServerPackages = false;\n      for (const {appId, appPath} of packagesInfo) {\n        if (appId === SERVER_TEST_PACKAGE_ID) {\n          const isAppInstalled = await this.adb.isAppInstalled(appId);\n\n          // There is no point in getting the state for test server,\n          // since it does not contain version info\n          if (!await this.adb.checkApkCert(appPath, appId)) {\n            await helpers.signApp(this.adb, appPath);\n            shouldUninstallServerPackages = shouldUninstallServerPackages || isAppInstalled;\n            shouldInstallServerPackages = true;\n          }\n\n          if (!isAppInstalled) {\n            shouldInstallServerPackages = true;\n          }\n          continue;\n        }\n\n        const appState = await this.adb.getApplicationInstallState(appPath, appId);\n        this.log.debug(`${appId} installation state: ${appState}`);\n        if (await this.adb.checkApkCert(appPath, appId)) {\n          shouldUninstallServerPackages = shouldUninstallServerPackages || [\n            this.adb.APP_INSTALL_STATE.OLDER_VERSION_INSTALLED,\n            this.adb.APP_INSTALL_STATE.NEWER_VERSION_INSTALLED,\n          ].includes(appState);\n        } else {\n          await helpers.signApp(this.adb, appPath);\n          shouldUninstallServerPackages = shouldUninstallServerPackages || ![\n            this.adb.APP_INSTALL_STATE.NOT_INSTALLED,\n          ].includes(appState);\n        }\n        shouldInstallServerPackages = shouldInstallServerPackages || shouldUninstallServerPackages || [\n          this.adb.APP_INSTALL_STATE.NOT_INSTALLED,\n        ].includes(appState);\n      }\n      this.log.info(`Server packages are ${shouldInstallServerPackages ? '' : 'not '}going to be (re)installed`);\n      if (shouldInstallServerPackages && shouldUninstallServerPackages) {\n        this.log.info('Full packages reinstall is going to be performed');\n      }\n      for (const {appId, appPath} of packagesInfo) {\n        if (shouldUninstallServerPackages) {\n          try {\n            await this.adb.uninstallApk(appId);\n          } catch (err) {\n            this.log.warn(`Error uninstalling '${appId}': ${err.message}`);\n          }\n        }\n        if (shouldInstallServerPackages) {\n          await this.adb.install(appPath, {\n            noIncremental: true,\n            replace: true,\n            timeout: installTimeout,\n            timeoutCapName: 'uiautomator2ServerInstallTimeout'\n          });\n        }\n      }\n    } finally {\n      await fs.rimraf(tmpRoot);\n    }\n\n    await this.verifyServicesAvailability();\n  }\n\n  async verifyServicesAvailability () {\n    this.log.debug(`Waiting up to ${SERVICES_LAUNCH_TIMEOUT}ms for services to be available`);\n    let isPmServiceAvailable = false;\n    let pmOutput = '';\n    let pmError = null;\n    try {\n      await waitForCondition(async () => {\n        if (!isPmServiceAvailable) {\n          pmError = null;\n          pmOutput = '';\n          try {\n            pmOutput = await this.adb.shell(['pm', 'list', 'instrumentation']);\n          } catch (e) {\n            pmError = e;\n          }\n          if (pmOutput.includes('Could not access the Package Manager')) {\n            pmError = new Error(`Problem running Package Manager: ${pmOutput}`);\n            pmOutput = ''; // remove output, so it is not printed below\n          } else if (pmOutput.includes(INSTRUMENTATION_TARGET)) {\n            pmOutput = ''; // remove output, so it is not printed below\n            this.log.debug(`Instrumentation target '${INSTRUMENTATION_TARGET}' is available`);\n            // eslint-disable-next-line require-atomic-updates\n            isPmServiceAvailable = true;\n          } else if (!pmError) {\n            pmError = new Error('The instrumentation target is not listed by Package Manager');\n          }\n        }\n        return isPmServiceAvailable;\n      }, {\n        waitMs: SERVICES_LAUNCH_TIMEOUT,\n        intervalMs: 1000,\n      });\n    } catch (err) {\n      this.log.error(`Unable to find instrumentation target '${INSTRUMENTATION_TARGET}': ${(pmError || {}).message}`);\n      if (pmOutput) {\n        this.log.debug('Available targets:');\n        for (const line of pmOutput.split('\\n')) {\n          this.log.debug(`    ${line.replace('instrumentation:', '')}`);\n        }\n      }\n    }\n  }\n\n  async startSession (caps) {\n    await this.cleanupAutomationLeftovers();\n    if (caps.skipServerInstallation) {\n      this.log.info(`'skipServerInstallation' is set. Attempting to use UIAutomator2 server from the device`);\n    } else {\n      this.log.info(`Starting UIAutomator2 server ${serverVersion}`);\n      this.log.info(`Using UIAutomator2 server from '${apkPath}' and test from '${testApkPath}'`);\n    }\n\n    const timeout = caps.uiautomator2ServerLaunchTimeout || SERVER_LAUNCH_TIMEOUT;\n    const timer = new timing.Timer().start();\n    let retries = 0;\n    const maxRetries = 2;\n    const delayBetweenRetries = 3000;\n    while (retries < maxRetries) {\n      this.log.info(`Waiting up to ${timeout}ms for UiAutomator2 to be online...`);\n      this.jwproxy.didInstrumentationExit = false;\n      await this.startInstrumentationProcess();\n      if (!this.jwproxy.didInstrumentationExit) {\n        try {\n          await waitForCondition(async () => {\n            try {\n              await this.jwproxy.command('/status', 'GET');\n              return true;\n            } catch (err) {\n              // short circuit to retry or fail fast\n              return this.jwproxy.didInstrumentationExit;\n            }\n          }, {\n            waitMs: timeout,\n            intervalMs: 1000,\n          });\n        } catch (err) {\n          this.log.errorAndThrow(`The instrumentation process cannot be initialized within ${timeout}ms timeout. `\n            + 'Make sure the application under test does not crash and investigate the logcat output. '\n            + `You could also try to increase the value of 'uiautomator2ServerLaunchTimeout' capability`);\n        }\n      }\n      if (!this.jwproxy.didInstrumentationExit) {\n        break;\n      }\n\n      retries++;\n      if (retries >= maxRetries) {\n        this.log.errorAndThrow('The instrumentation process cannot be initialized. '\n          + 'Make sure the application under test does not crash and investigate the logcat output.');\n      }\n      this.log.warn(`The instrumentation process has been unexpectedly terminated. `\n        + `Retrying UiAutomator2 startup (#${retries} of ${maxRetries - 1})`);\n      await this.cleanupAutomationLeftovers(true);\n      await B.delay(delayBetweenRetries);\n    }\n\n    this.log.debug(`The initialization of the instrumentation process took `\n      + `${timer.getDuration().asMilliSeconds.toFixed(0)}ms`);\n    await this.jwproxy.command('/session', 'POST', {\n      capabilities: {\n        firstMatch: [caps],\n        alwaysMatch: {},\n      }\n    });\n  }\n\n  async startInstrumentationProcess () {\n    const cmd = ['am', 'instrument', '-w'];\n    if (this.disableWindowAnimation) {\n      cmd.push('--no-window-animation');\n    }\n    if (_.isBoolean(this.disableSuppressAccessibilityService)) {\n      cmd.push('-e', 'DISABLE_SUPPRESS_ACCESSIBILITY_SERVICES', this.disableSuppressAccessibilityService);\n    }\n    // Disable Google analytics to prevent possible fatal exception\n    cmd.push('-e', 'disableAnalytics', true);\n    cmd.push(INSTRUMENTATION_TARGET);\n    const instrumentationProcess = this.adb.createSubProcess(['shell', ...cmd]);\n    instrumentationProcess.on('output', (stdout, stderr) => {\n      const output = _.trim(stdout || stderr);\n      if (output) {\n        instrumentationLogger.debug(output);\n      }\n    });\n    instrumentationProcess.on('exit', (code) => {\n      instrumentationLogger.debug(`The process has exited with code ${code}`);\n      this.jwproxy.didInstrumentationExit = true;\n    });\n    await instrumentationProcess.start(0);\n  }\n\n  async deleteSession () {\n    this.log.debug('Deleting UiAutomator2 server session');\n    // rely on jwproxy's intelligence to know what we're talking about and\n    // delete the current session\n    try {\n      await this.jwproxy.command('/', 'DELETE');\n    } catch (err) {\n      this.log.warn(`Did not get confirmation UiAutomator2 deleteSession worked; ` +\n          `Error was: ${err}`);\n    }\n  }\n\n  async cleanupAutomationLeftovers (strictCleanup = false) {\n    this.log.debug(`Performing ${strictCleanup ? 'strict' : 'shallow'} cleanup of automation leftovers`);\n\n    try {\n      const {value} = (await axios({\n        url: `http://${this.host}:${this.systemPort}/sessions`,\n        timeout: 500,\n      })).data;\n      const activeSessionIds = value.map(({id}) => id).filter(Boolean);\n      if (activeSessionIds.length) {\n        this.log.debug(`The following obsolete sessions are still running: ${JSON.stringify(activeSessionIds)}`);\n        this.log.debug(`Cleaning up ${util.pluralize('obsolete session', activeSessionIds.length, true)}`);\n        await B.all(activeSessionIds\n          .map((id) => axios.delete(`http://${this.host}:${this.systemPort}/session/${id}`))\n        );\n        // Let all sessions to be properly terminated before continuing\n        await B.delay(1000);\n      } else {\n        this.log.debug('No obsolete sessions have been detected');\n      }\n    } catch (e) {\n      this.log.debug(`No obsolete sessions have been detected (${e.message})`);\n    }\n\n    try {\n      await this.adb.forceStop(SERVER_TEST_PACKAGE_ID);\n    } catch (ignore) {}\n    if (!strictCleanup) {\n      return;\n    }\n    // https://github.com/appium/appium/issues/10749\n    try {\n      await this.adb.killProcessesByName('uiautomator');\n    } catch (ignore) {}\n  }\n}\n\nexport { UiAutomator2Server, INSTRUMENTATION_TARGET, SERVER_PACKAGE_ID, SERVER_TEST_PACKAGE_ID };\nexport default UiAutomator2Server;\n"],"file":"lib/uiautomator2.js","sourceRoot":"../.."}
|
package/lib/commands/element.js
CHANGED
|
@@ -1,4 +1,3 @@
|
|
|
1
|
-
import log from '../logger';
|
|
2
1
|
import _ from 'lodash';
|
|
3
2
|
import { util } from '@appium/support';
|
|
4
3
|
import { PROTOCOLS, W3C_ELEMENT_KEY } from '@appium/base-driver';
|
|
@@ -34,7 +33,6 @@ commands.getName = async function (elementId) {
|
|
|
34
33
|
};
|
|
35
34
|
|
|
36
35
|
commands.getLocation = async function (elementId) {
|
|
37
|
-
log.info(`calling get location: ${elementId}`);
|
|
38
36
|
return await this.uiautomator2.jwproxy.command(`/element/${elementId}/location`, 'GET', {});
|
|
39
37
|
};
|
|
40
38
|
|
|
@@ -113,7 +111,7 @@ commands.clear = async function (elementId) {
|
|
|
113
111
|
|
|
114
112
|
commands.getElementRect = async function (elementId) {
|
|
115
113
|
if (this.isWebContext()) {
|
|
116
|
-
log.debug(`Detected downstream chromedriver protocol: ${this.chromedriver.jwproxy.downstreamProtocol}`);
|
|
114
|
+
this.log.debug(`Detected downstream chromedriver protocol: ${this.chromedriver.jwproxy.downstreamProtocol}`);
|
|
117
115
|
if (this.chromedriver.jwproxy.downstreamProtocol === PROTOCOLS.MJSONWP) {
|
|
118
116
|
const {x, y} = await this.chromedriver.jwproxy.command(`/element/${elementId}/location`, 'GET');
|
|
119
117
|
const {width, height} = await this.chromedriver.jwproxy.command(`/element/${elementId}/size`, 'GET');
|
package/lib/commands/general.js
CHANGED
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
import _ from 'lodash';
|
|
2
|
-
import log from '../logger';
|
|
3
2
|
import B from 'bluebird';
|
|
4
3
|
import { errors } from '@appium/base-driver';
|
|
5
4
|
import { fs, tempDir } from '@appium/support';
|
|
@@ -26,7 +25,7 @@ commands.doSendKeys = async function (params) {
|
|
|
26
25
|
|
|
27
26
|
// uiautomator2 doesn't support metastate for keyevents
|
|
28
27
|
commands.keyevent = async function (keycode, metastate) {
|
|
29
|
-
log.debug(`Ignoring metastate ${metastate}`);
|
|
28
|
+
this.log.debug(`Ignoring metastate ${metastate}`);
|
|
30
29
|
await this.adb.keyevent(keycode);
|
|
31
30
|
};
|
|
32
31
|
|
|
@@ -38,7 +37,7 @@ commands.back = async function () {
|
|
|
38
37
|
commands.getStrings = async function (language) {
|
|
39
38
|
if (!language) {
|
|
40
39
|
language = await this.adb.getDeviceLanguage();
|
|
41
|
-
log.info(`No language specified, returning strings for: ${language}`);
|
|
40
|
+
this.log.info(`No language specified, returning strings for: ${language}`);
|
|
42
41
|
}
|
|
43
42
|
|
|
44
43
|
// Clients require the resulting mapping to have both keys
|
|
@@ -57,7 +56,7 @@ commands.getStrings = async function (language) {
|
|
|
57
56
|
}
|
|
58
57
|
|
|
59
58
|
if (!this.opts.app && !this.opts.appPackage) {
|
|
60
|
-
log.errorAndThrow("One of 'app' or 'appPackage' capabilities should must be specified");
|
|
59
|
+
this.log.errorAndThrow("One of 'app' or 'appPackage' capabilities should must be specified");
|
|
61
60
|
}
|
|
62
61
|
|
|
63
62
|
let app = this.opts.app;
|
|
@@ -67,12 +66,12 @@ commands.getStrings = async function (language) {
|
|
|
67
66
|
try {
|
|
68
67
|
app = await this.adb.pullApk(this.opts.appPackage, tmpRoot);
|
|
69
68
|
} catch (err) {
|
|
70
|
-
log.errorAndThrow(`Failed to pull an apk from '${this.opts.appPackage}'. Original error: ${err.message}`);
|
|
69
|
+
this.log.errorAndThrow(`Failed to pull an apk from '${this.opts.appPackage}'. Original error: ${err.message}`);
|
|
71
70
|
}
|
|
72
71
|
}
|
|
73
72
|
|
|
74
73
|
if (!await fs.exists(app)) {
|
|
75
|
-
log.errorAndThrow(`The app at '${app}' does not exist`);
|
|
74
|
+
this.log.errorAndThrow(`The app at '${app}' does not exist`);
|
|
76
75
|
}
|
|
77
76
|
|
|
78
77
|
try {
|
|
@@ -80,7 +79,7 @@ commands.getStrings = async function (language) {
|
|
|
80
79
|
this.apkStrings[language] = apkStrings;
|
|
81
80
|
return preprocessStringsMap(apkStrings);
|
|
82
81
|
} catch (err) {
|
|
83
|
-
log.errorAndThrow(`Cannot extract strings from '${app}'. Original error: ${err.message}`);
|
|
82
|
+
this.log.errorAndThrow(`Cannot extract strings from '${app}'. Original error: ${err.message}`);
|
|
84
83
|
}
|
|
85
84
|
} finally {
|
|
86
85
|
await fs.rimraf(tmpRoot);
|
|
@@ -285,7 +284,7 @@ commands.mobileType = async function mobileType (opts = {}) {
|
|
|
285
284
|
text,
|
|
286
285
|
} = opts;
|
|
287
286
|
if (_.isUndefined(text)) {
|
|
288
|
-
log.errorAndThrow(`The 'text' argument is mandatory`);
|
|
287
|
+
this.log.errorAndThrow(`The 'text' argument is mandatory`);
|
|
289
288
|
}
|
|
290
289
|
return await this.adb.typeUnicode(text);
|
|
291
290
|
};
|
|
@@ -1,6 +1,4 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
let commands = {};
|
|
1
|
+
const commands = {};
|
|
4
2
|
|
|
5
3
|
commands.getScreenshot = async function () {
|
|
6
4
|
if (this.mjpegStream) {
|
|
@@ -8,8 +6,8 @@ commands.getScreenshot = async function () {
|
|
|
8
6
|
if (data) {
|
|
9
7
|
return data;
|
|
10
8
|
}
|
|
11
|
-
|
|
12
|
-
|
|
9
|
+
this.log.warn('Tried to get screenshot from active MJPEG stream, but there ' +
|
|
10
|
+
'was no data yet. Falling back to regular screenshot methods.');
|
|
13
11
|
}
|
|
14
12
|
return await this.uiautomator2.jwproxy.command('/screenshot', 'GET');
|
|
15
13
|
};
|
package/lib/commands/touch.js
CHANGED
|
@@ -1,5 +1,3 @@
|
|
|
1
|
-
import log from '../logger';
|
|
2
|
-
|
|
3
1
|
let commands = {}, extensions = {};
|
|
4
2
|
|
|
5
3
|
commands.doPerformMultiAction = async function (elementId, states) {
|
|
@@ -20,7 +18,7 @@ commands.doPerformMultiAction = async function (elementId, states) {
|
|
|
20
18
|
};
|
|
21
19
|
|
|
22
20
|
commands.performActions = async function (actions) {
|
|
23
|
-
log.debug(`Received the following W3C actions: ${JSON.stringify(actions, null, ' ')}`);
|
|
21
|
+
this.log.debug(`Received the following W3C actions: ${JSON.stringify(actions, null, ' ')}`);
|
|
24
22
|
// This is mandatory, since Selenium API uses MOUSE as the default pointer type
|
|
25
23
|
const preprocessedActions = actions
|
|
26
24
|
.map((action) => Object.assign({}, action, action.type === 'pointer' ? {
|
|
@@ -28,8 +26,8 @@ commands.performActions = async function (actions) {
|
|
|
28
26
|
pointerType: 'touch'
|
|
29
27
|
}
|
|
30
28
|
} : {}));
|
|
31
|
-
log.debug(`Preprocessed actions: ${JSON.stringify(preprocessedActions, null, ' ')}`);
|
|
32
|
-
return await this.uiautomator2.jwproxy.command('/actions', 'POST', {actions});
|
|
29
|
+
this.log.debug(`Preprocessed actions: ${JSON.stringify(preprocessedActions, null, ' ')}`);
|
|
30
|
+
return await this.uiautomator2.jwproxy.command('/actions', 'POST', {actions: preprocessedActions});
|
|
33
31
|
};
|
|
34
32
|
|
|
35
33
|
Object.assign(extensions, commands);
|