appium-android-driver 5.1.0 → 5.2.2

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.
@@ -598,7 +598,17 @@ helpers.pushSettingsApp = async function pushSettingsApp(adb, throwError = false
598
598
  return;
599
599
  }
600
600
 
601
- if ((await adb.getApiLevel()) <= 23) {
601
+ const apiLevel = await adb.getApiLevel();
602
+
603
+ if (apiLevel >= 29) {
604
+ try {
605
+ await adb.shell(['appops', 'set', SETTINGS_HELPER_PKG_ID, 'PROJECT_MEDIA', 'allow']);
606
+ } catch (err) {
607
+ _logger.default.debug(err);
608
+ }
609
+ }
610
+
611
+ if (apiLevel <= 23) {
602
612
  const perms = ['SET_ANIMATION_SCALE', 'CHANGE_CONFIGURATION', 'ACCESS_FINE_LOCATION'];
603
613
 
604
614
  _logger.default.info(`Granting permissions ${perms} to '${SETTINGS_HELPER_PKG_ID}'`);
@@ -901,4 +911,4 @@ var _default = helpers;
901
911
  exports.default = _default;require('source-map-support').install();
902
912
 
903
913
 
904
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"file":"lib/android-helpers.js","names":["PACKAGE_INSTALL_TIMEOUT","CHROME_BROWSER_PACKAGE_ACTIVITY","chrome","pkg","activity","chromium","chromebeta","browser","default","SETTINGS_HELPER_PKG_ID","SETTINGS_HELPER_UNLOCK_ACTIVITY","EMULATOR_PATTERN","APP_STATE","Object","freeze","NOT_INSTALLED","NOT_RUNNING","RUNNING_IN_BACKGROUND","RUNNING_IN_FOREGROUND","ensureNetworkSpeed","adb","networkSpeed","_","values","NETWORK_SPEED","includes","logger","warn","FULL","prepareAvdArgs","opts","isHeadless","avdArgs","result","isArray","push","util","shellParse","toCredentialType","unlockType","PIN_UNLOCK","PIN_UNLOCK_KEY_EVENT","PASSWORD_UNLOCK","PATTERN_UNLOCK","Error","helpers","createBaseADB","adbPort","suppressKillServer","remoteAdbHost","clearDeviceLogsOnStart","adbExecTimeout","useKeystore","keystorePath","keystorePassword","keyAlias","keyPassword","remoteAppsCacheLimit","buildToolsVersion","allowOfflineDevices","allowDelayAdb","ADB","createADB","prepareEmulator","avd","avdEnv","env","language","locale","country","avdLaunchTimeout","launchTimeout","avdReadyTimeout","readyTimeout","avdName","replace","runningAVD","getRunningAVD","args","debug","killEmulator","launchAVD","ensureDeviceLocale","script","isString","setDeviceLanguageCountry","ensureCurrentLocale","message","getDeviceInfoFromCaps","udid","emPort","curDeviceId","emulatorPort","info","devices","getDevicesWithRetry","map","errorAndThrow","getPortFromEmulatorString","platformVersion","trim","semver","coerce","availDevices","partialMatchCandidate","device","setDeviceId","rawDeviceOS","getPlatformVersion","deviceOS","bothVersionsCanBeCoerced","valid","bothVersionsAreStrings","version","toLower","major","minor","gt","keys","join","setEmulatorPort","validatePackageActivityNames","key","name","match","exec","index","substring","getLaunchInfo","app","appPackage","appActivity","appWaitPackage","appWaitActivity","apkPackage","apkActivity","packageAndLaunchActivityFromManifest","resetApp","fastReset","fullReset","androidInstallTimeout","autoGrantPermissions","allowTestPackages","isInstalled","isAppInstalled","forceStop","ign","output","clear","toLowerCase","grantAllPermissions","error","uninstallApk","install","grantPermissions","timeout","installApk","enforceAppInstall","appState","wasUninstalled","installOrUpgrade","enforceCurrentBuild","isInstalledOverExistingApp","APP_INSTALL_STATE","installOtherApks","otherApps","B","all","otherApp","uninstallOtherPackages","appPackages","filterPackages","getThirdPartyPackages","packagesString","shell","appPackagesArray","split","EOL","difference","err","initUnicodeKeyboard","defaultIME","appiumIME","enableIME","setIME","setMockLocationApp","getApiLevel","installHelperApp","apkPath","packageId","retry","retryInstallHelperApp","pushSettingsApp","throwError","settingsApkPath","processExists","perms","x","requireRunningSettingsApp","pushStrings","remoteDir","stringsJson","remoteFile","path","posix","resolve","rimraf","pullApk","tmpDir","isEmpty","fs","exists","stringsTmpDir","apkStrings","localPath","extractStringsFromApk","unlockWithHelperApp","firstRun","launchHelper","isScreenLocked","e","delay","unlock","driver","capabilities","unlockKey","unlockStrategy","unlockSuccessTimeout","unlocker","validateUnlockCapabilities","FINGERPRINT_UNLOCK","isNil","isLockManagementSupported","fastUnlock","credential","credentialType","unlockMethod","pinUnlock","pinUnlockWithKeyEvent","passwordUnlock","patternUnlock","fingerprintUnlock","verifyUnlock","timeoutMs","waitForCondition","waitMs","intervalMs","initDevice","skipDeviceInitialization","localeScript","unicodeKeyboard","disableWindowAnimation","skipUnlock","mockLocationApp","skipLogcatCapture","logcatFormat","logcatFilterSpecs","waitForDevice","shouldThrowError","isEmulator","isUndefined","startLogcat","format","filterSpecs","removeNullProperties","obj","isNull","truncateDecimals","number","digits","multiplier","Math","pow","adjustedNum","truncatedNum","isChromeBrowser","getChromePkg","removeAllSessionWebSocketHandlers","server","sessionId","isFunction","getWebSocketHandlers","activeHandlers","pathname","removeWebSocketHandler","parseArray","cap","parsedCaps","JSON","parse","validateDesiredCaps","caps","browserName","adjustBrowserSessionCaps","stringify","possibleNames","some","test","bootstrap","Bootstrap"],"sourceRoot":"../..","sources":["lib/android-helpers.js"],"sourcesContent":["import _ from 'lodash';\nimport path from 'path';\nimport { retry, waitForCondition } from 'asyncbox';\nimport logger from './logger';\nimport { fs, util } from 'appium/support';\nimport { path as settingsApkPath } from 'io.appium.settings';\nimport Bootstrap from './bootstrap';\nimport B from 'bluebird';\nimport ADB from 'appium-adb';\nimport {\n  default as unlocker, PIN_UNLOCK, PIN_UNLOCK_KEY_EVENT,\n  PASSWORD_UNLOCK, PATTERN_UNLOCK, FINGERPRINT_UNLOCK\n} from './unlock-helpers';\nimport { EOL } from 'os';\nimport semver from 'semver';\n\nconst PACKAGE_INSTALL_TIMEOUT = 90000; // milliseconds\n// https://cs.chromium.org/chromium/src/chrome/browser/devtools/device/android_device_info_query.cc\nconst CHROME_BROWSER_PACKAGE_ACTIVITY = {\n  chrome: {\n    pkg: 'com.android.chrome',\n    activity: 'com.google.android.apps.chrome.Main',\n  },\n  chromium: {\n    pkg: 'org.chromium.chrome.shell',\n    activity: '.ChromeShellActivity',\n  },\n  chromebeta: {\n    pkg: 'com.chrome.beta',\n    activity: 'com.google.android.apps.chrome.Main',\n  },\n  browser: {\n    pkg: 'com.android.browser',\n    activity: 'com.android.browser.BrowserActivity',\n  },\n  'chromium-browser': {\n    pkg: 'org.chromium.chrome',\n    activity: 'com.google.android.apps.chrome.Main',\n  },\n  'chromium-webview': {\n    pkg: 'org.chromium.webview_shell',\n    activity: 'org.chromium.webview_shell.WebViewBrowserActivity',\n  },\n  default: {\n    pkg: 'com.android.chrome',\n    activity: 'com.google.android.apps.chrome.Main',\n  },\n};\nconst SETTINGS_HELPER_PKG_ID = 'io.appium.settings';\nconst SETTINGS_HELPER_UNLOCK_ACTIVITY = '.Unlock';\nconst EMULATOR_PATTERN = /\\bemulator\\b/i;\n// These constants are in sync with\n// https://developer.apple.com/documentation/xctest/xcuiapplicationstate/xcuiapplicationstaterunningbackground?language=objc\nconst APP_STATE = Object.freeze({\n  NOT_INSTALLED: 0,\n  NOT_RUNNING: 1,\n  RUNNING_IN_BACKGROUND: 3,\n  RUNNING_IN_FOREGROUND: 4\n});\n\n\nfunction ensureNetworkSpeed (adb, networkSpeed) {\n  if (_.values(adb.NETWORK_SPEED).includes(networkSpeed)) {\n    return networkSpeed;\n  }\n  logger.warn(`Wrong network speed param '${networkSpeed}', using default: ${adb.NETWORK_SPEED.FULL}. ` +\n    `Supported values: ${_.values(adb.NETWORK_SPEED)}`);\n  return adb.NETWORK_SPEED.FULL;\n}\n\nfunction prepareAvdArgs (adb, opts) {\n  const {\n    networkSpeed,\n    isHeadless,\n    avdArgs,\n  } = opts;\n  const result = [];\n  if (avdArgs) {\n    if (_.isArray(avdArgs)) {\n      result.push(...avdArgs);\n    } else {\n      result.push(...(util.shellParse(`${avdArgs}`)));\n    }\n  }\n  if (networkSpeed) {\n    result.push('-netspeed', ensureNetworkSpeed(adb, networkSpeed));\n  }\n  if (isHeadless) {\n    result.push('-no-window');\n  }\n  return result;\n}\n\nfunction toCredentialType (unlockType) {\n  const result = {\n    [PIN_UNLOCK]: 'pin',\n    [PIN_UNLOCK_KEY_EVENT]: 'pin',\n    [PASSWORD_UNLOCK]: 'password',\n    [PATTERN_UNLOCK]: 'pattern',\n  }[unlockType];\n  if (result) {\n    return result;\n  }\n  throw new Error(`Unlock type '${unlockType}' is not known`);\n}\n\n\nconst helpers = {};\n\nhelpers.createBaseADB = async function createBaseADB (opts = {}) {\n  // filter out any unwanted options sent in\n  // this list should be updated as ADB takes more arguments\n  const {\n    adbPort,\n    suppressKillServer,\n    remoteAdbHost,\n    clearDeviceLogsOnStart,\n    adbExecTimeout,\n    useKeystore,\n    keystorePath,\n    keystorePassword,\n    keyAlias,\n    keyPassword,\n    remoteAppsCacheLimit,\n    buildToolsVersion,\n    allowOfflineDevices,\n    allowDelayAdb,\n  } = opts;\n  return await ADB.createADB({\n    adbPort,\n    suppressKillServer,\n    remoteAdbHost,\n    clearDeviceLogsOnStart,\n    adbExecTimeout,\n    useKeystore,\n    keystorePath,\n    keystorePassword,\n    keyAlias,\n    keyPassword,\n    remoteAppsCacheLimit,\n    buildToolsVersion,\n    allowOfflineDevices,\n    allowDelayAdb,\n  });\n};\n\nhelpers.prepareEmulator = async function prepareEmulator (adb, opts) {\n  const {\n    avd,\n    avdEnv: env,\n    language,\n    locale: country,\n    avdLaunchTimeout: launchTimeout,\n    avdReadyTimeout: readyTimeout,\n  } = opts;\n  if (!avd) {\n    throw new Error('Cannot launch AVD without AVD name');\n  }\n\n  const avdName = avd.replace('@', '');\n  const runningAVD = await adb.getRunningAVD(avdName);\n  const args = prepareAvdArgs(adb, opts);\n  if (runningAVD) {\n    if (args.includes('-wipe-data')) {\n      logger.debug(`Killing '${avdName}' because it needs to be wiped at start.`);\n      await adb.killEmulator(avdName);\n    } else {\n      logger.debug('Not launching AVD because it is already running.');\n      return;\n    }\n  }\n  await adb.launchAVD(avd, {\n    args,\n    env,\n    language,\n    country,\n    launchTimeout,\n    readyTimeout,\n  });\n};\n\n/**\n * Set and ensure the locale name of the device under test.\n *\n * @param {Object} adb - The adb module instance.\n * @param {string} language - Language. The language field is case insensitive, but Locale always canonicalizes to lower case.\n *                            format: [a-zA-Z]{2,8}. e.g. en, ja : https://developer.android.com/reference/java/util/Locale.html\n * @param {string} country - Country. The country (region) field is case insensitive, but Locale always canonicalizes to upper case.\n *                            format: [a-zA-Z]{2} | [0-9]{3}. e.g. US, JP : https://developer.android.com/reference/java/util/Locale.html\n * @param {?string} script - Script. The script field is case insensitive but Locale always canonicalizes to title case.\n *                            format: [a-zA-Z]{4}. e.g. Hans in zh-Hans-CN : https://developer.android.com/reference/java/util/Locale.html\n * @throws {Error} If it failed to set locale properly\n */\nhelpers.ensureDeviceLocale = async function ensureDeviceLocale (adb, language, country, script = null) {\n  if (!_.isString(language) && !_.isString(country)) {\n    logger.warn(`setDeviceLanguageCountry requires language or country.`);\n    logger.warn(`Got language: '${language}' and country: '${country}'`);\n    return;\n  }\n\n  await adb.setDeviceLanguageCountry(language, country, script);\n\n  if (!await adb.ensureCurrentLocale(language, country, script)) {\n    const message = script ? `language: ${language}, country: ${country} and script: ${script}` : `language: ${language} and country: ${country}`;\n    throw new Error(`Failed to set ${message}`);\n  }\n};\n\nhelpers.getDeviceInfoFromCaps = async function getDeviceInfoFromCaps (opts = {}) {\n  // we can create a throwaway ADB instance here, so there is no dependency\n  // on instantiating on earlier (at this point, we have no udid)\n  // we can only use this ADB object for commands that would not be confused\n  // if multiple devices are connected\n  const adb = await helpers.createBaseADB(opts);\n  let udid = opts.udid;\n  let emPort = null;\n\n  // a specific avd name was given. try to initialize with that\n  if (opts.avd) {\n    await helpers.prepareEmulator(adb, opts);\n    udid = adb.curDeviceId;\n    emPort = adb.emulatorPort;\n  } else {\n    // no avd given. lets try whatever's plugged in devices/emulators\n    logger.info('Retrieving device list');\n    let devices = await adb.getDevicesWithRetry();\n\n    // udid was given, lets try to init with that device\n    if (udid) {\n      if (!_.includes(_.map(devices, 'udid'), udid)) {\n        logger.errorAndThrow(`Device ${udid} was not in the list of connected devices`);\n      }\n      emPort = adb.getPortFromEmulatorString(udid);\n    } else if (opts.platformVersion) {\n      opts.platformVersion = `${opts.platformVersion}`.trim();\n\n      // a platform version was given. lets try to find a device with the same os\n      const platformVersion = semver.coerce(opts.platformVersion) || opts.platformVersion;\n      logger.info(`Looking for a device with Android '${platformVersion}'`);\n\n      // in case we fail to find something, give the user a useful log that has\n      // the device udids and os versions so they know what's available\n      const availDevices = [];\n      let partialMatchCandidate = null;\n      // first try started devices/emulators\n      for (const device of devices) {\n        // direct adb calls to the specific device\n        await adb.setDeviceId(device.udid);\n        const rawDeviceOS = await adb.getPlatformVersion();\n        // The device OS could either be a number, like `6.0`\n        // or an abbreviation, like `R`\n        availDevices.push(`${device.udid} (${rawDeviceOS})`);\n        const deviceOS = semver.coerce(rawDeviceOS) || rawDeviceOS;\n        if (!deviceOS) {\n          continue;\n        }\n\n        const bothVersionsCanBeCoerced = semver.valid(deviceOS) && semver.valid(platformVersion);\n        const bothVersionsAreStrings = _.isString(deviceOS) && _.isString(platformVersion);\n        if (bothVersionsCanBeCoerced && deviceOS.version === platformVersion.version\n            || bothVersionsAreStrings && _.toLower(deviceOS) === _.toLower(platformVersion)) {\n          // Got an exact match - proceed immediately\n          udid = device.udid;\n          break;\n        } else if (!bothVersionsCanBeCoerced) {\n          // There is no point to check for partial match if either of version numbers is not coercible\n          continue;\n        }\n\n        if ((!_.includes(opts.platformVersion, '.') && platformVersion.major === deviceOS.major\n            || platformVersion.major === deviceOS.major && platformVersion.minor === deviceOS.minor)\n            // Got a partial match - make sure we consider the most recent\n            // device version available on the host system\n            && (partialMatchCandidate && semver.gt(deviceOS, _.values(partialMatchCandidate)[0])\n                || !partialMatchCandidate)) {\n          partialMatchCandidate = {[device.udid]: deviceOS};\n        }\n      }\n      if (!udid && partialMatchCandidate) {\n        udid = _.keys(partialMatchCandidate)[0];\n        await adb.setDeviceId(udid);\n      }\n\n      if (!udid) {\n        // we couldn't find anything! quit\n        logger.errorAndThrow(`Unable to find an active device or emulator ` +\n          `with OS ${opts.platformVersion}. The following are available: ` +\n          availDevices.join(', '));\n      }\n\n      emPort = adb.getPortFromEmulatorString(udid);\n    } else {\n      // a udid was not given, grab the first device we see\n      udid = devices[0].udid;\n      emPort = adb.getPortFromEmulatorString(udid);\n    }\n  }\n\n  logger.info(`Using device: ${udid}`);\n  return {udid, emPort};\n};\n\n// returns a new adb instance with deviceId set\nhelpers.createADB = async function createADB (opts = {}) {\n  const {udid, emPort} = opts;\n  const adb = await helpers.createBaseADB(opts);\n  adb.setDeviceId(udid);\n  if (emPort) {\n    adb.setEmulatorPort(emPort);\n  }\n\n  return adb;\n};\n\nhelpers.validatePackageActivityNames = function validatePackageActivityNames (opts) {\n  for (const key of ['appPackage', 'appActivity', 'appWaitPackage', 'appWaitActivity']) {\n    const name = opts[key];\n    if (!name) {\n      continue;\n    }\n\n    const match = /([^\\w.*,])+/.exec(name);\n    if (!match) {\n      continue;\n    }\n\n    logger.warn(`Capability '${key}' is expected to only include latin letters, digits, underscore, dot, comma and asterisk characters.`);\n    logger.warn(`Current value '${name}' has non-matching character at index ${match.index}: '${name.substring(0, match.index + 1)}'`);\n  }\n};\n\nhelpers.getLaunchInfo = async function getLaunchInfo (adb, opts) {\n  let {app, appPackage, appActivity, appWaitPackage, appWaitActivity} = opts;\n  if (!app) {\n    logger.warn('No app sent in, not parsing package/activity');\n    return;\n  }\n\n  this.validatePackageActivityNames(opts);\n\n  if (appPackage && appActivity) {\n    return;\n  }\n\n  logger.debug('Parsing package and activity from app manifest');\n  let {apkPackage, apkActivity} =\n    await adb.packageAndLaunchActivityFromManifest(app);\n  if (apkPackage && !appPackage) {\n    appPackage = apkPackage;\n  }\n  if (!appWaitPackage) {\n    appWaitPackage = appPackage;\n  }\n  if (apkActivity && !appActivity) {\n    appActivity = apkActivity;\n  }\n  if (!appWaitActivity) {\n    appWaitActivity = appActivity;\n  }\n  logger.debug(`Parsed package and activity are: ${apkPackage}/${apkActivity}`);\n  return {appPackage, appWaitPackage, appActivity, appWaitActivity};\n};\n\nhelpers.resetApp = async function resetApp (adb, opts = {}) {\n  const {\n    app,\n    appPackage,\n    fastReset,\n    fullReset,\n    androidInstallTimeout = PACKAGE_INSTALL_TIMEOUT,\n    autoGrantPermissions,\n    allowTestPackages\n  } = opts;\n\n  if (!appPackage) {\n    throw new Error(\"'appPackage' option is required\");\n  }\n\n  const isInstalled = await adb.isAppInstalled(appPackage);\n\n  if (isInstalled) {\n    try {\n      await adb.forceStop(appPackage);\n    } catch (ign) {}\n    // fullReset has priority over fastReset\n    if (!fullReset && fastReset) {\n      const output = await adb.clear(appPackage);\n      if (_.isString(output) && output.toLowerCase().includes('failed')) {\n        throw new Error(`Cannot clear the application data of '${appPackage}'. Original error: ${output}`);\n      }\n      // executing `shell pm clear` resets previously assigned application permissions as well\n      if (autoGrantPermissions) {\n        try {\n          await adb.grantAllPermissions(appPackage);\n        } catch (error) {\n          logger.error(`Unable to grant permissions requested. Original error: ${error.message}`);\n        }\n      }\n      logger.debug(`Performed fast reset on the installed '${appPackage}' application (stop and clear)`);\n      return;\n    }\n  }\n\n  if (!app) {\n    throw new Error(\"'app' option is required for reinstall\");\n  }\n\n  logger.debug(`Running full reset on '${appPackage}' (reinstall)`);\n  if (isInstalled) {\n    await adb.uninstallApk(appPackage);\n  }\n  await adb.install(app, {\n    grantPermissions: autoGrantPermissions,\n    timeout: androidInstallTimeout,\n    allowTestPackages,\n  });\n};\n\nhelpers.installApk = async function installApk (adb, opts = {}) {\n  const {\n    app,\n    appPackage,\n    fastReset,\n    fullReset,\n    androidInstallTimeout = PACKAGE_INSTALL_TIMEOUT,\n    autoGrantPermissions,\n    allowTestPackages,\n    enforceAppInstall,\n  } = opts;\n\n  if (!app || !appPackage) {\n    throw new Error(\"'app' and 'appPackage' options are required\");\n  }\n\n  if (fullReset) {\n    await this.resetApp(adb, opts);\n    return;\n  }\n\n  const {\n    appState,\n    wasUninstalled\n  } = await adb.installOrUpgrade(app, appPackage, {\n    grantPermissions: autoGrantPermissions,\n    timeout: androidInstallTimeout,\n    allowTestPackages,\n    enforceCurrentBuild: enforceAppInstall,\n  });\n\n  // There is no need to reset the newly installed app\n  const isInstalledOverExistingApp = !wasUninstalled\n    && appState !== adb.APP_INSTALL_STATE.NOT_INSTALLED;\n  if (fastReset && isInstalledOverExistingApp) {\n    logger.info(`Performing fast reset on '${appPackage}'`);\n    await this.resetApp(adb, opts);\n  }\n};\n\n/**\n * Installs an array of apks\n * @param {ADB} adb Instance of Appium ADB object\n * @param {Object} opts Opts defined in driver.js\n */\nhelpers.installOtherApks = async function installOtherApks (otherApps, adb, opts) {\n  let {\n    androidInstallTimeout = PACKAGE_INSTALL_TIMEOUT,\n    autoGrantPermissions,\n    allowTestPackages\n  } = opts;\n\n  // Install all of the APK's asynchronously\n  await B.all(otherApps.map((otherApp) => {\n    logger.debug(`Installing app: ${otherApp}`);\n    return adb.installOrUpgrade(otherApp, null, {\n      grantPermissions: autoGrantPermissions,\n      timeout: androidInstallTimeout,\n      allowTestPackages,\n    });\n  }));\n};\n\n/**\n * Uninstall an array of packages\n * @param {ADB} adb Instance of Appium ADB object\n * @param {Array<string>} appPackages An array of package names to uninstall. If this includes `'*'`, uninstall all of 3rd party apps\n * @param {Array<string>} filterPackages An array of packages does not uninstall when `*` is provided as `appPackages`\n */\nhelpers.uninstallOtherPackages = async function uninstallOtherPackages (adb, appPackages, filterPackages = []) {\n  if (appPackages.includes('*')) {\n    logger.debug('Uninstall third party packages');\n    appPackages = await this.getThirdPartyPackages(adb, filterPackages);\n  }\n\n  logger.debug(`Uninstalling packages: ${appPackages}`);\n  await B.all(appPackages.map((appPackage) => adb.uninstallApk(appPackage)));\n};\n\n/**\n * Get third party packages filtered with `filterPackages`\n * @param {ADB} adb Instance of Appium ADB object\n * @param {Array<string>} filterPackages An array of packages does not uninstall when `*` is provided as `appPackages`\n * @returns {Array<string>} An array of installed third pary packages\n */\nhelpers.getThirdPartyPackages = async function getThirdPartyPackages (adb, filterPackages = []) {\n  try {\n    const packagesString = await adb.shell(['pm', 'list', 'packages', '-3']);\n    const appPackagesArray = packagesString.trim().replace(/package:/g, '').split(EOL);\n    logger.debug(`'${appPackagesArray}' filtered with '${filterPackages}'`);\n    return _.difference(appPackagesArray, filterPackages);\n  } catch (err) {\n    logger.warn(`Unable to get packages with 'adb shell pm list packages -3': ${err.message}`);\n    return [];\n  }\n};\n\nhelpers.initUnicodeKeyboard = async function initUnicodeKeyboard (adb) {\n  logger.debug('Enabling Unicode keyboard support');\n\n  // get the default IME so we can return back to it later if we want\n  let defaultIME = await adb.defaultIME();\n\n  logger.debug(`Unsetting previous IME ${defaultIME}`);\n  const appiumIME = `${SETTINGS_HELPER_PKG_ID}/.UnicodeIME`;\n  logger.debug(`Setting IME to '${appiumIME}'`);\n  await adb.enableIME(appiumIME);\n  await adb.setIME(appiumIME);\n  return defaultIME;\n};\n\nhelpers.setMockLocationApp = async function setMockLocationApp (adb, app) {\n  try {\n    if (await adb.getApiLevel() < 23) {\n      await adb.shell(['settings', 'put', 'secure', 'mock_location', '1']);\n    } else {\n      await adb.shell(['appops', 'set', app, 'android:mock_location', 'allow']);\n    }\n  } catch (err) {\n    logger.warn(`Unable to set mock location for app '${app}': ${err.message}`);\n  }\n};\n\nhelpers.installHelperApp = async function installHelperApp (adb, apkPath, packageId) {\n  // Sometimes adb push or adb instal take more time than expected to install an app\n  // e.g. https://github.com/appium/io.appium.settings/issues/40#issuecomment-476593174\n  await retry(2, async function retryInstallHelperApp () {\n    await adb.installOrUpgrade(apkPath, packageId, {grantPermissions: true});\n  });\n};\n\n/**\n * Pushes and installs io.appium.settings app.\n * Throws an error if the setting app is required\n *\n * @param {Adb} adb - The adb module instance.\n * @param {boolean} throwError[false] - Whether throw error or not\n * @throws {Error} If throwError is true and something happens in installation step\n */\nhelpers.pushSettingsApp = async function pushSettingsApp (adb, throwError = false) {\n  logger.debug('Pushing settings apk to device...');\n\n  try {\n    await helpers.installHelperApp(adb, settingsApkPath, SETTINGS_HELPER_PKG_ID, throwError);\n  } catch (err) {\n    if (throwError) {\n      throw err;\n    }\n\n    logger.warn(`Ignored error while installing '${settingsApkPath}': ` +\n                `'${err.message}'. Features that rely on this helper ` +\n                'require the apk such as toggle WiFi and getting location ' +\n                'will raise an error if you try to use them.');\n  }\n\n  // Reinstall will stop the settings helper process anyway, so\n  // there is no need to continue if the application is still running\n  if (await adb.processExists(SETTINGS_HELPER_PKG_ID)) {\n    logger.debug(`${SETTINGS_HELPER_PKG_ID} is already running. ` +\n      `There is no need to reset its permissions.`);\n    return;\n  }\n\n  if (await adb.getApiLevel() <= 23) { // Android 6- devices should have granted permissions\n    // https://github.com/appium/appium/pull/11640#issuecomment-438260477\n    const perms = ['SET_ANIMATION_SCALE', 'CHANGE_CONFIGURATION', 'ACCESS_FINE_LOCATION'];\n    logger.info(`Granting permissions ${perms} to '${SETTINGS_HELPER_PKG_ID}'`);\n    await adb.grantPermissions(SETTINGS_HELPER_PKG_ID, perms.map((x) => `android.permission.${x}`));\n  }\n\n  // launch io.appium.settings app due to settings failing to be set\n  // if the app is not launched prior to start the session on android 7+\n  // see https://github.com/appium/appium/issues/8957\n  try {\n    await adb.requireRunningSettingsApp();\n  } catch (err) {\n    logger.debug(err);\n    if (throwError) {\n      throw err;\n    }\n  }\n};\n\n/**\n * Extracts string.xml and converts it to string.json and pushes\n * it to /data/local/tmp/string.json on for use of bootstrap\n * If app is not present to extract string.xml it deletes remote strings.json\n * If app does not have strings.xml we push an empty json object to remote\n *\n * @param {?string} language - Language abbreviation, for example 'fr'. The default language\n * is used if this argument is not defined.\n * @param {Object} adb - The adb module instance.\n * @param {Object} opts - Driver options dictionary.\n * @returns {Object} The dictionary, where string resource identifiers are keys\n * along with their corresponding values for the given language or an empty object\n * if no matching resources were extracted.\n */\nhelpers.pushStrings = async function pushStrings (language, adb, opts) {\n  const remoteDir = '/data/local/tmp';\n  const stringsJson = 'strings.json';\n  const remoteFile = path.posix.resolve(remoteDir, stringsJson);\n\n  // clean up remote string.json if present\n  await adb.rimraf(remoteFile);\n\n  let app;\n  try {\n    app = opts.app || await adb.pullApk(opts.appPackage, opts.tmpDir);\n  } catch (err) {\n    logger.info(`Failed to pull an apk from '${opts.appPackage}' to '${opts.tmpDir}'. Original error: ${err.message}`);\n  }\n\n  if (_.isEmpty(opts.appPackage) || !(await fs.exists(app))) {\n    logger.debug(`No app or package specified. Returning empty strings`);\n    return {};\n  }\n\n  const stringsTmpDir = path.resolve(opts.tmpDir, opts.appPackage);\n  try {\n    logger.debug('Extracting strings from apk', app, language, stringsTmpDir);\n    const {apkStrings, localPath} = await adb.extractStringsFromApk(app, language, stringsTmpDir);\n    await adb.push(localPath, remoteDir);\n    return apkStrings;\n  } catch (err) {\n    logger.warn(`Could not get strings, continuing anyway. Original error: ${err.message}`);\n    await adb.shell('echo', [`'{}' > ${remoteFile}`]);\n  } finally {\n    await fs.rimraf(stringsTmpDir);\n  }\n  return {};\n};\n\nhelpers.unlockWithHelperApp = async function unlockWithHelperApp (adb) {\n  logger.info('Unlocking screen');\n\n  // Unlock succeed with a couple of retries.\n  let firstRun = true;\n  await retry(3, async function launchHelper () {\n    // To reduce a time to call adb.isScreenLocked() since `adb shell dumpsys window` is easy to hang adb commands\n    if (firstRun) {\n      firstRun = false;\n    } else {\n      try {\n        if (!(await adb.isScreenLocked())) {\n          return;\n        }\n      } catch (e) {\n        logger.warn(`Error in isScreenLocked: ${e.message}`);\n        logger.warn('\"adb shell dumpsys window\" command has timed out.');\n        logger.warn('The reason of this timeout is the delayed adb response. Resetting adb server can improve it.');\n      }\n    }\n\n    logger.info(`Launching ${SETTINGS_HELPER_UNLOCK_ACTIVITY}`);\n    await adb.shell([\n      'am', 'start',\n      '-n', `${SETTINGS_HELPER_PKG_ID}/${SETTINGS_HELPER_UNLOCK_ACTIVITY}`,\n      '-c', 'android.intent.category.LAUNCHER',\n      '-a', 'android.intent.action.MAIN',\n      '-f', '0x10200000',\n    ]);\n    await B.delay(1000);\n  });\n};\n\nhelpers.unlock = async function unlock (driver, adb, capabilities) {\n  if (!(await adb.isScreenLocked())) {\n    logger.info('Screen already unlocked, doing nothing');\n    return;\n  }\n  logger.debug('Screen is locked, trying to unlock');\n\n  if (!capabilities.unlockType && !capabilities.unlockKey) {\n    logger.warn('Using app unlock, this is going to be deprecated!');\n    await helpers.unlockWithHelperApp(adb);\n    return;\n  }\n\n  const {\n    unlockType,\n    unlockKey,\n    unlockStrategy,\n    unlockSuccessTimeout,\n  } = unlocker.validateUnlockCapabilities(capabilities);\n  if (unlockKey && unlockType !== FINGERPRINT_UNLOCK\n      && (_.isNil(unlockStrategy) || _.toLower(unlockStrategy) === 'locksettings')\n      && await adb.isLockManagementSupported()) {\n    await unlocker.fastUnlock(adb, {\n      credential: unlockKey,\n      credentialType: toCredentialType(unlockType),\n    });\n  } else {\n    const unlockMethod = {\n      [PIN_UNLOCK]: unlocker.pinUnlock,\n      [PIN_UNLOCK_KEY_EVENT]: unlocker.pinUnlockWithKeyEvent,\n      [PASSWORD_UNLOCK]: unlocker.passwordUnlock,\n      [PATTERN_UNLOCK]: unlocker.patternUnlock,\n      [FINGERPRINT_UNLOCK]: unlocker.fingerprintUnlock,\n    }[unlockType];\n    await unlockMethod(adb, driver, capabilities);\n  }\n  await helpers.verifyUnlock(adb, unlockSuccessTimeout);\n};\n\nhelpers.verifyUnlock = async function verifyUnlock (adb, timeoutMs = null) {\n  try {\n    await waitForCondition(async () => !(await adb.isScreenLocked()), {\n      waitMs: timeoutMs ?? 2000,\n      intervalMs: 500,\n    });\n  } catch (ign) {\n    throw new Error('The device has failed to be unlocked');\n  }\n  logger.info('The device has been successfully unlocked');\n};\n\nhelpers.initDevice = async function initDevice (adb, opts) {\n  const {\n    skipDeviceInitialization,\n    locale,\n    language,\n    localeScript,\n    unicodeKeyboard,\n    disableWindowAnimation,\n    skipUnlock,\n    mockLocationApp,\n    skipLogcatCapture,\n    logcatFormat,\n    logcatFilterSpecs,\n  } = opts;\n\n  if (skipDeviceInitialization) {\n    logger.info(`'skipDeviceInitialization' is set. Skipping device initialization.`);\n  } else {\n    await adb.waitForDevice();\n    // pushSettingsApp required before calling ensureDeviceLocale for API Level 24+\n\n    // Some feature such as location/wifi are not necessary for all users,\n    // but they require the settings app. So, try to configure it while Appium\n    // does not throw error even if they fail.\n    const shouldThrowError = language\n      || locale\n      || localeScript\n      || unicodeKeyboard\n      || disableWindowAnimation\n      || !skipUnlock;\n    await helpers.pushSettingsApp(adb, shouldThrowError);\n  }\n\n  if (!helpers.isEmulator(adb, opts) && (mockLocationApp || _.isUndefined(mockLocationApp))) {\n    await helpers.setMockLocationApp(adb, mockLocationApp || SETTINGS_HELPER_PKG_ID);\n  }\n\n  if (language || locale) {\n    await helpers.ensureDeviceLocale(adb, language, locale, localeScript);\n  }\n\n  if (skipLogcatCapture) {\n    logger.info(`'skipLogcatCapture' is set. Skipping starting logcat capture.`);\n  } else {\n    await adb.startLogcat({\n      format: logcatFormat,\n      filterSpecs: logcatFilterSpecs,\n    });\n  }\n\n  if (unicodeKeyboard) {\n    return await helpers.initUnicodeKeyboard(adb);\n  }\n};\n\nhelpers.removeNullProperties = function removeNullProperties (obj) {\n  for (let key of _.keys(obj)) {\n    if (_.isNull(obj[key]) || _.isUndefined(obj[key])) {\n      delete obj[key];\n    }\n  }\n};\n\nhelpers.truncateDecimals = function truncateDecimals (number, digits) {\n  let multiplier = Math.pow(10, digits),\n      adjustedNum = number * multiplier,\n      truncatedNum = Math[adjustedNum < 0 ? 'ceil' : 'floor'](adjustedNum);\n\n  return truncatedNum / multiplier;\n};\n\nhelpers.isChromeBrowser = function isChromeBrowser (browser) {\n  return _.includes(Object.keys(CHROME_BROWSER_PACKAGE_ACTIVITY), (browser || '').toLowerCase());\n};\n\nhelpers.getChromePkg = function getChromePkg (browser) {\n  return CHROME_BROWSER_PACKAGE_ACTIVITY[browser.toLowerCase()] || CHROME_BROWSER_PACKAGE_ACTIVITY.default;\n};\n\nhelpers.removeAllSessionWebSocketHandlers = async function removeAllSessionWebSocketHandlers (server, sessionId) {\n  if (!server || !_.isFunction(server.getWebSocketHandlers)) {\n    return;\n  }\n\n  const activeHandlers = await server.getWebSocketHandlers(sessionId);\n  for (const pathname of _.keys(activeHandlers)) {\n    await server.removeWebSocketHandler(pathname);\n  }\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 {any} cap A desired capability\n */\nhelpers.parseArray = function parseArray (cap) {\n  let parsedCaps;\n  try {\n    parsedCaps = JSON.parse(cap);\n  } catch (ign) { }\n\n  if (_.isArray(parsedCaps)) {\n    return parsedCaps;\n  } else if (_.isString(cap)) {\n    return [cap];\n  }\n\n  throw new Error(`must provide a string or JSON Array; received ${cap}`);\n};\n\n/**\n * Validate desired capabilities. Returns true if capability is valid\n *\n * @param {*} cap A desired capability\n * @return {boolean} Returns true if the capability is valid\n * @throws {Error} If the caps has invalid capability\n */\nhelpers.validateDesiredCaps = function validateDesiredCaps (caps) {\n  if (caps.browserName) {\n    if (caps.app) {\n      // warn if the capabilities have both `app` and `browser, although this is common with selenium grid\n      logger.warn(`The desired capabilities should generally not include both an 'app' and a 'browserName'`);\n    }\n    if (caps.appPackage) {\n      logger.errorAndThrow(`The desired should not include both of an 'appPackage' and a 'browserName'`);\n    }\n  }\n\n  if (caps.uninstallOtherPackages) {\n    try {\n      this.parseArray(caps.uninstallOtherPackages);\n    } catch (e) {\n      logger.errorAndThrow(`Could not parse \"uninstallOtherPackages\" capability: ${e.message}`);\n    }\n  }\n\n  return true;\n};\n\n/**\n * Adjust the capabilities for a browser session\n *\n * @param {Object} caps - Current capabilities object\n * !!! The object is mutated by this method call !!!\n * @returns {Object} The same possibly mutated `opts` instance.\n * No mutation is happening is the current session if\n * appPackage/appActivity caps have already been provided.\n */\nhelpers.adjustBrowserSessionCaps = function adjustBrowserSessionCaps (caps = {}) {\n  const { browserName } = caps;\n  logger.info(`The current session is considered browser-based`);\n  logger.info(`Supported browser names: ${JSON.stringify(_.keys(CHROME_BROWSER_PACKAGE_ACTIVITY))}`);\n  if (caps.appPackage || caps.appActivity) {\n    logger.info(`Not overriding appPackage/appActivity capability values for '${browserName}' ` +\n      'because some of them have been already provided');\n    return caps;\n  }\n\n  const {pkg, activity} = this.getChromePkg(browserName);\n  caps.appPackage = pkg;\n  caps.appActivity = activity;\n  logger.info(`appPackage/appActivity capabilities have been automatically set to ${pkg}/${activity} ` +\n    `for '${browserName}'`);\n  logger.info(`Consider changing the browserName to the one from the list of supported browser names ` +\n    `or provide custom appPackage/appActivity capability values if the automatically assigned ones do ` +\n    `not make sense`);\n  return caps;\n};\n\n/**\n * Checks whether the current device under test is an emulator\n *\n * @param {ADB} adb - appium-adb instance\n * @param {Object} opts - driver options mapping\n * @returns {boolean} `true` if the device is an Android emulator\n */\nhelpers.isEmulator = function isEmulator (adb, opts) {\n  const possibleNames = [opts.udid, adb?.curDeviceId];\n  return !!opts.avd || possibleNames.some((x) => EMULATOR_PATTERN.test(x));\n};\n\nhelpers.bootstrap = Bootstrap;\nhelpers.unlocker = unlocker;\n\nexport { helpers, SETTINGS_HELPER_PKG_ID, APP_STATE, prepareAvdArgs, ensureNetworkSpeed };\nexport default helpers;\n"],"mappings":"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"}
914
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"file":"lib/android-helpers.js","names":["PACKAGE_INSTALL_TIMEOUT","CHROME_BROWSER_PACKAGE_ACTIVITY","chrome","pkg","activity","chromium","chromebeta","browser","default","SETTINGS_HELPER_PKG_ID","SETTINGS_HELPER_UNLOCK_ACTIVITY","EMULATOR_PATTERN","APP_STATE","Object","freeze","NOT_INSTALLED","NOT_RUNNING","RUNNING_IN_BACKGROUND","RUNNING_IN_FOREGROUND","ensureNetworkSpeed","adb","networkSpeed","_","values","NETWORK_SPEED","includes","logger","warn","FULL","prepareAvdArgs","opts","isHeadless","avdArgs","result","isArray","push","util","shellParse","toCredentialType","unlockType","PIN_UNLOCK","PIN_UNLOCK_KEY_EVENT","PASSWORD_UNLOCK","PATTERN_UNLOCK","Error","helpers","createBaseADB","adbPort","suppressKillServer","remoteAdbHost","clearDeviceLogsOnStart","adbExecTimeout","useKeystore","keystorePath","keystorePassword","keyAlias","keyPassword","remoteAppsCacheLimit","buildToolsVersion","allowOfflineDevices","allowDelayAdb","ADB","createADB","prepareEmulator","avd","avdEnv","env","language","locale","country","avdLaunchTimeout","launchTimeout","avdReadyTimeout","readyTimeout","avdName","replace","runningAVD","getRunningAVD","args","debug","killEmulator","launchAVD","ensureDeviceLocale","script","isString","setDeviceLanguageCountry","ensureCurrentLocale","message","getDeviceInfoFromCaps","udid","emPort","curDeviceId","emulatorPort","info","devices","getDevicesWithRetry","map","errorAndThrow","getPortFromEmulatorString","platformVersion","trim","semver","coerce","availDevices","partialMatchCandidate","device","setDeviceId","rawDeviceOS","getPlatformVersion","deviceOS","bothVersionsCanBeCoerced","valid","bothVersionsAreStrings","version","toLower","major","minor","gt","keys","join","setEmulatorPort","validatePackageActivityNames","key","name","match","exec","index","substring","getLaunchInfo","app","appPackage","appActivity","appWaitPackage","appWaitActivity","apkPackage","apkActivity","packageAndLaunchActivityFromManifest","resetApp","fastReset","fullReset","androidInstallTimeout","autoGrantPermissions","allowTestPackages","isInstalled","isAppInstalled","forceStop","ign","output","clear","toLowerCase","grantAllPermissions","error","uninstallApk","install","grantPermissions","timeout","installApk","enforceAppInstall","appState","wasUninstalled","installOrUpgrade","enforceCurrentBuild","isInstalledOverExistingApp","APP_INSTALL_STATE","installOtherApks","otherApps","B","all","otherApp","uninstallOtherPackages","appPackages","filterPackages","getThirdPartyPackages","packagesString","shell","appPackagesArray","split","EOL","difference","err","initUnicodeKeyboard","defaultIME","appiumIME","enableIME","setIME","setMockLocationApp","getApiLevel","installHelperApp","apkPath","packageId","retry","retryInstallHelperApp","pushSettingsApp","throwError","settingsApkPath","processExists","apiLevel","perms","x","requireRunningSettingsApp","pushStrings","remoteDir","stringsJson","remoteFile","path","posix","resolve","rimraf","pullApk","tmpDir","isEmpty","fs","exists","stringsTmpDir","apkStrings","localPath","extractStringsFromApk","unlockWithHelperApp","firstRun","launchHelper","isScreenLocked","e","delay","unlock","driver","capabilities","unlockKey","unlockStrategy","unlockSuccessTimeout","unlocker","validateUnlockCapabilities","FINGERPRINT_UNLOCK","isNil","isLockManagementSupported","fastUnlock","credential","credentialType","unlockMethod","pinUnlock","pinUnlockWithKeyEvent","passwordUnlock","patternUnlock","fingerprintUnlock","verifyUnlock","timeoutMs","waitForCondition","waitMs","intervalMs","initDevice","skipDeviceInitialization","localeScript","unicodeKeyboard","disableWindowAnimation","skipUnlock","mockLocationApp","skipLogcatCapture","logcatFormat","logcatFilterSpecs","waitForDevice","shouldThrowError","isEmulator","isUndefined","startLogcat","format","filterSpecs","removeNullProperties","obj","isNull","truncateDecimals","number","digits","multiplier","Math","pow","adjustedNum","truncatedNum","isChromeBrowser","getChromePkg","removeAllSessionWebSocketHandlers","server","sessionId","isFunction","getWebSocketHandlers","activeHandlers","pathname","removeWebSocketHandler","parseArray","cap","parsedCaps","JSON","parse","validateDesiredCaps","caps","browserName","adjustBrowserSessionCaps","stringify","possibleNames","some","test","bootstrap","Bootstrap"],"sourceRoot":"../..","sources":["lib/android-helpers.js"],"sourcesContent":["import _ from 'lodash';\nimport path from 'path';\nimport { retry, waitForCondition } from 'asyncbox';\nimport logger from './logger';\nimport { fs, util } from 'appium/support';\nimport { path as settingsApkPath } from 'io.appium.settings';\nimport Bootstrap from './bootstrap';\nimport B from 'bluebird';\nimport ADB from 'appium-adb';\nimport {\n  default as unlocker, PIN_UNLOCK, PIN_UNLOCK_KEY_EVENT,\n  PASSWORD_UNLOCK, PATTERN_UNLOCK, FINGERPRINT_UNLOCK\n} from './unlock-helpers';\nimport { EOL } from 'os';\nimport semver from 'semver';\n\nconst PACKAGE_INSTALL_TIMEOUT = 90000; // milliseconds\n// https://cs.chromium.org/chromium/src/chrome/browser/devtools/device/android_device_info_query.cc\nconst CHROME_BROWSER_PACKAGE_ACTIVITY = {\n  chrome: {\n    pkg: 'com.android.chrome',\n    activity: 'com.google.android.apps.chrome.Main',\n  },\n  chromium: {\n    pkg: 'org.chromium.chrome.shell',\n    activity: '.ChromeShellActivity',\n  },\n  chromebeta: {\n    pkg: 'com.chrome.beta',\n    activity: 'com.google.android.apps.chrome.Main',\n  },\n  browser: {\n    pkg: 'com.android.browser',\n    activity: 'com.android.browser.BrowserActivity',\n  },\n  'chromium-browser': {\n    pkg: 'org.chromium.chrome',\n    activity: 'com.google.android.apps.chrome.Main',\n  },\n  'chromium-webview': {\n    pkg: 'org.chromium.webview_shell',\n    activity: 'org.chromium.webview_shell.WebViewBrowserActivity',\n  },\n  default: {\n    pkg: 'com.android.chrome',\n    activity: 'com.google.android.apps.chrome.Main',\n  },\n};\nconst SETTINGS_HELPER_PKG_ID = 'io.appium.settings';\nconst SETTINGS_HELPER_UNLOCK_ACTIVITY = '.Unlock';\nconst EMULATOR_PATTERN = /\\bemulator\\b/i;\n// These constants are in sync with\n// https://developer.apple.com/documentation/xctest/xcuiapplicationstate/xcuiapplicationstaterunningbackground?language=objc\nconst APP_STATE = Object.freeze({\n  NOT_INSTALLED: 0,\n  NOT_RUNNING: 1,\n  RUNNING_IN_BACKGROUND: 3,\n  RUNNING_IN_FOREGROUND: 4\n});\n\n\nfunction ensureNetworkSpeed (adb, networkSpeed) {\n  if (_.values(adb.NETWORK_SPEED).includes(networkSpeed)) {\n    return networkSpeed;\n  }\n  logger.warn(`Wrong network speed param '${networkSpeed}', using default: ${adb.NETWORK_SPEED.FULL}. ` +\n    `Supported values: ${_.values(adb.NETWORK_SPEED)}`);\n  return adb.NETWORK_SPEED.FULL;\n}\n\nfunction prepareAvdArgs (adb, opts) {\n  const {\n    networkSpeed,\n    isHeadless,\n    avdArgs,\n  } = opts;\n  const result = [];\n  if (avdArgs) {\n    if (_.isArray(avdArgs)) {\n      result.push(...avdArgs);\n    } else {\n      result.push(...(util.shellParse(`${avdArgs}`)));\n    }\n  }\n  if (networkSpeed) {\n    result.push('-netspeed', ensureNetworkSpeed(adb, networkSpeed));\n  }\n  if (isHeadless) {\n    result.push('-no-window');\n  }\n  return result;\n}\n\nfunction toCredentialType (unlockType) {\n  const result = {\n    [PIN_UNLOCK]: 'pin',\n    [PIN_UNLOCK_KEY_EVENT]: 'pin',\n    [PASSWORD_UNLOCK]: 'password',\n    [PATTERN_UNLOCK]: 'pattern',\n  }[unlockType];\n  if (result) {\n    return result;\n  }\n  throw new Error(`Unlock type '${unlockType}' is not known`);\n}\n\n\nconst helpers = {};\n\nhelpers.createBaseADB = async function createBaseADB (opts = {}) {\n  // filter out any unwanted options sent in\n  // this list should be updated as ADB takes more arguments\n  const {\n    adbPort,\n    suppressKillServer,\n    remoteAdbHost,\n    clearDeviceLogsOnStart,\n    adbExecTimeout,\n    useKeystore,\n    keystorePath,\n    keystorePassword,\n    keyAlias,\n    keyPassword,\n    remoteAppsCacheLimit,\n    buildToolsVersion,\n    allowOfflineDevices,\n    allowDelayAdb,\n  } = opts;\n  return await ADB.createADB({\n    adbPort,\n    suppressKillServer,\n    remoteAdbHost,\n    clearDeviceLogsOnStart,\n    adbExecTimeout,\n    useKeystore,\n    keystorePath,\n    keystorePassword,\n    keyAlias,\n    keyPassword,\n    remoteAppsCacheLimit,\n    buildToolsVersion,\n    allowOfflineDevices,\n    allowDelayAdb,\n  });\n};\n\nhelpers.prepareEmulator = async function prepareEmulator (adb, opts) {\n  const {\n    avd,\n    avdEnv: env,\n    language,\n    locale: country,\n    avdLaunchTimeout: launchTimeout,\n    avdReadyTimeout: readyTimeout,\n  } = opts;\n  if (!avd) {\n    throw new Error('Cannot launch AVD without AVD name');\n  }\n\n  const avdName = avd.replace('@', '');\n  const runningAVD = await adb.getRunningAVD(avdName);\n  const args = prepareAvdArgs(adb, opts);\n  if (runningAVD) {\n    if (args.includes('-wipe-data')) {\n      logger.debug(`Killing '${avdName}' because it needs to be wiped at start.`);\n      await adb.killEmulator(avdName);\n    } else {\n      logger.debug('Not launching AVD because it is already running.');\n      return;\n    }\n  }\n  await adb.launchAVD(avd, {\n    args,\n    env,\n    language,\n    country,\n    launchTimeout,\n    readyTimeout,\n  });\n};\n\n/**\n * Set and ensure the locale name of the device under test.\n *\n * @param {Object} adb - The adb module instance.\n * @param {string} language - Language. The language field is case insensitive, but Locale always canonicalizes to lower case.\n *                            format: [a-zA-Z]{2,8}. e.g. en, ja : https://developer.android.com/reference/java/util/Locale.html\n * @param {string} country - Country. The country (region) field is case insensitive, but Locale always canonicalizes to upper case.\n *                            format: [a-zA-Z]{2} | [0-9]{3}. e.g. US, JP : https://developer.android.com/reference/java/util/Locale.html\n * @param {?string} script - Script. The script field is case insensitive but Locale always canonicalizes to title case.\n *                            format: [a-zA-Z]{4}. e.g. Hans in zh-Hans-CN : https://developer.android.com/reference/java/util/Locale.html\n * @throws {Error} If it failed to set locale properly\n */\nhelpers.ensureDeviceLocale = async function ensureDeviceLocale (adb, language, country, script = null) {\n  if (!_.isString(language) && !_.isString(country)) {\n    logger.warn(`setDeviceLanguageCountry requires language or country.`);\n    logger.warn(`Got language: '${language}' and country: '${country}'`);\n    return;\n  }\n\n  await adb.setDeviceLanguageCountry(language, country, script);\n\n  if (!await adb.ensureCurrentLocale(language, country, script)) {\n    const message = script ? `language: ${language}, country: ${country} and script: ${script}` : `language: ${language} and country: ${country}`;\n    throw new Error(`Failed to set ${message}`);\n  }\n};\n\nhelpers.getDeviceInfoFromCaps = async function getDeviceInfoFromCaps (opts = {}) {\n  // we can create a throwaway ADB instance here, so there is no dependency\n  // on instantiating on earlier (at this point, we have no udid)\n  // we can only use this ADB object for commands that would not be confused\n  // if multiple devices are connected\n  const adb = await helpers.createBaseADB(opts);\n  let udid = opts.udid;\n  let emPort = null;\n\n  // a specific avd name was given. try to initialize with that\n  if (opts.avd) {\n    await helpers.prepareEmulator(adb, opts);\n    udid = adb.curDeviceId;\n    emPort = adb.emulatorPort;\n  } else {\n    // no avd given. lets try whatever's plugged in devices/emulators\n    logger.info('Retrieving device list');\n    let devices = await adb.getDevicesWithRetry();\n\n    // udid was given, lets try to init with that device\n    if (udid) {\n      if (!_.includes(_.map(devices, 'udid'), udid)) {\n        logger.errorAndThrow(`Device ${udid} was not in the list of connected devices`);\n      }\n      emPort = adb.getPortFromEmulatorString(udid);\n    } else if (opts.platformVersion) {\n      opts.platformVersion = `${opts.platformVersion}`.trim();\n\n      // a platform version was given. lets try to find a device with the same os\n      const platformVersion = semver.coerce(opts.platformVersion) || opts.platformVersion;\n      logger.info(`Looking for a device with Android '${platformVersion}'`);\n\n      // in case we fail to find something, give the user a useful log that has\n      // the device udids and os versions so they know what's available\n      const availDevices = [];\n      let partialMatchCandidate = null;\n      // first try started devices/emulators\n      for (const device of devices) {\n        // direct adb calls to the specific device\n        await adb.setDeviceId(device.udid);\n        const rawDeviceOS = await adb.getPlatformVersion();\n        // The device OS could either be a number, like `6.0`\n        // or an abbreviation, like `R`\n        availDevices.push(`${device.udid} (${rawDeviceOS})`);\n        const deviceOS = semver.coerce(rawDeviceOS) || rawDeviceOS;\n        if (!deviceOS) {\n          continue;\n        }\n\n        const bothVersionsCanBeCoerced = semver.valid(deviceOS) && semver.valid(platformVersion);\n        const bothVersionsAreStrings = _.isString(deviceOS) && _.isString(platformVersion);\n        if (bothVersionsCanBeCoerced && deviceOS.version === platformVersion.version\n            || bothVersionsAreStrings && _.toLower(deviceOS) === _.toLower(platformVersion)) {\n          // Got an exact match - proceed immediately\n          udid = device.udid;\n          break;\n        } else if (!bothVersionsCanBeCoerced) {\n          // There is no point to check for partial match if either of version numbers is not coercible\n          continue;\n        }\n\n        if ((!_.includes(opts.platformVersion, '.') && platformVersion.major === deviceOS.major\n            || platformVersion.major === deviceOS.major && platformVersion.minor === deviceOS.minor)\n            // Got a partial match - make sure we consider the most recent\n            // device version available on the host system\n            && (partialMatchCandidate && semver.gt(deviceOS, _.values(partialMatchCandidate)[0])\n                || !partialMatchCandidate)) {\n          partialMatchCandidate = {[device.udid]: deviceOS};\n        }\n      }\n      if (!udid && partialMatchCandidate) {\n        udid = _.keys(partialMatchCandidate)[0];\n        await adb.setDeviceId(udid);\n      }\n\n      if (!udid) {\n        // we couldn't find anything! quit\n        logger.errorAndThrow(`Unable to find an active device or emulator ` +\n          `with OS ${opts.platformVersion}. The following are available: ` +\n          availDevices.join(', '));\n      }\n\n      emPort = adb.getPortFromEmulatorString(udid);\n    } else {\n      // a udid was not given, grab the first device we see\n      udid = devices[0].udid;\n      emPort = adb.getPortFromEmulatorString(udid);\n    }\n  }\n\n  logger.info(`Using device: ${udid}`);\n  return {udid, emPort};\n};\n\n// returns a new adb instance with deviceId set\nhelpers.createADB = async function createADB (opts = {}) {\n  const {udid, emPort} = opts;\n  const adb = await helpers.createBaseADB(opts);\n  adb.setDeviceId(udid);\n  if (emPort) {\n    adb.setEmulatorPort(emPort);\n  }\n\n  return adb;\n};\n\nhelpers.validatePackageActivityNames = function validatePackageActivityNames (opts) {\n  for (const key of ['appPackage', 'appActivity', 'appWaitPackage', 'appWaitActivity']) {\n    const name = opts[key];\n    if (!name) {\n      continue;\n    }\n\n    const match = /([^\\w.*,])+/.exec(name);\n    if (!match) {\n      continue;\n    }\n\n    logger.warn(`Capability '${key}' is expected to only include latin letters, digits, underscore, dot, comma and asterisk characters.`);\n    logger.warn(`Current value '${name}' has non-matching character at index ${match.index}: '${name.substring(0, match.index + 1)}'`);\n  }\n};\n\nhelpers.getLaunchInfo = async function getLaunchInfo (adb, opts) {\n  let {app, appPackage, appActivity, appWaitPackage, appWaitActivity} = opts;\n  if (!app) {\n    logger.warn('No app sent in, not parsing package/activity');\n    return;\n  }\n\n  this.validatePackageActivityNames(opts);\n\n  if (appPackage && appActivity) {\n    return;\n  }\n\n  logger.debug('Parsing package and activity from app manifest');\n  let {apkPackage, apkActivity} =\n    await adb.packageAndLaunchActivityFromManifest(app);\n  if (apkPackage && !appPackage) {\n    appPackage = apkPackage;\n  }\n  if (!appWaitPackage) {\n    appWaitPackage = appPackage;\n  }\n  if (apkActivity && !appActivity) {\n    appActivity = apkActivity;\n  }\n  if (!appWaitActivity) {\n    appWaitActivity = appActivity;\n  }\n  logger.debug(`Parsed package and activity are: ${apkPackage}/${apkActivity}`);\n  return {appPackage, appWaitPackage, appActivity, appWaitActivity};\n};\n\nhelpers.resetApp = async function resetApp (adb, opts = {}) {\n  const {\n    app,\n    appPackage,\n    fastReset,\n    fullReset,\n    androidInstallTimeout = PACKAGE_INSTALL_TIMEOUT,\n    autoGrantPermissions,\n    allowTestPackages\n  } = opts;\n\n  if (!appPackage) {\n    throw new Error(\"'appPackage' option is required\");\n  }\n\n  const isInstalled = await adb.isAppInstalled(appPackage);\n\n  if (isInstalled) {\n    try {\n      await adb.forceStop(appPackage);\n    } catch (ign) {}\n    // fullReset has priority over fastReset\n    if (!fullReset && fastReset) {\n      const output = await adb.clear(appPackage);\n      if (_.isString(output) && output.toLowerCase().includes('failed')) {\n        throw new Error(`Cannot clear the application data of '${appPackage}'. Original error: ${output}`);\n      }\n      // executing `shell pm clear` resets previously assigned application permissions as well\n      if (autoGrantPermissions) {\n        try {\n          await adb.grantAllPermissions(appPackage);\n        } catch (error) {\n          logger.error(`Unable to grant permissions requested. Original error: ${error.message}`);\n        }\n      }\n      logger.debug(`Performed fast reset on the installed '${appPackage}' application (stop and clear)`);\n      return;\n    }\n  }\n\n  if (!app) {\n    throw new Error(\"'app' option is required for reinstall\");\n  }\n\n  logger.debug(`Running full reset on '${appPackage}' (reinstall)`);\n  if (isInstalled) {\n    await adb.uninstallApk(appPackage);\n  }\n  await adb.install(app, {\n    grantPermissions: autoGrantPermissions,\n    timeout: androidInstallTimeout,\n    allowTestPackages,\n  });\n};\n\nhelpers.installApk = async function installApk (adb, opts = {}) {\n  const {\n    app,\n    appPackage,\n    fastReset,\n    fullReset,\n    androidInstallTimeout = PACKAGE_INSTALL_TIMEOUT,\n    autoGrantPermissions,\n    allowTestPackages,\n    enforceAppInstall,\n  } = opts;\n\n  if (!app || !appPackage) {\n    throw new Error(\"'app' and 'appPackage' options are required\");\n  }\n\n  if (fullReset) {\n    await this.resetApp(adb, opts);\n    return;\n  }\n\n  const {\n    appState,\n    wasUninstalled\n  } = await adb.installOrUpgrade(app, appPackage, {\n    grantPermissions: autoGrantPermissions,\n    timeout: androidInstallTimeout,\n    allowTestPackages,\n    enforceCurrentBuild: enforceAppInstall,\n  });\n\n  // There is no need to reset the newly installed app\n  const isInstalledOverExistingApp = !wasUninstalled\n    && appState !== adb.APP_INSTALL_STATE.NOT_INSTALLED;\n  if (fastReset && isInstalledOverExistingApp) {\n    logger.info(`Performing fast reset on '${appPackage}'`);\n    await this.resetApp(adb, opts);\n  }\n};\n\n/**\n * Installs an array of apks\n * @param {ADB} adb Instance of Appium ADB object\n * @param {Object} opts Opts defined in driver.js\n */\nhelpers.installOtherApks = async function installOtherApks (otherApps, adb, opts) {\n  let {\n    androidInstallTimeout = PACKAGE_INSTALL_TIMEOUT,\n    autoGrantPermissions,\n    allowTestPackages\n  } = opts;\n\n  // Install all of the APK's asynchronously\n  await B.all(otherApps.map((otherApp) => {\n    logger.debug(`Installing app: ${otherApp}`);\n    return adb.installOrUpgrade(otherApp, null, {\n      grantPermissions: autoGrantPermissions,\n      timeout: androidInstallTimeout,\n      allowTestPackages,\n    });\n  }));\n};\n\n/**\n * Uninstall an array of packages\n * @param {ADB} adb Instance of Appium ADB object\n * @param {Array<string>} appPackages An array of package names to uninstall. If this includes `'*'`, uninstall all of 3rd party apps\n * @param {Array<string>} filterPackages An array of packages does not uninstall when `*` is provided as `appPackages`\n */\nhelpers.uninstallOtherPackages = async function uninstallOtherPackages (adb, appPackages, filterPackages = []) {\n  if (appPackages.includes('*')) {\n    logger.debug('Uninstall third party packages');\n    appPackages = await this.getThirdPartyPackages(adb, filterPackages);\n  }\n\n  logger.debug(`Uninstalling packages: ${appPackages}`);\n  await B.all(appPackages.map((appPackage) => adb.uninstallApk(appPackage)));\n};\n\n/**\n * Get third party packages filtered with `filterPackages`\n * @param {ADB} adb Instance of Appium ADB object\n * @param {Array<string>} filterPackages An array of packages does not uninstall when `*` is provided as `appPackages`\n * @returns {Array<string>} An array of installed third pary packages\n */\nhelpers.getThirdPartyPackages = async function getThirdPartyPackages (adb, filterPackages = []) {\n  try {\n    const packagesString = await adb.shell(['pm', 'list', 'packages', '-3']);\n    const appPackagesArray = packagesString.trim().replace(/package:/g, '').split(EOL);\n    logger.debug(`'${appPackagesArray}' filtered with '${filterPackages}'`);\n    return _.difference(appPackagesArray, filterPackages);\n  } catch (err) {\n    logger.warn(`Unable to get packages with 'adb shell pm list packages -3': ${err.message}`);\n    return [];\n  }\n};\n\nhelpers.initUnicodeKeyboard = async function initUnicodeKeyboard (adb) {\n  logger.debug('Enabling Unicode keyboard support');\n\n  // get the default IME so we can return back to it later if we want\n  let defaultIME = await adb.defaultIME();\n\n  logger.debug(`Unsetting previous IME ${defaultIME}`);\n  const appiumIME = `${SETTINGS_HELPER_PKG_ID}/.UnicodeIME`;\n  logger.debug(`Setting IME to '${appiumIME}'`);\n  await adb.enableIME(appiumIME);\n  await adb.setIME(appiumIME);\n  return defaultIME;\n};\n\nhelpers.setMockLocationApp = async function setMockLocationApp (adb, app) {\n  try {\n    if (await adb.getApiLevel() < 23) {\n      await adb.shell(['settings', 'put', 'secure', 'mock_location', '1']);\n    } else {\n      await adb.shell(['appops', 'set', app, 'android:mock_location', 'allow']);\n    }\n  } catch (err) {\n    logger.warn(`Unable to set mock location for app '${app}': ${err.message}`);\n  }\n};\n\nhelpers.installHelperApp = async function installHelperApp (adb, apkPath, packageId) {\n  // Sometimes adb push or adb instal take more time than expected to install an app\n  // e.g. https://github.com/appium/io.appium.settings/issues/40#issuecomment-476593174\n  await retry(2, async function retryInstallHelperApp () {\n    await adb.installOrUpgrade(apkPath, packageId, {grantPermissions: true});\n  });\n};\n\n/**\n * Pushes and installs io.appium.settings app.\n * Throws an error if the setting app is required\n *\n * @param {Adb} adb - The adb module instance.\n * @param {boolean} throwError[false] - Whether throw error or not\n * @throws {Error} If throwError is true and something happens in installation step\n */\nhelpers.pushSettingsApp = async function pushSettingsApp (adb, throwError = false) {\n  logger.debug('Pushing settings apk to device...');\n\n  try {\n    await helpers.installHelperApp(adb, settingsApkPath, SETTINGS_HELPER_PKG_ID, throwError);\n  } catch (err) {\n    if (throwError) {\n      throw err;\n    }\n\n    logger.warn(`Ignored error while installing '${settingsApkPath}': ` +\n                `'${err.message}'. Features that rely on this helper ` +\n                'require the apk such as toggle WiFi and getting location ' +\n                'will raise an error if you try to use them.');\n  }\n\n  // Reinstall would stop the settings helper process anyway, so\n  // there is no need to continue if the application is still running\n  if (await adb.processExists(SETTINGS_HELPER_PKG_ID)) {\n    logger.debug(`${SETTINGS_HELPER_PKG_ID} is already running. ` +\n      `There is no need to reset its permissions.`);\n    return;\n  }\n\n  const apiLevel = await adb.getApiLevel();\n  if (apiLevel >= 29) {\n    // https://github.com/appium/io.appium.settings#internal-audio--video-recording\n    try {\n      await adb.shell(['appops', 'set', SETTINGS_HELPER_PKG_ID, 'PROJECT_MEDIA', 'allow']);\n    } catch (err) {\n      logger.debug(err);\n    }\n  }\n  if (apiLevel <= 23) { // Android 6- devices should have granted permissions\n    // https://github.com/appium/appium/pull/11640#issuecomment-438260477\n    const perms = ['SET_ANIMATION_SCALE', 'CHANGE_CONFIGURATION', 'ACCESS_FINE_LOCATION'];\n    logger.info(`Granting permissions ${perms} to '${SETTINGS_HELPER_PKG_ID}'`);\n    await adb.grantPermissions(SETTINGS_HELPER_PKG_ID, perms.map((x) => `android.permission.${x}`));\n  }\n\n  // launch io.appium.settings app due to settings failing to be set\n  // if the app is not launched prior to start the session on android 7+\n  // see https://github.com/appium/appium/issues/8957\n  try {\n    await adb.requireRunningSettingsApp();\n  } catch (err) {\n    logger.debug(err);\n    if (throwError) {\n      throw err;\n    }\n  }\n};\n\n/**\n * Extracts string.xml and converts it to string.json and pushes\n * it to /data/local/tmp/string.json on for use of bootstrap\n * If app is not present to extract string.xml it deletes remote strings.json\n * If app does not have strings.xml we push an empty json object to remote\n *\n * @param {?string} language - Language abbreviation, for example 'fr'. The default language\n * is used if this argument is not defined.\n * @param {Object} adb - The adb module instance.\n * @param {Object} opts - Driver options dictionary.\n * @returns {Object} The dictionary, where string resource identifiers are keys\n * along with their corresponding values for the given language or an empty object\n * if no matching resources were extracted.\n */\nhelpers.pushStrings = async function pushStrings (language, adb, opts) {\n  const remoteDir = '/data/local/tmp';\n  const stringsJson = 'strings.json';\n  const remoteFile = path.posix.resolve(remoteDir, stringsJson);\n\n  // clean up remote string.json if present\n  await adb.rimraf(remoteFile);\n\n  let app;\n  try {\n    app = opts.app || await adb.pullApk(opts.appPackage, opts.tmpDir);\n  } catch (err) {\n    logger.info(`Failed to pull an apk from '${opts.appPackage}' to '${opts.tmpDir}'. Original error: ${err.message}`);\n  }\n\n  if (_.isEmpty(opts.appPackage) || !(await fs.exists(app))) {\n    logger.debug(`No app or package specified. Returning empty strings`);\n    return {};\n  }\n\n  const stringsTmpDir = path.resolve(opts.tmpDir, opts.appPackage);\n  try {\n    logger.debug('Extracting strings from apk', app, language, stringsTmpDir);\n    const {apkStrings, localPath} = await adb.extractStringsFromApk(app, language, stringsTmpDir);\n    await adb.push(localPath, remoteDir);\n    return apkStrings;\n  } catch (err) {\n    logger.warn(`Could not get strings, continuing anyway. Original error: ${err.message}`);\n    await adb.shell('echo', [`'{}' > ${remoteFile}`]);\n  } finally {\n    await fs.rimraf(stringsTmpDir);\n  }\n  return {};\n};\n\nhelpers.unlockWithHelperApp = async function unlockWithHelperApp (adb) {\n  logger.info('Unlocking screen');\n\n  // Unlock succeed with a couple of retries.\n  let firstRun = true;\n  await retry(3, async function launchHelper () {\n    // To reduce a time to call adb.isScreenLocked() since `adb shell dumpsys window` is easy to hang adb commands\n    if (firstRun) {\n      firstRun = false;\n    } else {\n      try {\n        if (!(await adb.isScreenLocked())) {\n          return;\n        }\n      } catch (e) {\n        logger.warn(`Error in isScreenLocked: ${e.message}`);\n        logger.warn('\"adb shell dumpsys window\" command has timed out.');\n        logger.warn('The reason of this timeout is the delayed adb response. Resetting adb server can improve it.');\n      }\n    }\n\n    logger.info(`Launching ${SETTINGS_HELPER_UNLOCK_ACTIVITY}`);\n    await adb.shell([\n      'am', 'start',\n      '-n', `${SETTINGS_HELPER_PKG_ID}/${SETTINGS_HELPER_UNLOCK_ACTIVITY}`,\n      '-c', 'android.intent.category.LAUNCHER',\n      '-a', 'android.intent.action.MAIN',\n      '-f', '0x10200000',\n    ]);\n    await B.delay(1000);\n  });\n};\n\nhelpers.unlock = async function unlock (driver, adb, capabilities) {\n  if (!(await adb.isScreenLocked())) {\n    logger.info('Screen already unlocked, doing nothing');\n    return;\n  }\n  logger.debug('Screen is locked, trying to unlock');\n\n  if (!capabilities.unlockType && !capabilities.unlockKey) {\n    logger.warn('Using app unlock, this is going to be deprecated!');\n    await helpers.unlockWithHelperApp(adb);\n    return;\n  }\n\n  const {\n    unlockType,\n    unlockKey,\n    unlockStrategy,\n    unlockSuccessTimeout,\n  } = unlocker.validateUnlockCapabilities(capabilities);\n  if (unlockKey && unlockType !== FINGERPRINT_UNLOCK\n      && (_.isNil(unlockStrategy) || _.toLower(unlockStrategy) === 'locksettings')\n      && await adb.isLockManagementSupported()) {\n    await unlocker.fastUnlock(adb, {\n      credential: unlockKey,\n      credentialType: toCredentialType(unlockType),\n    });\n  } else {\n    const unlockMethod = {\n      [PIN_UNLOCK]: unlocker.pinUnlock,\n      [PIN_UNLOCK_KEY_EVENT]: unlocker.pinUnlockWithKeyEvent,\n      [PASSWORD_UNLOCK]: unlocker.passwordUnlock,\n      [PATTERN_UNLOCK]: unlocker.patternUnlock,\n      [FINGERPRINT_UNLOCK]: unlocker.fingerprintUnlock,\n    }[unlockType];\n    await unlockMethod(adb, driver, capabilities);\n  }\n  await helpers.verifyUnlock(adb, unlockSuccessTimeout);\n};\n\nhelpers.verifyUnlock = async function verifyUnlock (adb, timeoutMs = null) {\n  try {\n    await waitForCondition(async () => !(await adb.isScreenLocked()), {\n      waitMs: timeoutMs ?? 2000,\n      intervalMs: 500,\n    });\n  } catch (ign) {\n    throw new Error('The device has failed to be unlocked');\n  }\n  logger.info('The device has been successfully unlocked');\n};\n\nhelpers.initDevice = async function initDevice (adb, opts) {\n  const {\n    skipDeviceInitialization,\n    locale,\n    language,\n    localeScript,\n    unicodeKeyboard,\n    disableWindowAnimation,\n    skipUnlock,\n    mockLocationApp,\n    skipLogcatCapture,\n    logcatFormat,\n    logcatFilterSpecs,\n  } = opts;\n\n  if (skipDeviceInitialization) {\n    logger.info(`'skipDeviceInitialization' is set. Skipping device initialization.`);\n  } else {\n    await adb.waitForDevice();\n    // pushSettingsApp required before calling ensureDeviceLocale for API Level 24+\n\n    // Some feature such as location/wifi are not necessary for all users,\n    // but they require the settings app. So, try to configure it while Appium\n    // does not throw error even if they fail.\n    const shouldThrowError = language\n      || locale\n      || localeScript\n      || unicodeKeyboard\n      || disableWindowAnimation\n      || !skipUnlock;\n    await helpers.pushSettingsApp(adb, shouldThrowError);\n  }\n\n  if (!helpers.isEmulator(adb, opts) && (mockLocationApp || _.isUndefined(mockLocationApp))) {\n    await helpers.setMockLocationApp(adb, mockLocationApp || SETTINGS_HELPER_PKG_ID);\n  }\n\n  if (language || locale) {\n    await helpers.ensureDeviceLocale(adb, language, locale, localeScript);\n  }\n\n  if (skipLogcatCapture) {\n    logger.info(`'skipLogcatCapture' is set. Skipping starting logcat capture.`);\n  } else {\n    await adb.startLogcat({\n      format: logcatFormat,\n      filterSpecs: logcatFilterSpecs,\n    });\n  }\n\n  if (unicodeKeyboard) {\n    return await helpers.initUnicodeKeyboard(adb);\n  }\n};\n\nhelpers.removeNullProperties = function removeNullProperties (obj) {\n  for (let key of _.keys(obj)) {\n    if (_.isNull(obj[key]) || _.isUndefined(obj[key])) {\n      delete obj[key];\n    }\n  }\n};\n\nhelpers.truncateDecimals = function truncateDecimals (number, digits) {\n  let multiplier = Math.pow(10, digits),\n      adjustedNum = number * multiplier,\n      truncatedNum = Math[adjustedNum < 0 ? 'ceil' : 'floor'](adjustedNum);\n\n  return truncatedNum / multiplier;\n};\n\nhelpers.isChromeBrowser = function isChromeBrowser (browser) {\n  return _.includes(Object.keys(CHROME_BROWSER_PACKAGE_ACTIVITY), (browser || '').toLowerCase());\n};\n\nhelpers.getChromePkg = function getChromePkg (browser) {\n  return CHROME_BROWSER_PACKAGE_ACTIVITY[browser.toLowerCase()] || CHROME_BROWSER_PACKAGE_ACTIVITY.default;\n};\n\nhelpers.removeAllSessionWebSocketHandlers = async function removeAllSessionWebSocketHandlers (server, sessionId) {\n  if (!server || !_.isFunction(server.getWebSocketHandlers)) {\n    return;\n  }\n\n  const activeHandlers = await server.getWebSocketHandlers(sessionId);\n  for (const pathname of _.keys(activeHandlers)) {\n    await server.removeWebSocketHandler(pathname);\n  }\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 {any} cap A desired capability\n */\nhelpers.parseArray = function parseArray (cap) {\n  let parsedCaps;\n  try {\n    parsedCaps = JSON.parse(cap);\n  } catch (ign) { }\n\n  if (_.isArray(parsedCaps)) {\n    return parsedCaps;\n  } else if (_.isString(cap)) {\n    return [cap];\n  }\n\n  throw new Error(`must provide a string or JSON Array; received ${cap}`);\n};\n\n/**\n * Validate desired capabilities. Returns true if capability is valid\n *\n * @param {*} cap A desired capability\n * @return {boolean} Returns true if the capability is valid\n * @throws {Error} If the caps has invalid capability\n */\nhelpers.validateDesiredCaps = function validateDesiredCaps (caps) {\n  if (caps.browserName) {\n    if (caps.app) {\n      // warn if the capabilities have both `app` and `browser, although this is common with selenium grid\n      logger.warn(`The desired capabilities should generally not include both an 'app' and a 'browserName'`);\n    }\n    if (caps.appPackage) {\n      logger.errorAndThrow(`The desired should not include both of an 'appPackage' and a 'browserName'`);\n    }\n  }\n\n  if (caps.uninstallOtherPackages) {\n    try {\n      this.parseArray(caps.uninstallOtherPackages);\n    } catch (e) {\n      logger.errorAndThrow(`Could not parse \"uninstallOtherPackages\" capability: ${e.message}`);\n    }\n  }\n\n  return true;\n};\n\n/**\n * Adjust the capabilities for a browser session\n *\n * @param {Object} caps - Current capabilities object\n * !!! The object is mutated by this method call !!!\n * @returns {Object} The same possibly mutated `opts` instance.\n * No mutation is happening is the current session if\n * appPackage/appActivity caps have already been provided.\n */\nhelpers.adjustBrowserSessionCaps = function adjustBrowserSessionCaps (caps = {}) {\n  const { browserName } = caps;\n  logger.info(`The current session is considered browser-based`);\n  logger.info(`Supported browser names: ${JSON.stringify(_.keys(CHROME_BROWSER_PACKAGE_ACTIVITY))}`);\n  if (caps.appPackage || caps.appActivity) {\n    logger.info(`Not overriding appPackage/appActivity capability values for '${browserName}' ` +\n      'because some of them have been already provided');\n    return caps;\n  }\n\n  const {pkg, activity} = this.getChromePkg(browserName);\n  caps.appPackage = pkg;\n  caps.appActivity = activity;\n  logger.info(`appPackage/appActivity capabilities have been automatically set to ${pkg}/${activity} ` +\n    `for '${browserName}'`);\n  logger.info(`Consider changing the browserName to the one from the list of supported browser names ` +\n    `or provide custom appPackage/appActivity capability values if the automatically assigned ones do ` +\n    `not make sense`);\n  return caps;\n};\n\n/**\n * Checks whether the current device under test is an emulator\n *\n * @param {ADB} adb - appium-adb instance\n * @param {Object} opts - driver options mapping\n * @returns {boolean} `true` if the device is an Android emulator\n */\nhelpers.isEmulator = function isEmulator (adb, opts) {\n  const possibleNames = [opts.udid, adb?.curDeviceId];\n  return !!opts.avd || possibleNames.some((x) => EMULATOR_PATTERN.test(x));\n};\n\nhelpers.bootstrap = Bootstrap;\nhelpers.unlocker = unlocker;\n\nexport { helpers, SETTINGS_HELPER_PKG_ID, APP_STATE, prepareAvdArgs, ensureNetworkSpeed };\nexport default helpers;\n"],"mappings":"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"}
@@ -20,6 +20,7 @@ var _androidHelpers = require("../android-helpers");
20
20
  var _driver = require("appium/driver");
21
21
 
22
22
  const APP_EXTENSIONS = ['.apk', '.apks'];
23
+ const RESOLVER_ACTIVITY_NAME = 'android/com.android.internal.app.ResolverActivity';
23
24
  let commands = {};
24
25
  exports.commands = commands;
25
26
 
@@ -71,7 +72,16 @@ commands.activateApp = async function activateApp(appId) {
71
72
  return;
72
73
  }
73
74
 
74
- const stdout = await this.adb.shell(['am', apiLevel < 26 ? 'start' : 'start-activity', '-a', 'android.intent.action.MAIN', '-c', 'android.intent.category.LAUNCHER', '-f', '0x10200000', '-n', await this.adb.resolveLaunchableActivity(appId)]);
75
+ let activityName = await this.adb.resolveLaunchableActivity(appId);
76
+
77
+ if (activityName === RESOLVER_ACTIVITY_NAME) {
78
+ this.log.debug(`The launchable activity name of '${appId}' was resolved to '${activityName}'. ` + `Switching the resolver to not use cmd`);
79
+ activityName = await this.adb.resolveLaunchableActivity(appId, {
80
+ preferCmd: false
81
+ });
82
+ }
83
+
84
+ const stdout = await this.adb.shell(['am', apiLevel < 26 ? 'start' : 'start-activity', '-a', 'android.intent.action.MAIN', '-c', 'android.intent.category.LAUNCHER', '-f', '0x10200000', '-n', activityName]);
75
85
  this.log.debug(stdout);
76
86
 
77
87
  if (/^error:/mi.test(stdout)) {
@@ -128,4 +138,4 @@ var _default = commands;
128
138
  exports.default = _default;require('source-map-support').install();
129
139
 
130
140
 
131
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"file":"lib/commands/app-management.js","names":["APP_EXTENSIONS","commands","isAppInstalled","appId","adb","queryAppState","log","info","APP_STATE","NOT_INSTALLED","processExists","NOT_RUNNING","appIdRe","RegExp","_","escapeRegExp","line","dumpWindows","split","test","some","x","includes","RUNNING_IN_FOREGROUND","RUNNING_IN_BACKGROUND","activateApp","debug","apiLevel","getApiLevel","cmd","output","shell","e","errorAndThrow","message","stdout","resolveLaunchableActivity","Error","removeApp","options","uninstallApk","terminateApp","forceStop","timeout","util","hasValue","isNaN","parseInt","waitForCondition","waitMs","intervalMs","installApp","appPath","localPath","helpers","configureApp","install","mobileClearApp","opts","errors","InvalidArgumentError","clear"],"sourceRoot":"../../..","sources":["lib/commands/app-management.js"],"sourcesContent":["import _ from 'lodash';\nimport { waitForCondition } from 'asyncbox';\nimport { util } from 'appium/support';\nimport { APP_STATE } from '../android-helpers';\nimport { errors } from 'appium/driver';\n\nconst APP_EXTENSIONS = ['.apk', '.apks'];\n\nlet commands = {};\n\n/**\n * Verify whether an application is installed or not\n *\n * @param {string} appId - Application package identifier\n * @returns {boolean} true if the app is installed\n */\ncommands.isAppInstalled = async function isAppInstalled (appId) {\n  return await this.adb.isAppInstalled(appId);\n};\n\n/**\n * Queries the current state of the app.\n *\n * @param {string} appId - Application package identifier\n * @returns {number} The corresponding constant, which describes\n *                   the current application state:\n * 0 - is the app is not installed\n * 1 - if the app is installed, but is not running\n * 3 - if the app is running in the background\n * 4 - if the app is running in the foreground\n */\ncommands.queryAppState = async function queryAppState (appId) {\n  this.log.info(`Querying the state of '${appId}'`);\n  if (!await this.adb.isAppInstalled(appId)) {\n    return APP_STATE.NOT_INSTALLED;\n  }\n  if (!await this.adb.processExists(appId)) {\n    return APP_STATE.NOT_RUNNING;\n  }\n  const appIdRe = new RegExp(`\\\\b${_.escapeRegExp(appId)}/`);\n  for (const line of (await this.adb.dumpWindows()).split('\\n')) {\n    if (appIdRe.test(line) && ['mCurrentFocus', 'mFocusedApp'].some((x) => line.includes(x))) {\n      return APP_STATE.RUNNING_IN_FOREGROUND;\n    }\n  }\n  return APP_STATE.RUNNING_IN_BACKGROUND;\n};\n\n/**\n * Activates the given application or launches it if necessary.\n * The action is done with monkey tool and literally simulates\n * clicking the corresponding application icon on the dashboard.\n *\n * @param {string} appId - Application package identifier\n */\ncommands.activateApp = async function activateApp (appId) {\n  this.log.debug(`Activating '${appId}'`);\n  const apiLevel = await this.adb.getApiLevel();\n  // Fallback to Monkey in older APIs\n  if (apiLevel < 24) {\n    // The monkey command could raise an issue as https://stackoverflow.com/questions/44860475/how-to-use-the-monkey-command-with-an-android-system-that-doesnt-have-physical\n    // but '--pct-syskeys 0' could cause another background process issue. https://github.com/appium/appium/issues/16941#issuecomment-1129837285\n    const cmd = ['monkey',\n      '-p', appId,\n      '-c', 'android.intent.category.LAUNCHER',\n      '1'];\n    let output = '';\n    try {\n      output = await this.adb.shell(cmd);\n      this.log.debug(`Command stdout: ${output}`);\n    } catch (e) {\n      this.log.errorAndThrow(`Cannot activate '${appId}'. Original error: ${e.message}`);\n    }\n    if (output.includes('monkey aborted')) {\n      this.log.errorAndThrow(`Cannot activate '${appId}'. Are you sure it is installed?`);\n    }\n    return;\n  }\n\n  const stdout = await this.adb.shell([\n    'am', (apiLevel < 26) ? 'start' : 'start-activity',\n    '-a', 'android.intent.action.MAIN',\n    '-c', 'android.intent.category.LAUNCHER',\n    // FLAG_ACTIVITY_NEW_TASK | FLAG_ACTIVITY_RESET_TASK_IF_NEEDED\n    // https://developer.android.com/reference/android/content/Intent#FLAG_ACTIVITY_NEW_TASK\n    // https://developer.android.com/reference/android/content/Intent#FLAG_ACTIVITY_RESET_TASK_IF_NEEDED\n    '-f', '0x10200000',\n    '-n', await this.adb.resolveLaunchableActivity(appId),\n  ]);\n  this.log.debug(stdout);\n  if (/^error:/mi.test(stdout)) {\n    throw new Error(`Cannot activate '${appId}'. Original error: ${stdout}`);\n  }\n};\n\n/**\n * @typedef {Object} UninstallOptions\n * @property {number} timeout [20000] - The count of milliseconds to wait until the\n *                                      app is uninstalled.\n * @property {boolean} keepData [false] - Set to true in order to keep the\n *                                        application data and cache folders after uninstall.\n */\n\n/**\n * Remove the corresponding application if is installed.\n * The call is ignored if the app is not installed.\n *\n * @param {string} appId - Application package identifier\n * @param {?UninstallOptions} options - The set of removal options\n * @returns {boolean} True if the package was found on the device and\n *                    successfully uninstalled.\n */\ncommands.removeApp = async function removeApp (appId, options = {}) {\n  return await this.adb.uninstallApk(appId, options);\n};\n\n/**\n * @typedef {Object} TerminateOptions\n * @property {number|string} timeout [500] - The count of milliseconds to wait until the\n *                                           app is terminated.\n */\n\n/**\n * Terminates the app if it is running.\n *\n * @param {string} appId - Application package identifier\n * @param {?TerminateOptions} options - The set of application termination options\n * @returns {boolean} True if the app has been successfully terminated.\n * @throws {Error} if the app has not been terminated within the given timeout.\n */\ncommands.terminateApp = async function terminateApp (appId, options = {}) {\n  this.log.info(`Terminating '${appId}'`);\n  if (!(await this.adb.processExists(appId))) {\n    this.log.info(`The app '${appId}' is not running`);\n    return false;\n  }\n  await this.adb.forceStop(appId);\n  const timeout = util.hasValue(options.timeout) && !isNaN(options.timeout) ? parseInt(options.timeout, 10) : 500;\n  try {\n    await waitForCondition(async () => await this.queryAppState(appId) <= APP_STATE.NOT_RUNNING,\n      {waitMs: timeout, intervalMs: 100});\n  } catch (e) {\n    this.log.errorAndThrow(`'${appId}' is still running after ${timeout}ms timeout`);\n  }\n  this.log.info(`'${appId}' has been successfully terminated`);\n  return true;\n};\n\n/**\n * @typedef {Object} InstallOptions\n * @property {number} timeout [60000] - The count of milliseconds to wait until the\n *                                      app is installed.\n * @property {boolean} allowTestPackages [false] - Set to true in order to allow test\n *                                                 packages installation.\n * @property {boolean} useSdcard [false] - Set to true to install the app on sdcard\n *                                         instead of the device memory.\n * @property {boolean} grantPermissions [false] - Set to true in order to grant all the\n *                                                permissions requested in the application's manifest\n *                                                automatically after the installation is completed\n *                                                under Android 6+.\n * @property {boolean} replace [true] - Set it to false if you don't want\n *                                      the application to be upgraded/reinstalled\n *                                      if it is already present on the device.\n */\n\n/**\n * Installs the given application to the device under test\n *\n * @param {string} appPath - The local apk path or a remote url\n * @param {?InstallOptions} options - The set of installation options\n * @throws {Error} if the given apk does not exist or is not reachable\n */\ncommands.installApp = async function installApp (appPath, options = {}) {\n  const localPath = await this.helpers.configureApp(appPath, APP_EXTENSIONS);\n  await this.adb.install(localPath, options);\n};\n\n/**\n * @typedef {Object} ClearAppOptions\n * @property {!string} appId The identifier of the application package to be cleared\n */\n\n/**\n * Deletes all data associated with a package.\n *\n * @param {ClearAppOptions} opts\n * @throws {Error} If cleaning of the app data fails\n */\ncommands.mobileClearApp = async function mobileClearApp (opts = {}) {\n  const {appId} = opts;\n  if (!appId) {\n    throw new errors.InvalidArgumentError(`The 'appId' argument is required`);\n  }\n  await this.adb.clear(appId);\n};\n\nexport { commands };\nexport default commands;\n"],"mappings":"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"}
141
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"file":"lib/commands/app-management.js","names":["APP_EXTENSIONS","RESOLVER_ACTIVITY_NAME","commands","isAppInstalled","appId","adb","queryAppState","log","info","APP_STATE","NOT_INSTALLED","processExists","NOT_RUNNING","appIdRe","RegExp","_","escapeRegExp","line","dumpWindows","split","test","some","x","includes","RUNNING_IN_FOREGROUND","RUNNING_IN_BACKGROUND","activateApp","debug","apiLevel","getApiLevel","cmd","output","shell","e","errorAndThrow","message","activityName","resolveLaunchableActivity","preferCmd","stdout","Error","removeApp","options","uninstallApk","terminateApp","forceStop","timeout","util","hasValue","isNaN","parseInt","waitForCondition","waitMs","intervalMs","installApp","appPath","localPath","helpers","configureApp","install","mobileClearApp","opts","errors","InvalidArgumentError","clear"],"sourceRoot":"../../..","sources":["lib/commands/app-management.js"],"sourcesContent":["import _ from 'lodash';\nimport { waitForCondition } from 'asyncbox';\nimport { util } from 'appium/support';\nimport { APP_STATE } from '../android-helpers';\nimport { errors } from 'appium/driver';\n\nconst APP_EXTENSIONS = ['.apk', '.apks'];\nconst RESOLVER_ACTIVITY_NAME = 'android/com.android.internal.app.ResolverActivity';\n\nlet commands = {};\n\n/**\n * Verify whether an application is installed or not\n *\n * @param {string} appId - Application package identifier\n * @returns {boolean} true if the app is installed\n */\ncommands.isAppInstalled = async function isAppInstalled (appId) {\n  return await this.adb.isAppInstalled(appId);\n};\n\n/**\n * Queries the current state of the app.\n *\n * @param {string} appId - Application package identifier\n * @returns {number} The corresponding constant, which describes\n *                   the current application state:\n * 0 - is the app is not installed\n * 1 - if the app is installed, but is not running\n * 3 - if the app is running in the background\n * 4 - if the app is running in the foreground\n */\ncommands.queryAppState = async function queryAppState (appId) {\n  this.log.info(`Querying the state of '${appId}'`);\n  if (!await this.adb.isAppInstalled(appId)) {\n    return APP_STATE.NOT_INSTALLED;\n  }\n  if (!await this.adb.processExists(appId)) {\n    return APP_STATE.NOT_RUNNING;\n  }\n  const appIdRe = new RegExp(`\\\\b${_.escapeRegExp(appId)}/`);\n  for (const line of (await this.adb.dumpWindows()).split('\\n')) {\n    if (appIdRe.test(line) && ['mCurrentFocus', 'mFocusedApp'].some((x) => line.includes(x))) {\n      return APP_STATE.RUNNING_IN_FOREGROUND;\n    }\n  }\n  return APP_STATE.RUNNING_IN_BACKGROUND;\n};\n\n/**\n * Activates the given application or launches it if necessary.\n * The action is done with monkey tool and literally simulates\n * clicking the corresponding application icon on the dashboard.\n *\n * @param {string} appId - Application package identifier\n */\ncommands.activateApp = async function activateApp (appId) {\n  this.log.debug(`Activating '${appId}'`);\n  const apiLevel = await this.adb.getApiLevel();\n  // Fallback to Monkey in older APIs\n  if (apiLevel < 24) {\n    // The monkey command could raise an issue as https://stackoverflow.com/questions/44860475/how-to-use-the-monkey-command-with-an-android-system-that-doesnt-have-physical\n    // but '--pct-syskeys 0' could cause another background process issue. https://github.com/appium/appium/issues/16941#issuecomment-1129837285\n    const cmd = ['monkey',\n      '-p', appId,\n      '-c', 'android.intent.category.LAUNCHER',\n      '1'];\n    let output = '';\n    try {\n      output = await this.adb.shell(cmd);\n      this.log.debug(`Command stdout: ${output}`);\n    } catch (e) {\n      this.log.errorAndThrow(`Cannot activate '${appId}'. Original error: ${e.message}`);\n    }\n    if (output.includes('monkey aborted')) {\n      this.log.errorAndThrow(`Cannot activate '${appId}'. Are you sure it is installed?`);\n    }\n    return;\n  }\n\n  let activityName = await this.adb.resolveLaunchableActivity(appId);\n  if (activityName === RESOLVER_ACTIVITY_NAME) {\n    // https://github.com/appium/appium/issues/17128\n    this.log.debug(\n      `The launchable activity name of '${appId}' was resolved to '${activityName}'. ` +\n      `Switching the resolver to not use cmd`\n    );\n    activityName = await this.adb.resolveLaunchableActivity(appId, {preferCmd: false});\n  }\n\n  const stdout = await this.adb.shell([\n    'am', (apiLevel < 26) ? 'start' : 'start-activity',\n    '-a', 'android.intent.action.MAIN',\n    '-c', 'android.intent.category.LAUNCHER',\n    // FLAG_ACTIVITY_NEW_TASK | FLAG_ACTIVITY_RESET_TASK_IF_NEEDED\n    // https://developer.android.com/reference/android/content/Intent#FLAG_ACTIVITY_NEW_TASK\n    // https://developer.android.com/reference/android/content/Intent#FLAG_ACTIVITY_RESET_TASK_IF_NEEDED\n    '-f', '0x10200000',\n    '-n', activityName,\n  ]);\n  this.log.debug(stdout);\n  if (/^error:/mi.test(stdout)) {\n    throw new Error(`Cannot activate '${appId}'. Original error: ${stdout}`);\n  }\n};\n\n/**\n * @typedef {Object} UninstallOptions\n * @property {number} timeout [20000] - The count of milliseconds to wait until the\n *                                      app is uninstalled.\n * @property {boolean} keepData [false] - Set to true in order to keep the\n *                                        application data and cache folders after uninstall.\n */\n\n/**\n * Remove the corresponding application if is installed.\n * The call is ignored if the app is not installed.\n *\n * @param {string} appId - Application package identifier\n * @param {?UninstallOptions} options - The set of removal options\n * @returns {boolean} True if the package was found on the device and\n *                    successfully uninstalled.\n */\ncommands.removeApp = async function removeApp (appId, options = {}) {\n  return await this.adb.uninstallApk(appId, options);\n};\n\n/**\n * @typedef {Object} TerminateOptions\n * @property {number|string} timeout [500] - The count of milliseconds to wait until the\n *                                           app is terminated.\n */\n\n/**\n * Terminates the app if it is running.\n *\n * @param {string} appId - Application package identifier\n * @param {?TerminateOptions} options - The set of application termination options\n * @returns {boolean} True if the app has been successfully terminated.\n * @throws {Error} if the app has not been terminated within the given timeout.\n */\ncommands.terminateApp = async function terminateApp (appId, options = {}) {\n  this.log.info(`Terminating '${appId}'`);\n  if (!(await this.adb.processExists(appId))) {\n    this.log.info(`The app '${appId}' is not running`);\n    return false;\n  }\n  await this.adb.forceStop(appId);\n  const timeout = util.hasValue(options.timeout) && !isNaN(options.timeout) ? parseInt(options.timeout, 10) : 500;\n  try {\n    await waitForCondition(async () => await this.queryAppState(appId) <= APP_STATE.NOT_RUNNING,\n      {waitMs: timeout, intervalMs: 100});\n  } catch (e) {\n    this.log.errorAndThrow(`'${appId}' is still running after ${timeout}ms timeout`);\n  }\n  this.log.info(`'${appId}' has been successfully terminated`);\n  return true;\n};\n\n/**\n * @typedef {Object} InstallOptions\n * @property {number} timeout [60000] - The count of milliseconds to wait until the\n *                                      app is installed.\n * @property {boolean} allowTestPackages [false] - Set to true in order to allow test\n *                                                 packages installation.\n * @property {boolean} useSdcard [false] - Set to true to install the app on sdcard\n *                                         instead of the device memory.\n * @property {boolean} grantPermissions [false] - Set to true in order to grant all the\n *                                                permissions requested in the application's manifest\n *                                                automatically after the installation is completed\n *                                                under Android 6+.\n * @property {boolean} replace [true] - Set it to false if you don't want\n *                                      the application to be upgraded/reinstalled\n *                                      if it is already present on the device.\n */\n\n/**\n * Installs the given application to the device under test\n *\n * @param {string} appPath - The local apk path or a remote url\n * @param {?InstallOptions} options - The set of installation options\n * @throws {Error} if the given apk does not exist or is not reachable\n */\ncommands.installApp = async function installApp (appPath, options = {}) {\n  const localPath = await this.helpers.configureApp(appPath, APP_EXTENSIONS);\n  await this.adb.install(localPath, options);\n};\n\n/**\n * @typedef {Object} ClearAppOptions\n * @property {!string} appId The identifier of the application package to be cleared\n */\n\n/**\n * Deletes all data associated with a package.\n *\n * @param {ClearAppOptions} opts\n * @throws {Error} If cleaning of the app data fails\n */\ncommands.mobileClearApp = async function mobileClearApp (opts = {}) {\n  const {appId} = opts;\n  if (!appId) {\n    throw new errors.InvalidArgumentError(`The 'appId' argument is required`);\n  }\n  await this.adb.clear(appId);\n};\n\nexport { commands };\nexport default commands;\n"],"mappings":"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"}
@@ -56,7 +56,10 @@ extensions.executeMobile = async function executeMobile(mobileCommand, opts = {}
56
56
  broadcast: 'mobileBroadcast',
57
57
  getContexts: 'mobileGetContexts',
58
58
  unlock: 'mobileUnlock',
59
- refreshGpsCache: 'mobileRefreshGpsCache'
59
+ refreshGpsCache: 'mobileRefreshGpsCache',
60
+ startMediaProjectionRecording: 'mobileStartMediaProjectionRecording',
61
+ isMediaProjectionRecordingRunning: 'mobileIsMediaProjectionRecordingRunning',
62
+ stopMediaProjectionRecording: 'mobileStopMediaProjectionRecording'
60
63
  };
61
64
 
62
65
  if (!_lodash.default.has(mobileCommandsMapping, mobileCommand)) {
@@ -70,4 +73,4 @@ var _default = extensions;
70
73
  exports.default = _default;require('source-map-support').install();
71
74
 
72
75
 
73
- //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGliL2NvbW1hbmRzL2V4ZWN1dGUuanMiLCJuYW1lcyI6WyJleHRlbnNpb25zIiwiZXhlY3V0ZSIsInNjcmlwdCIsImFyZ3MiLCJtYXRjaCIsImxvZyIsImluZm8iLCJyZXBsYWNlIiwidHJpbSIsImV4ZWN1dGVNb2JpbGUiLCJfIiwiaXNBcnJheSIsImlzV2ViQ29udGV4dCIsImVycm9ycyIsIk5vdEltcGxlbWVudGVkRXJyb3IiLCJlbmRwb2ludCIsImNocm9tZWRyaXZlciIsImp3cHJveHkiLCJkb3duc3RyZWFtUHJvdG9jb2wiLCJQUk9UT0NPTFMiLCJNSlNPTldQIiwiY29tbWFuZCIsIm1vYmlsZUNvbW1hbmQiLCJvcHRzIiwibW9iaWxlQ29tbWFuZHNNYXBwaW5nIiwic2hlbGwiLCJleGVjRW11Q29uc29sZUNvbW1hbmQiLCJzdGFydExvZ3NCcm9hZGNhc3QiLCJzdG9wTG9nc0Jyb2FkY2FzdCIsImNoYW5nZVBlcm1pc3Npb25zIiwiZ2V0UGVybWlzc2lvbnMiLCJwZXJmb3JtRWRpdG9yQWN0aW9uIiwic2Vuc29yU2V0IiwiZ2V0RGV2aWNlVGltZSIsInN0YXJ0U2NyZWVuU3RyZWFtaW5nIiwic3RvcFNjcmVlblN0cmVhbWluZyIsImdldE5vdGlmaWNhdGlvbnMiLCJsaXN0U21zIiwiZGVsZXRlRmlsZSIsImNsZWFyQXBwIiwic3RhcnRTZXJ2aWNlIiwic3RvcFNlcnZpY2UiLCJzdGFydEFjdGl2aXR5IiwiYnJvYWRjYXN0IiwiZ2V0Q29udGV4dHMiLCJ1bmxvY2siLCJyZWZyZXNoR3BzQ2FjaGUiLCJoYXMiLCJVbmtub3duQ29tbWFuZEVycm9yIiwia2V5cyJdLCJzb3VyY2VSb290IjoiLi4vLi4vLi4iLCJzb3VyY2VzIjpbImxpYi9jb21tYW5kcy9leGVjdXRlLmpzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgeyBlcnJvcnMsIFBST1RPQ09MUyB9IGZyb20gJ2FwcGl1bS9kcml2ZXInO1xuXG5jb25zdCBleHRlbnNpb25zID0ge307XG5cbmV4dGVuc2lvbnMuZXhlY3V0ZSA9IGFzeW5jIGZ1bmN0aW9uIGV4ZWN1dGUgKHNjcmlwdCwgYXJncykge1xuICBpZiAoc2NyaXB0Lm1hdGNoKC9ebW9iaWxlOi8pKSB7XG4gICAgdGhpcy5sb2cuaW5mbyhgRXhlY3V0aW5nIG5hdGl2ZSBjb21tYW5kICcke3NjcmlwdH0nYCk7XG4gICAgc2NyaXB0ID0gc2NyaXB0LnJlcGxhY2UoL15tb2JpbGU6LywgJycpLnRyaW0oKTtcbiAgICByZXR1cm4gYXdhaXQgdGhpcy5leGVjdXRlTW9iaWxlKHNjcmlwdCwgXy5pc0FycmF5KGFyZ3MpID8gYXJnc1swXSA6IGFyZ3MpO1xuICB9XG4gIGlmICghdGhpcy5pc1dlYkNvbnRleHQoKSkge1xuICAgIHRocm93IG5ldyBlcnJvcnMuTm90SW1wbGVtZW50ZWRFcnJvcigpO1xuICB9XG4gIGNvbnN0IGVuZHBvaW50ID0gdGhpcy5jaHJvbWVkcml2ZXIuandwcm94eS5kb3duc3RyZWFtUHJvdG9jb2wgPT09IFBST1RPQ09MUy5NSlNPTldQXG4gICAgPyAnL2V4ZWN1dGUnXG4gICAgOiAnL2V4ZWN1dGUvc3luYyc7XG4gIHJldHVybiBhd2FpdCB0aGlzLmNocm9tZWRyaXZlci5qd3Byb3h5LmNvbW1hbmQoZW5kcG9pbnQsICdQT1NUJywge1xuICAgIHNjcmlwdCxcbiAgICBhcmdzLFxuICB9KTtcbn07XG5cbmV4dGVuc2lvbnMuZXhlY3V0ZU1vYmlsZSA9IGFzeW5jIGZ1bmN0aW9uIGV4ZWN1dGVNb2JpbGUgKG1vYmlsZUNvbW1hbmQsIG9wdHMgPSB7fSkge1xuICBjb25zdCBtb2JpbGVDb21tYW5kc01hcHBpbmcgPSB7XG4gICAgc2hlbGw6ICdtb2JpbGVTaGVsbCcsXG5cbiAgICBleGVjRW11Q29uc29sZUNvbW1hbmQ6ICdtb2JpbGVFeGVjRW11Q29uc29sZUNvbW1hbmQnLFxuXG4gICAgc3RhcnRMb2dzQnJvYWRjYXN0OiAnbW9iaWxlU3RhcnRMb2dzQnJvYWRjYXN0JyxcbiAgICBzdG9wTG9nc0Jyb2FkY2FzdDogJ21vYmlsZVN0b3BMb2dzQnJvYWRjYXN0JyxcblxuICAgIGNoYW5nZVBlcm1pc3Npb25zOiAnbW9iaWxlQ2hhbmdlUGVybWlzc2lvbnMnLFxuICAgIGdldFBlcm1pc3Npb25zOiAnbW9iaWxlR2V0UGVybWlzc2lvbnMnLFxuXG4gICAgcGVyZm9ybUVkaXRvckFjdGlvbjogJ21vYmlsZVBlcmZvcm1FZGl0b3JBY3Rpb24nLFxuXG4gICAgc2Vuc29yU2V0OiAnc2Vuc29yU2V0JyxcblxuICAgIGdldERldmljZVRpbWU6ICdtb2JpbGVHZXREZXZpY2VUaW1lJyxcblxuICAgIHN0YXJ0U2NyZWVuU3RyZWFtaW5nOiAnbW9iaWxlU3RhcnRTY3JlZW5TdHJlYW1pbmcnLFxuICAgIHN0b3BTY3JlZW5TdHJlYW1pbmc6ICdtb2JpbGVTdG9wU2NyZWVuU3RyZWFtaW5nJyxcblxuICAgIGdldE5vdGlmaWNhdGlvbnM6ICdtb2JpbGVHZXROb3RpZmljYXRpb25zJyxcblxuICAgIGxpc3RTbXM6ICdtb2JpbGVMaXN0U21zJyxcblxuICAgIGRlbGV0ZUZpbGU6ICdtb2JpbGVEZWxldGVGaWxlJyxcblxuICAgIGNsZWFyQXBwOiAnbW9iaWxlQ2xlYXJBcHAnLFxuXG4gICAgc3RhcnRTZXJ2aWNlOiAnbW9iaWxlU3RhcnRTZXJ2aWNlJyxcbiAgICBzdG9wU2VydmljZTogJ21vYmlsZVN0b3BTZXJ2aWNlJyxcbiAgICBzdGFydEFjdGl2aXR5OiAnbW9iaWxlU3RhcnRBY3Rpdml0eScsXG4gICAgYnJvYWRjYXN0OiAnbW9iaWxlQnJvYWRjYXN0JyxcblxuICAgIGdldENvbnRleHRzOiAnbW9iaWxlR2V0Q29udGV4dHMnLFxuXG4gICAgdW5sb2NrOiAnbW9iaWxlVW5sb2NrJyxcblxuICAgIHJlZnJlc2hHcHNDYWNoZTogJ21vYmlsZVJlZnJlc2hHcHNDYWNoZScsXG4gIH07XG5cbiAgaWYgKCFfLmhhcyhtb2JpbGVDb21tYW5kc01hcHBpbmcsIG1vYmlsZUNvbW1hbmQpKSB7XG4gICAgdGhyb3cgbmV3IGVycm9ycy5Vbmtub3duQ29tbWFuZEVycm9yKGBVbmtub3duIG1vYmlsZSBjb21tYW5kIFwiJHttb2JpbGVDb21tYW5kfVwiLiBgICtcbiAgICAgIGBPbmx5ICR7Xy5rZXlzKG1vYmlsZUNvbW1hbmRzTWFwcGluZyl9IGNvbW1hbmRzIGFyZSBzdXBwb3J0ZWQuYCk7XG4gIH1cbiAgcmV0dXJuIGF3YWl0IHRoaXNbbW9iaWxlQ29tbWFuZHNNYXBwaW5nW21vYmlsZUNvbW1hbmRdXShvcHRzKTtcbn07XG5cbmV4cG9ydCBkZWZhdWx0IGV4dGVuc2lvbnM7XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7O0FBQ0E7O0FBRUEsTUFBTUEsVUFBVSxHQUFHLEVBQW5COztBQUVBQSxVQUFVLENBQUNDLE9BQVgsR0FBcUIsZUFBZUEsT0FBZixDQUF3QkMsTUFBeEIsRUFBZ0NDLElBQWhDLEVBQXNDO0VBQ3pELElBQUlELE1BQU0sQ0FBQ0UsS0FBUCxDQUFhLFVBQWIsQ0FBSixFQUE4QjtJQUM1QixLQUFLQyxHQUFMLENBQVNDLElBQVQsQ0FBZSw2QkFBNEJKLE1BQU8sR0FBbEQ7SUFDQUEsTUFBTSxHQUFHQSxNQUFNLENBQUNLLE9BQVAsQ0FBZSxVQUFmLEVBQTJCLEVBQTNCLEVBQStCQyxJQUEvQixFQUFUO0lBQ0EsT0FBTyxNQUFNLEtBQUtDLGFBQUwsQ0FBbUJQLE1BQW5CLEVBQTJCUSxlQUFBLENBQUVDLE9BQUYsQ0FBVVIsSUFBVixJQUFrQkEsSUFBSSxDQUFDLENBQUQsQ0FBdEIsR0FBNEJBLElBQXZELENBQWI7RUFDRDs7RUFDRCxJQUFJLENBQUMsS0FBS1MsWUFBTCxFQUFMLEVBQTBCO0lBQ3hCLE1BQU0sSUFBSUMsY0FBQSxDQUFPQyxtQkFBWCxFQUFOO0VBQ0Q7O0VBQ0QsTUFBTUMsUUFBUSxHQUFHLEtBQUtDLFlBQUwsQ0FBa0JDLE9BQWxCLENBQTBCQyxrQkFBMUIsS0FBaURDLGlCQUFBLENBQVVDLE9BQTNELEdBQ2IsVUFEYSxHQUViLGVBRko7RUFHQSxPQUFPLE1BQU0sS0FBS0osWUFBTCxDQUFrQkMsT0FBbEIsQ0FBMEJJLE9BQTFCLENBQWtDTixRQUFsQyxFQUE0QyxNQUE1QyxFQUFvRDtJQUMvRGIsTUFEK0Q7SUFFL0RDO0VBRitELENBQXBELENBQWI7QUFJRCxDQWhCRDs7QUFrQkFILFVBQVUsQ0FBQ1MsYUFBWCxHQUEyQixlQUFlQSxhQUFmLENBQThCYSxhQUE5QixFQUE2Q0MsSUFBSSxHQUFHLEVBQXBELEVBQXdEO0VBQ2pGLE1BQU1DLHFCQUFxQixHQUFHO0lBQzVCQyxLQUFLLEVBQUUsYUFEcUI7SUFHNUJDLHFCQUFxQixFQUFFLDZCQUhLO0lBSzVCQyxrQkFBa0IsRUFBRSwwQkFMUTtJQU01QkMsaUJBQWlCLEVBQUUseUJBTlM7SUFRNUJDLGlCQUFpQixFQUFFLHlCQVJTO0lBUzVCQyxjQUFjLEVBQUUsc0JBVFk7SUFXNUJDLG1CQUFtQixFQUFFLDJCQVhPO0lBYTVCQyxTQUFTLEVBQUUsV0FiaUI7SUFlNUJDLGFBQWEsRUFBRSxxQkFmYTtJQWlCNUJDLG9CQUFvQixFQUFFLDRCQWpCTTtJQWtCNUJDLG1CQUFtQixFQUFFLDJCQWxCTztJQW9CNUJDLGdCQUFnQixFQUFFLHdCQXBCVTtJQXNCNUJDLE9BQU8sRUFBRSxlQXRCbUI7SUF3QjVCQyxVQUFVLEVBQUUsa0JBeEJnQjtJQTBCNUJDLFFBQVEsRUFBRSxnQkExQmtCO0lBNEI1QkMsWUFBWSxFQUFFLG9CQTVCYztJQTZCNUJDLFdBQVcsRUFBRSxtQkE3QmU7SUE4QjVCQyxhQUFhLEVBQUUscUJBOUJhO0lBK0I1QkMsU0FBUyxFQUFFLGlCQS9CaUI7SUFpQzVCQyxXQUFXLEVBQUUsbUJBakNlO0lBbUM1QkMsTUFBTSxFQUFFLGNBbkNvQjtJQXFDNUJDLGVBQWUsRUFBRTtFQXJDVyxDQUE5Qjs7RUF3Q0EsSUFBSSxDQUFDcEMsZUFBQSxDQUFFcUMsR0FBRixDQUFNdkIscUJBQU4sRUFBNkJGLGFBQTdCLENBQUwsRUFBa0Q7SUFDaEQsTUFBTSxJQUFJVCxjQUFBLENBQU9tQyxtQkFBWCxDQUFnQywyQkFBMEIxQixhQUFjLEtBQXpDLEdBQ2xDLFFBQU9aLGVBQUEsQ0FBRXVDLElBQUYsQ0FBT3pCLHFCQUFQLENBQThCLDBCQURsQyxDQUFOO0VBRUQ7O0VBQ0QsT0FBTyxNQUFNLEtBQUtBLHFCQUFxQixDQUFDRixhQUFELENBQTFCLEVBQTJDQyxJQUEzQyxDQUFiO0FBQ0QsQ0E5Q0Q7O2VBZ0RldkIsVSJ9
76
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGliL2NvbW1hbmRzL2V4ZWN1dGUuanMiLCJuYW1lcyI6WyJleHRlbnNpb25zIiwiZXhlY3V0ZSIsInNjcmlwdCIsImFyZ3MiLCJtYXRjaCIsImxvZyIsImluZm8iLCJyZXBsYWNlIiwidHJpbSIsImV4ZWN1dGVNb2JpbGUiLCJfIiwiaXNBcnJheSIsImlzV2ViQ29udGV4dCIsImVycm9ycyIsIk5vdEltcGxlbWVudGVkRXJyb3IiLCJlbmRwb2ludCIsImNocm9tZWRyaXZlciIsImp3cHJveHkiLCJkb3duc3RyZWFtUHJvdG9jb2wiLCJQUk9UT0NPTFMiLCJNSlNPTldQIiwiY29tbWFuZCIsIm1vYmlsZUNvbW1hbmQiLCJvcHRzIiwibW9iaWxlQ29tbWFuZHNNYXBwaW5nIiwic2hlbGwiLCJleGVjRW11Q29uc29sZUNvbW1hbmQiLCJzdGFydExvZ3NCcm9hZGNhc3QiLCJzdG9wTG9nc0Jyb2FkY2FzdCIsImNoYW5nZVBlcm1pc3Npb25zIiwiZ2V0UGVybWlzc2lvbnMiLCJwZXJmb3JtRWRpdG9yQWN0aW9uIiwic2Vuc29yU2V0IiwiZ2V0RGV2aWNlVGltZSIsInN0YXJ0U2NyZWVuU3RyZWFtaW5nIiwic3RvcFNjcmVlblN0cmVhbWluZyIsImdldE5vdGlmaWNhdGlvbnMiLCJsaXN0U21zIiwiZGVsZXRlRmlsZSIsImNsZWFyQXBwIiwic3RhcnRTZXJ2aWNlIiwic3RvcFNlcnZpY2UiLCJzdGFydEFjdGl2aXR5IiwiYnJvYWRjYXN0IiwiZ2V0Q29udGV4dHMiLCJ1bmxvY2siLCJyZWZyZXNoR3BzQ2FjaGUiLCJzdGFydE1lZGlhUHJvamVjdGlvblJlY29yZGluZyIsImlzTWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nUnVubmluZyIsInN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmciLCJoYXMiLCJVbmtub3duQ29tbWFuZEVycm9yIiwia2V5cyJdLCJzb3VyY2VSb290IjoiLi4vLi4vLi4iLCJzb3VyY2VzIjpbImxpYi9jb21tYW5kcy9leGVjdXRlLmpzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgeyBlcnJvcnMsIFBST1RPQ09MUyB9IGZyb20gJ2FwcGl1bS9kcml2ZXInO1xuXG5jb25zdCBleHRlbnNpb25zID0ge307XG5cbmV4dGVuc2lvbnMuZXhlY3V0ZSA9IGFzeW5jIGZ1bmN0aW9uIGV4ZWN1dGUgKHNjcmlwdCwgYXJncykge1xuICBpZiAoc2NyaXB0Lm1hdGNoKC9ebW9iaWxlOi8pKSB7XG4gICAgdGhpcy5sb2cuaW5mbyhgRXhlY3V0aW5nIG5hdGl2ZSBjb21tYW5kICcke3NjcmlwdH0nYCk7XG4gICAgc2NyaXB0ID0gc2NyaXB0LnJlcGxhY2UoL15tb2JpbGU6LywgJycpLnRyaW0oKTtcbiAgICByZXR1cm4gYXdhaXQgdGhpcy5leGVjdXRlTW9iaWxlKHNjcmlwdCwgXy5pc0FycmF5KGFyZ3MpID8gYXJnc1swXSA6IGFyZ3MpO1xuICB9XG4gIGlmICghdGhpcy5pc1dlYkNvbnRleHQoKSkge1xuICAgIHRocm93IG5ldyBlcnJvcnMuTm90SW1wbGVtZW50ZWRFcnJvcigpO1xuICB9XG4gIGNvbnN0IGVuZHBvaW50ID0gdGhpcy5jaHJvbWVkcml2ZXIuandwcm94eS5kb3duc3RyZWFtUHJvdG9jb2wgPT09IFBST1RPQ09MUy5NSlNPTldQXG4gICAgPyAnL2V4ZWN1dGUnXG4gICAgOiAnL2V4ZWN1dGUvc3luYyc7XG4gIHJldHVybiBhd2FpdCB0aGlzLmNocm9tZWRyaXZlci5qd3Byb3h5LmNvbW1hbmQoZW5kcG9pbnQsICdQT1NUJywge1xuICAgIHNjcmlwdCxcbiAgICBhcmdzLFxuICB9KTtcbn07XG5cbmV4dGVuc2lvbnMuZXhlY3V0ZU1vYmlsZSA9IGFzeW5jIGZ1bmN0aW9uIGV4ZWN1dGVNb2JpbGUgKG1vYmlsZUNvbW1hbmQsIG9wdHMgPSB7fSkge1xuICBjb25zdCBtb2JpbGVDb21tYW5kc01hcHBpbmcgPSB7XG4gICAgc2hlbGw6ICdtb2JpbGVTaGVsbCcsXG5cbiAgICBleGVjRW11Q29uc29sZUNvbW1hbmQ6ICdtb2JpbGVFeGVjRW11Q29uc29sZUNvbW1hbmQnLFxuXG4gICAgc3RhcnRMb2dzQnJvYWRjYXN0OiAnbW9iaWxlU3RhcnRMb2dzQnJvYWRjYXN0JyxcbiAgICBzdG9wTG9nc0Jyb2FkY2FzdDogJ21vYmlsZVN0b3BMb2dzQnJvYWRjYXN0JyxcblxuICAgIGNoYW5nZVBlcm1pc3Npb25zOiAnbW9iaWxlQ2hhbmdlUGVybWlzc2lvbnMnLFxuICAgIGdldFBlcm1pc3Npb25zOiAnbW9iaWxlR2V0UGVybWlzc2lvbnMnLFxuXG4gICAgcGVyZm9ybUVkaXRvckFjdGlvbjogJ21vYmlsZVBlcmZvcm1FZGl0b3JBY3Rpb24nLFxuXG4gICAgc2Vuc29yU2V0OiAnc2Vuc29yU2V0JyxcblxuICAgIGdldERldmljZVRpbWU6ICdtb2JpbGVHZXREZXZpY2VUaW1lJyxcblxuICAgIHN0YXJ0U2NyZWVuU3RyZWFtaW5nOiAnbW9iaWxlU3RhcnRTY3JlZW5TdHJlYW1pbmcnLFxuICAgIHN0b3BTY3JlZW5TdHJlYW1pbmc6ICdtb2JpbGVTdG9wU2NyZWVuU3RyZWFtaW5nJyxcblxuICAgIGdldE5vdGlmaWNhdGlvbnM6ICdtb2JpbGVHZXROb3RpZmljYXRpb25zJyxcblxuICAgIGxpc3RTbXM6ICdtb2JpbGVMaXN0U21zJyxcblxuICAgIGRlbGV0ZUZpbGU6ICdtb2JpbGVEZWxldGVGaWxlJyxcblxuICAgIGNsZWFyQXBwOiAnbW9iaWxlQ2xlYXJBcHAnLFxuXG4gICAgc3RhcnRTZXJ2aWNlOiAnbW9iaWxlU3RhcnRTZXJ2aWNlJyxcbiAgICBzdG9wU2VydmljZTogJ21vYmlsZVN0b3BTZXJ2aWNlJyxcbiAgICBzdGFydEFjdGl2aXR5OiAnbW9iaWxlU3RhcnRBY3Rpdml0eScsXG4gICAgYnJvYWRjYXN0OiAnbW9iaWxlQnJvYWRjYXN0JyxcblxuICAgIGdldENvbnRleHRzOiAnbW9iaWxlR2V0Q29udGV4dHMnLFxuXG4gICAgdW5sb2NrOiAnbW9iaWxlVW5sb2NrJyxcblxuICAgIHJlZnJlc2hHcHNDYWNoZTogJ21vYmlsZVJlZnJlc2hHcHNDYWNoZScsXG5cbiAgICBzdGFydE1lZGlhUHJvamVjdGlvblJlY29yZGluZzogJ21vYmlsZVN0YXJ0TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nJyxcbiAgICBpc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmc6ICdtb2JpbGVJc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmcnLFxuICAgIHN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmc6ICdtb2JpbGVTdG9wTWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nJyxcbiAgfTtcblxuICBpZiAoIV8uaGFzKG1vYmlsZUNvbW1hbmRzTWFwcGluZywgbW9iaWxlQ29tbWFuZCkpIHtcbiAgICB0aHJvdyBuZXcgZXJyb3JzLlVua25vd25Db21tYW5kRXJyb3IoYFVua25vd24gbW9iaWxlIGNvbW1hbmQgXCIke21vYmlsZUNvbW1hbmR9XCIuIGAgK1xuICAgICAgYE9ubHkgJHtfLmtleXMobW9iaWxlQ29tbWFuZHNNYXBwaW5nKX0gY29tbWFuZHMgYXJlIHN1cHBvcnRlZC5gKTtcbiAgfVxuICByZXR1cm4gYXdhaXQgdGhpc1ttb2JpbGVDb21tYW5kc01hcHBpbmdbbW9iaWxlQ29tbWFuZF1dKG9wdHMpO1xufTtcblxuZXhwb3J0IGRlZmF1bHQgZXh0ZW5zaW9ucztcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFFQSxNQUFNQSxVQUFVLEdBQUcsRUFBbkI7O0FBRUFBLFVBQVUsQ0FBQ0MsT0FBWCxHQUFxQixlQUFlQSxPQUFmLENBQXdCQyxNQUF4QixFQUFnQ0MsSUFBaEMsRUFBc0M7RUFDekQsSUFBSUQsTUFBTSxDQUFDRSxLQUFQLENBQWEsVUFBYixDQUFKLEVBQThCO0lBQzVCLEtBQUtDLEdBQUwsQ0FBU0MsSUFBVCxDQUFlLDZCQUE0QkosTUFBTyxHQUFsRDtJQUNBQSxNQUFNLEdBQUdBLE1BQU0sQ0FBQ0ssT0FBUCxDQUFlLFVBQWYsRUFBMkIsRUFBM0IsRUFBK0JDLElBQS9CLEVBQVQ7SUFDQSxPQUFPLE1BQU0sS0FBS0MsYUFBTCxDQUFtQlAsTUFBbkIsRUFBMkJRLGVBQUEsQ0FBRUMsT0FBRixDQUFVUixJQUFWLElBQWtCQSxJQUFJLENBQUMsQ0FBRCxDQUF0QixHQUE0QkEsSUFBdkQsQ0FBYjtFQUNEOztFQUNELElBQUksQ0FBQyxLQUFLUyxZQUFMLEVBQUwsRUFBMEI7SUFDeEIsTUFBTSxJQUFJQyxjQUFBLENBQU9DLG1CQUFYLEVBQU47RUFDRDs7RUFDRCxNQUFNQyxRQUFRLEdBQUcsS0FBS0MsWUFBTCxDQUFrQkMsT0FBbEIsQ0FBMEJDLGtCQUExQixLQUFpREMsaUJBQUEsQ0FBVUMsT0FBM0QsR0FDYixVQURhLEdBRWIsZUFGSjtFQUdBLE9BQU8sTUFBTSxLQUFLSixZQUFMLENBQWtCQyxPQUFsQixDQUEwQkksT0FBMUIsQ0FBa0NOLFFBQWxDLEVBQTRDLE1BQTVDLEVBQW9EO0lBQy9EYixNQUQrRDtJQUUvREM7RUFGK0QsQ0FBcEQsQ0FBYjtBQUlELENBaEJEOztBQWtCQUgsVUFBVSxDQUFDUyxhQUFYLEdBQTJCLGVBQWVBLGFBQWYsQ0FBOEJhLGFBQTlCLEVBQTZDQyxJQUFJLEdBQUcsRUFBcEQsRUFBd0Q7RUFDakYsTUFBTUMscUJBQXFCLEdBQUc7SUFDNUJDLEtBQUssRUFBRSxhQURxQjtJQUc1QkMscUJBQXFCLEVBQUUsNkJBSEs7SUFLNUJDLGtCQUFrQixFQUFFLDBCQUxRO0lBTTVCQyxpQkFBaUIsRUFBRSx5QkFOUztJQVE1QkMsaUJBQWlCLEVBQUUseUJBUlM7SUFTNUJDLGNBQWMsRUFBRSxzQkFUWTtJQVc1QkMsbUJBQW1CLEVBQUUsMkJBWE87SUFhNUJDLFNBQVMsRUFBRSxXQWJpQjtJQWU1QkMsYUFBYSxFQUFFLHFCQWZhO0lBaUI1QkMsb0JBQW9CLEVBQUUsNEJBakJNO0lBa0I1QkMsbUJBQW1CLEVBQUUsMkJBbEJPO0lBb0I1QkMsZ0JBQWdCLEVBQUUsd0JBcEJVO0lBc0I1QkMsT0FBTyxFQUFFLGVBdEJtQjtJQXdCNUJDLFVBQVUsRUFBRSxrQkF4QmdCO0lBMEI1QkMsUUFBUSxFQUFFLGdCQTFCa0I7SUE0QjVCQyxZQUFZLEVBQUUsb0JBNUJjO0lBNkI1QkMsV0FBVyxFQUFFLG1CQTdCZTtJQThCNUJDLGFBQWEsRUFBRSxxQkE5QmE7SUErQjVCQyxTQUFTLEVBQUUsaUJBL0JpQjtJQWlDNUJDLFdBQVcsRUFBRSxtQkFqQ2U7SUFtQzVCQyxNQUFNLEVBQUUsY0FuQ29CO0lBcUM1QkMsZUFBZSxFQUFFLHVCQXJDVztJQXVDNUJDLDZCQUE2QixFQUFFLHFDQXZDSDtJQXdDNUJDLGlDQUFpQyxFQUFFLHlDQXhDUDtJQXlDNUJDLDRCQUE0QixFQUFFO0VBekNGLENBQTlCOztFQTRDQSxJQUFJLENBQUN2QyxlQUFBLENBQUV3QyxHQUFGLENBQU0xQixxQkFBTixFQUE2QkYsYUFBN0IsQ0FBTCxFQUFrRDtJQUNoRCxNQUFNLElBQUlULGNBQUEsQ0FBT3NDLG1CQUFYLENBQWdDLDJCQUEwQjdCLGFBQWMsS0FBekMsR0FDbEMsUUFBT1osZUFBQSxDQUFFMEMsSUFBRixDQUFPNUIscUJBQVAsQ0FBOEIsMEJBRGxDLENBQU47RUFFRDs7RUFDRCxPQUFPLE1BQU0sS0FBS0EscUJBQXFCLENBQUNGLGFBQUQsQ0FBMUIsRUFBMkNDLElBQTNDLENBQWI7QUFDRCxDQWxERDs7ZUFvRGV2QixVIn0=
@@ -51,11 +51,13 @@ var _systemBars = _interopRequireDefault(require("./system-bars"));
51
51
 
52
52
  var _log = _interopRequireDefault(require("./log"));
53
53
 
54
+ var _mediaProjection = _interopRequireDefault(require("./media-projection"));
55
+
54
56
  let commands = {};
55
57
  exports.commands = commands;
56
- Object.assign(commands, _find.default, _general.default, _alert.default, _element.default, _context.default, _actions.default, _touch.default, _ime.default, _network.default, _coverage.default, _recordscreen.default, _intent.default, _streamscreen.default, _performance.default, _execute.default, _shell.default, _emuConsole.default, _systemBars.default, _appManagement.default, _fileActions.default, _log.default);
58
+ Object.assign(commands, _find.default, _general.default, _alert.default, _element.default, _context.default, _actions.default, _touch.default, _ime.default, _network.default, _coverage.default, _recordscreen.default, _intent.default, _streamscreen.default, _performance.default, _execute.default, _shell.default, _emuConsole.default, _systemBars.default, _appManagement.default, _fileActions.default, _log.default, _mediaProjection.default);
57
59
  var _default = commands;
58
60
  exports.default = _default;require('source-map-support').install();
59
61
 
60
62
 
61
- //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGliL2NvbW1hbmRzL2luZGV4LmpzIiwibmFtZXMiOlsiY29tbWFuZHMiLCJPYmplY3QiLCJhc3NpZ24iLCJmaW5kQ21kcyIsImdlbmVyYWxDbWRzIiwiYWxlcnRDbWRzIiwiZWxlbWVudENtZHMiLCJjb250ZXh0Q21kcyIsImFjdGlvbkNtZHMiLCJ0b3VjaENtZHMiLCJpbWVDbWRzIiwibmV0d29ya0NtZHMiLCJjb3ZlcmFnZUNtZHMiLCJyZWNvcmRzY3JlZW5DbWRzIiwiaW50ZW50Q21kcyIsInNjcmVlblN0cmVhbUNtZHMiLCJwZXJmb3JtYW5jZUNtZHMiLCJleGVjdXRlQ21kcyIsInNoZWxsQ21kcyIsImVtdUNvbnNvbGVDbWRzIiwic3lzdGVtQmFyc0NtZHMiLCJhcHBNYW5hZ2VtZW50Q21kcyIsImZpbGVBY3Rpb25zQ21kcyIsImxvZ0NtZHMiXSwic291cmNlUm9vdCI6Ii4uLy4uLy4uIiwic291cmNlcyI6WyJsaWIvY29tbWFuZHMvaW5kZXguanMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGZpbmRDbWRzIGZyb20gJy4vZmluZCc7XG5pbXBvcnQgZ2VuZXJhbENtZHMgZnJvbSAnLi9nZW5lcmFsJztcbmltcG9ydCBhbGVydENtZHMgZnJvbSAnLi9hbGVydCc7XG5pbXBvcnQgZWxlbWVudENtZHMgZnJvbSAnLi9lbGVtZW50JztcbmltcG9ydCBjb250ZXh0Q21kcyBmcm9tICcuL2NvbnRleHQnO1xuaW1wb3J0IGFjdGlvbkNtZHMgZnJvbSAnLi9hY3Rpb25zJztcbmltcG9ydCB0b3VjaENtZHMgZnJvbSAnLi90b3VjaCc7XG5pbXBvcnQgaW1lQ21kcyBmcm9tICcuL2ltZSc7XG5pbXBvcnQgbmV0d29ya0NtZHMgZnJvbSAnLi9uZXR3b3JrJztcbmltcG9ydCBjb3ZlcmFnZUNtZHMgZnJvbSAnLi9jb3ZlcmFnZSc7XG5pbXBvcnQgcmVjb3Jkc2NyZWVuQ21kcyBmcm9tICcuL3JlY29yZHNjcmVlbic7XG5pbXBvcnQgc2NyZWVuU3RyZWFtQ21kcyBmcm9tICcuL3N0cmVhbXNjcmVlbic7XG5pbXBvcnQgcGVyZm9ybWFuY2VDbWRzIGZyb20gJy4vcGVyZm9ybWFuY2UnO1xuaW1wb3J0IGV4ZWN1dGVDbWRzIGZyb20gJy4vZXhlY3V0ZSc7XG5pbXBvcnQgc2hlbGxDbWRzIGZyb20gJy4vc2hlbGwnO1xuaW1wb3J0IGVtdUNvbnNvbGVDbWRzIGZyb20gJy4vZW11LWNvbnNvbGUnO1xuaW1wb3J0IGZpbGVBY3Rpb25zQ21kcyBmcm9tICcuL2ZpbGUtYWN0aW9ucyc7XG5pbXBvcnQgYXBwTWFuYWdlbWVudENtZHMgZnJvbSAnLi9hcHAtbWFuYWdlbWVudCc7XG5pbXBvcnQgaW50ZW50Q21kcyBmcm9tICcuL2ludGVudCc7XG5pbXBvcnQgc3lzdGVtQmFyc0NtZHMgZnJvbSAnLi9zeXN0ZW0tYmFycyc7XG5pbXBvcnQgbG9nQ21kcyBmcm9tICcuL2xvZyc7XG5cblxubGV0IGNvbW1hbmRzID0ge307XG5PYmplY3QuYXNzaWduKFxuICBjb21tYW5kcyxcbiAgZmluZENtZHMsXG4gIGdlbmVyYWxDbWRzLFxuICBhbGVydENtZHMsXG4gIGVsZW1lbnRDbWRzLFxuICBjb250ZXh0Q21kcyxcbiAgYWN0aW9uQ21kcyxcbiAgdG91Y2hDbWRzLFxuICBpbWVDbWRzLFxuICBuZXR3b3JrQ21kcyxcbiAgY292ZXJhZ2VDbWRzLFxuICByZWNvcmRzY3JlZW5DbWRzLFxuICBpbnRlbnRDbWRzLFxuICBzY3JlZW5TdHJlYW1DbWRzLFxuICBwZXJmb3JtYW5jZUNtZHMsXG4gIGV4ZWN1dGVDbWRzLFxuICBzaGVsbENtZHMsXG4gIGVtdUNvbnNvbGVDbWRzLFxuICBzeXN0ZW1CYXJzQ21kcyxcbiAgYXBwTWFuYWdlbWVudENtZHMsXG4gIGZpbGVBY3Rpb25zQ21kcyxcbiAgbG9nQ21kcyxcbiAgLy8gYWRkIG90aGVyIGNvbW1hbmQgdHlwZXMgaGVyZVxuKTtcblxuZXhwb3J0IHsgY29tbWFuZHMgfTtcbmV4cG9ydCBkZWZhdWx0IGNvbW1hbmRzO1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7OztBQUFBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUNBOztBQUdBLElBQUlBLFFBQVEsR0FBRyxFQUFmOztBQUNBQyxNQUFNLENBQUNDLE1BQVAsQ0FDRUYsUUFERixFQUVFRyxhQUZGLEVBR0VDLGdCQUhGLEVBSUVDLGNBSkYsRUFLRUMsZ0JBTEYsRUFNRUMsZ0JBTkYsRUFPRUMsZ0JBUEYsRUFRRUMsY0FSRixFQVNFQyxZQVRGLEVBVUVDLGdCQVZGLEVBV0VDLGlCQVhGLEVBWUVDLHFCQVpGLEVBYUVDLGVBYkYsRUFjRUMscUJBZEYsRUFlRUMsb0JBZkYsRUFnQkVDLGdCQWhCRixFQWlCRUMsY0FqQkYsRUFrQkVDLG1CQWxCRixFQW1CRUMsbUJBbkJGLEVBb0JFQyxzQkFwQkYsRUFxQkVDLG9CQXJCRixFQXNCRUMsWUF0QkY7ZUEyQmV2QixRIn0=
63
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGliL2NvbW1hbmRzL2luZGV4LmpzIiwibmFtZXMiOlsiY29tbWFuZHMiLCJPYmplY3QiLCJhc3NpZ24iLCJmaW5kQ21kcyIsImdlbmVyYWxDbWRzIiwiYWxlcnRDbWRzIiwiZWxlbWVudENtZHMiLCJjb250ZXh0Q21kcyIsImFjdGlvbkNtZHMiLCJ0b3VjaENtZHMiLCJpbWVDbWRzIiwibmV0d29ya0NtZHMiLCJjb3ZlcmFnZUNtZHMiLCJyZWNvcmRzY3JlZW5DbWRzIiwiaW50ZW50Q21kcyIsInNjcmVlblN0cmVhbUNtZHMiLCJwZXJmb3JtYW5jZUNtZHMiLCJleGVjdXRlQ21kcyIsInNoZWxsQ21kcyIsImVtdUNvbnNvbGVDbWRzIiwic3lzdGVtQmFyc0NtZHMiLCJhcHBNYW5hZ2VtZW50Q21kcyIsImZpbGVBY3Rpb25zQ21kcyIsImxvZ0NtZHMiLCJtZWRpYVByb2plY3Rpb25DbWRzIl0sInNvdXJjZVJvb3QiOiIuLi8uLi8uLiIsInNvdXJjZXMiOlsibGliL2NvbW1hbmRzL2luZGV4LmpzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBmaW5kQ21kcyBmcm9tICcuL2ZpbmQnO1xuaW1wb3J0IGdlbmVyYWxDbWRzIGZyb20gJy4vZ2VuZXJhbCc7XG5pbXBvcnQgYWxlcnRDbWRzIGZyb20gJy4vYWxlcnQnO1xuaW1wb3J0IGVsZW1lbnRDbWRzIGZyb20gJy4vZWxlbWVudCc7XG5pbXBvcnQgY29udGV4dENtZHMgZnJvbSAnLi9jb250ZXh0JztcbmltcG9ydCBhY3Rpb25DbWRzIGZyb20gJy4vYWN0aW9ucyc7XG5pbXBvcnQgdG91Y2hDbWRzIGZyb20gJy4vdG91Y2gnO1xuaW1wb3J0IGltZUNtZHMgZnJvbSAnLi9pbWUnO1xuaW1wb3J0IG5ldHdvcmtDbWRzIGZyb20gJy4vbmV0d29yayc7XG5pbXBvcnQgY292ZXJhZ2VDbWRzIGZyb20gJy4vY292ZXJhZ2UnO1xuaW1wb3J0IHJlY29yZHNjcmVlbkNtZHMgZnJvbSAnLi9yZWNvcmRzY3JlZW4nO1xuaW1wb3J0IHNjcmVlblN0cmVhbUNtZHMgZnJvbSAnLi9zdHJlYW1zY3JlZW4nO1xuaW1wb3J0IHBlcmZvcm1hbmNlQ21kcyBmcm9tICcuL3BlcmZvcm1hbmNlJztcbmltcG9ydCBleGVjdXRlQ21kcyBmcm9tICcuL2V4ZWN1dGUnO1xuaW1wb3J0IHNoZWxsQ21kcyBmcm9tICcuL3NoZWxsJztcbmltcG9ydCBlbXVDb25zb2xlQ21kcyBmcm9tICcuL2VtdS1jb25zb2xlJztcbmltcG9ydCBmaWxlQWN0aW9uc0NtZHMgZnJvbSAnLi9maWxlLWFjdGlvbnMnO1xuaW1wb3J0IGFwcE1hbmFnZW1lbnRDbWRzIGZyb20gJy4vYXBwLW1hbmFnZW1lbnQnO1xuaW1wb3J0IGludGVudENtZHMgZnJvbSAnLi9pbnRlbnQnO1xuaW1wb3J0IHN5c3RlbUJhcnNDbWRzIGZyb20gJy4vc3lzdGVtLWJhcnMnO1xuaW1wb3J0IGxvZ0NtZHMgZnJvbSAnLi9sb2cnO1xuaW1wb3J0IG1lZGlhUHJvamVjdGlvbkNtZHMgZnJvbSAnLi9tZWRpYS1wcm9qZWN0aW9uJztcblxuXG5sZXQgY29tbWFuZHMgPSB7fTtcbk9iamVjdC5hc3NpZ24oXG4gIGNvbW1hbmRzLFxuICBmaW5kQ21kcyxcbiAgZ2VuZXJhbENtZHMsXG4gIGFsZXJ0Q21kcyxcbiAgZWxlbWVudENtZHMsXG4gIGNvbnRleHRDbWRzLFxuICBhY3Rpb25DbWRzLFxuICB0b3VjaENtZHMsXG4gIGltZUNtZHMsXG4gIG5ldHdvcmtDbWRzLFxuICBjb3ZlcmFnZUNtZHMsXG4gIHJlY29yZHNjcmVlbkNtZHMsXG4gIGludGVudENtZHMsXG4gIHNjcmVlblN0cmVhbUNtZHMsXG4gIHBlcmZvcm1hbmNlQ21kcyxcbiAgZXhlY3V0ZUNtZHMsXG4gIHNoZWxsQ21kcyxcbiAgZW11Q29uc29sZUNtZHMsXG4gIHN5c3RlbUJhcnNDbWRzLFxuICBhcHBNYW5hZ2VtZW50Q21kcyxcbiAgZmlsZUFjdGlvbnNDbWRzLFxuICBsb2dDbWRzLFxuICBtZWRpYVByb2plY3Rpb25DbWRzLFxuICAvLyBhZGQgb3RoZXIgY29tbWFuZCB0eXBlcyBoZXJlXG4pO1xuXG5leHBvcnQgeyBjb21tYW5kcyB9O1xuZXhwb3J0IGRlZmF1bHQgY29tbWFuZHM7XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBR0EsSUFBSUEsUUFBUSxHQUFHLEVBQWY7O0FBQ0FDLE1BQU0sQ0FBQ0MsTUFBUCxDQUNFRixRQURGLEVBRUVHLGFBRkYsRUFHRUMsZ0JBSEYsRUFJRUMsY0FKRixFQUtFQyxnQkFMRixFQU1FQyxnQkFORixFQU9FQyxnQkFQRixFQVFFQyxjQVJGLEVBU0VDLFlBVEYsRUFVRUMsZ0JBVkYsRUFXRUMsaUJBWEYsRUFZRUMscUJBWkYsRUFhRUMsZUFiRixFQWNFQyxxQkFkRixFQWVFQyxvQkFmRixFQWdCRUMsZ0JBaEJGLEVBaUJFQyxjQWpCRixFQWtCRUMsbUJBbEJGLEVBbUJFQyxtQkFuQkYsRUFvQkVDLHNCQXBCRixFQXFCRUMsb0JBckJGLEVBc0JFQyxZQXRCRixFQXVCRUMsd0JBdkJGO2VBNEJleEIsUSJ9
@@ -0,0 +1,243 @@
1
+ "use strict";
2
+
3
+ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
+
5
+ Object.defineProperty(exports, "__esModule", {
6
+ value: true
7
+ });
8
+ exports.default = exports.commands = void 0;
9
+
10
+ require("source-map-support/register");
11
+
12
+ var _lodash = _interopRequireDefault(require("lodash"));
13
+
14
+ var _asyncbox = require("asyncbox");
15
+
16
+ var _support = require("appium/support");
17
+
18
+ var _path = _interopRequireDefault(require("path"));
19
+
20
+ var _bluebird = _interopRequireDefault(require("bluebird"));
21
+
22
+ var _androidHelpers = require("../android-helpers");
23
+
24
+ var _moment = _interopRequireDefault(require("moment"));
25
+
26
+ const commands = {};
27
+ exports.commands = commands;
28
+ const DEFAULT_EXT = '.mp4';
29
+ const RECORDING_STARTUP_TIMEOUT_MS = 3 * 1000;
30
+ const RECORDING_STOP_TIMEOUT_MS = 3 * 1000;
31
+ const MIN_API_LEVEL = 29;
32
+ const RECORDING_SERVICE_NAME = `${_androidHelpers.SETTINGS_HELPER_PKG_ID}/.recorder.RecorderService`;
33
+ const RECORDING_ACTIVITY_NAME = `${_androidHelpers.SETTINGS_HELPER_PKG_ID}/io.appium.settings.Settings`;
34
+ const RECORDING_ACTION_START = `${_androidHelpers.SETTINGS_HELPER_PKG_ID}.recording.ACTION_START`;
35
+ const RECORDING_ACTION_STOP = `${_androidHelpers.SETTINGS_HELPER_PKG_ID}.recording.ACTION_STOP`;
36
+ const RECORDINGS_ROOT = `/storage/emulated/0/Android/data/${_androidHelpers.SETTINGS_HELPER_PKG_ID}/files`;
37
+ const DEFAULT_FILENAME_FORMAT = 'YYYY-MM-DDTHH-mm-ss';
38
+
39
+ async function uploadRecordedMedia(localFile, remotePath = null, uploadOptions = {}) {
40
+ if (_lodash.default.isEmpty(remotePath)) {
41
+ return (await _support.util.toInMemoryBase64(localFile)).toString();
42
+ }
43
+
44
+ const {
45
+ user,
46
+ pass,
47
+ method,
48
+ headers,
49
+ fileFieldName,
50
+ formFields
51
+ } = uploadOptions;
52
+ const options = {
53
+ method: method || 'PUT',
54
+ headers,
55
+ fileFieldName,
56
+ formFields
57
+ };
58
+
59
+ if (user && pass) {
60
+ options.auth = {
61
+ user,
62
+ pass
63
+ };
64
+ }
65
+
66
+ await _support.net.uploadFile(localFile, remotePath, options);
67
+ return '';
68
+ }
69
+
70
+ function adjustMediaExtension(name) {
71
+ return _lodash.default.toLower(name).endsWith(DEFAULT_EXT) ? name : `${name}${DEFAULT_EXT}`;
72
+ }
73
+
74
+ async function verifyMediaProjectionRecordingIsSupported(adb) {
75
+ const apiLevel = await adb.getApiLevel();
76
+
77
+ if (apiLevel < MIN_API_LEVEL) {
78
+ throw new Error(`Media projection-based recording is not available on API Level ${apiLevel}. ` + `Minimum required API Level is ${MIN_API_LEVEL}.`);
79
+ }
80
+ }
81
+
82
+ class MediaProjectionRecorder {
83
+ constructor(adb) {
84
+ this.adb = adb;
85
+ }
86
+
87
+ async isRunning() {
88
+ const stdout = await this.adb.shell(['dumpsys', 'activity', 'services', RECORDING_SERVICE_NAME]);
89
+ return stdout.includes(RECORDING_SERVICE_NAME);
90
+ }
91
+
92
+ async start(opts = {}) {
93
+ if (await this.isRunning()) {
94
+ return false;
95
+ }
96
+
97
+ await this.cleanup();
98
+ const {
99
+ filename,
100
+ maxDurationSec,
101
+ priority,
102
+ resolution
103
+ } = opts;
104
+ const args = ['am', 'start', '-n', RECORDING_ACTIVITY_NAME, '-a', RECORDING_ACTION_START];
105
+
106
+ if (filename) {
107
+ args.push('--es', 'filename', filename);
108
+ }
109
+
110
+ if (maxDurationSec) {
111
+ args.push('--es', 'max_duration_sec', `${maxDurationSec}`);
112
+ }
113
+
114
+ if (priority) {
115
+ args.push('--es', 'priority', priority);
116
+ }
117
+
118
+ if (resolution) {
119
+ args.push('--es', 'resolution', resolution);
120
+ }
121
+
122
+ await this.adb.shell(args);
123
+ await new _bluebird.default((resolve, reject) => {
124
+ setTimeout(async () => {
125
+ if (!(await this.isRunning())) {
126
+ return reject(new Error(`The media projection recording is not running after ${RECORDING_STARTUP_TIMEOUT_MS}ms. ` + `Please check the logcat output for more details.`));
127
+ }
128
+
129
+ resolve();
130
+ }, RECORDING_STARTUP_TIMEOUT_MS);
131
+ });
132
+ return true;
133
+ }
134
+
135
+ async cleanup() {
136
+ await this.adb.shell([`rm -f ${RECORDINGS_ROOT}/*`]);
137
+ }
138
+
139
+ async pullRecent() {
140
+ const recordings = await this.adb.ls(RECORDINGS_ROOT, ['-tr']);
141
+
142
+ if (_lodash.default.isEmpty(recordings)) {
143
+ return null;
144
+ }
145
+
146
+ const dstPath = _path.default.join(await _support.tempDir.openDir(), recordings[0]);
147
+
148
+ await this.adb.pull(`${RECORDINGS_ROOT}/${recordings[0]}`, dstPath);
149
+ return dstPath;
150
+ }
151
+
152
+ async stop() {
153
+ if (!(await this.isRunning())) {
154
+ return false;
155
+ }
156
+
157
+ await this.adb.shell(['am', 'start', '-n', RECORDING_ACTIVITY_NAME, '-a', RECORDING_ACTION_STOP]);
158
+
159
+ try {
160
+ await (0, _asyncbox.waitForCondition)(async () => !(await this.isRunning()), {
161
+ waitMs: RECORDING_STOP_TIMEOUT_MS,
162
+ intervalMs: 500
163
+ });
164
+ } catch (e) {
165
+ throw new Error(`The attempt to stop the current media projection recording timed out after ` + `${RECORDING_STOP_TIMEOUT_MS}ms`);
166
+ }
167
+
168
+ return true;
169
+ }
170
+
171
+ }
172
+
173
+ commands.mobileStartMediaProjectionRecording = async function mobileStartMediaProjectionRecording(options = {}) {
174
+ await verifyMediaProjectionRecordingIsSupported(this.adb);
175
+ const {
176
+ resolution,
177
+ priority,
178
+ maxDurationSec,
179
+ filename
180
+ } = options;
181
+ const recorder = new MediaProjectionRecorder(this.adb);
182
+ const fname = adjustMediaExtension(filename || (0, _moment.default)().format(DEFAULT_FILENAME_FORMAT));
183
+ const didStart = await recorder.start({
184
+ resolution,
185
+ priority,
186
+ maxDurationSec,
187
+ filename: fname
188
+ });
189
+
190
+ if (didStart) {
191
+ this.log.info(`A new media projection recording '${fname}' has been successfully started`);
192
+ } else {
193
+ this.log.info('Another media projection recording is already in progress. There is nothing to start');
194
+ }
195
+
196
+ return didStart;
197
+ };
198
+
199
+ commands.mobileIsMediaProjectionRecordingRunning = async function mobileIsMediaProjectionRecordingRunning() {
200
+ await verifyMediaProjectionRecordingIsSupported(this.adb);
201
+ const recorder = new MediaProjectionRecorder(this.adb);
202
+ return await recorder.isRunning();
203
+ };
204
+
205
+ commands.mobileStopMediaProjectionRecording = async function mobileStopMediaProjectionRecording(options = {}) {
206
+ await verifyMediaProjectionRecordingIsSupported(this.adb);
207
+ const recorder = new MediaProjectionRecorder(this.adb);
208
+
209
+ if (await recorder.stop()) {
210
+ this.log.info('Successfully stopped a media projection recording. Pulling the recorded media');
211
+ } else {
212
+ this.log.info('Media projection recording is not running. There is nothing to stop');
213
+ }
214
+
215
+ const recentRecordingPath = await recorder.pullRecent();
216
+
217
+ if (!recentRecordingPath) {
218
+ throw new Error(`No recent media projection recording have been found. Did you start any?`);
219
+ }
220
+
221
+ const {
222
+ remotePath
223
+ } = options;
224
+
225
+ if (_lodash.default.isEmpty(remotePath)) {
226
+ const {
227
+ size
228
+ } = await _support.fs.stat(recentRecordingPath);
229
+ this.log.debug(`The size of the resulting media projection recording is ${_support.util.toReadableSizeString(size)}`);
230
+ }
231
+
232
+ try {
233
+ return await uploadRecordedMedia(recentRecordingPath, remotePath, options);
234
+ } finally {
235
+ await _support.fs.rimraf(_path.default.dirname(recentRecordingPath));
236
+ }
237
+ };
238
+
239
+ var _default = commands;
240
+ exports.default = _default;require('source-map-support').install();
241
+
242
+
243
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"file":"lib/commands/media-projection.js","names":["commands","DEFAULT_EXT","RECORDING_STARTUP_TIMEOUT_MS","RECORDING_STOP_TIMEOUT_MS","MIN_API_LEVEL","RECORDING_SERVICE_NAME","SETTINGS_HELPER_PKG_ID","RECORDING_ACTIVITY_NAME","RECORDING_ACTION_START","RECORDING_ACTION_STOP","RECORDINGS_ROOT","DEFAULT_FILENAME_FORMAT","uploadRecordedMedia","localFile","remotePath","uploadOptions","_","isEmpty","util","toInMemoryBase64","toString","user","pass","method","headers","fileFieldName","formFields","options","auth","net","uploadFile","adjustMediaExtension","name","toLower","endsWith","verifyMediaProjectionRecordingIsSupported","adb","apiLevel","getApiLevel","Error","MediaProjectionRecorder","constructor","isRunning","stdout","shell","includes","start","opts","cleanup","filename","maxDurationSec","priority","resolution","args","push","B","resolve","reject","setTimeout","pullRecent","recordings","ls","dstPath","path","join","tempDir","openDir","pull","stop","waitForCondition","waitMs","intervalMs","e","mobileStartMediaProjectionRecording","recorder","fname","moment","format","didStart","log","info","mobileIsMediaProjectionRecordingRunning","mobileStopMediaProjectionRecording","recentRecordingPath","size","fs","stat","debug","toReadableSizeString","rimraf","dirname"],"sourceRoot":"../../..","sources":["lib/commands/media-projection.js"],"sourcesContent":["import _ from 'lodash';\nimport { waitForCondition } from 'asyncbox';\nimport { util, fs, net, tempDir } from 'appium/support';\nimport path from 'path';\nimport B from 'bluebird';\nimport { SETTINGS_HELPER_PKG_ID } from '../android-helpers';\nimport moment from 'moment';\n\n\nconst commands = {};\n\n// https://github.com/appium/io.appium.settings#internal-audio--video-recording\nconst DEFAULT_EXT = '.mp4';\nconst RECORDING_STARTUP_TIMEOUT_MS = 3 * 1000;\nconst RECORDING_STOP_TIMEOUT_MS = 3 * 1000;\nconst MIN_API_LEVEL = 29;\nconst RECORDING_SERVICE_NAME = `${SETTINGS_HELPER_PKG_ID}/.recorder.RecorderService`;\nconst RECORDING_ACTIVITY_NAME = `${SETTINGS_HELPER_PKG_ID}/io.appium.settings.Settings`;\nconst RECORDING_ACTION_START = `${SETTINGS_HELPER_PKG_ID}.recording.ACTION_START`;\nconst RECORDING_ACTION_STOP = `${SETTINGS_HELPER_PKG_ID}.recording.ACTION_STOP`;\nconst RECORDINGS_ROOT = `/storage/emulated/0/Android/data/${SETTINGS_HELPER_PKG_ID}/files`;\nconst DEFAULT_FILENAME_FORMAT = 'YYYY-MM-DDTHH-mm-ss';\n\n\nasync function uploadRecordedMedia (localFile, remotePath = null, uploadOptions = {}) {\n  if (_.isEmpty(remotePath)) {\n    return (await util.toInMemoryBase64(localFile)).toString();\n  }\n\n  const {user, pass, method, headers, fileFieldName, formFields} = uploadOptions;\n  const options = {\n    method: method || 'PUT',\n    headers,\n    fileFieldName,\n    formFields,\n  };\n  if (user && pass) {\n    options.auth = {user, pass};\n  }\n  await net.uploadFile(localFile, remotePath, options);\n  return '';\n}\n\nfunction adjustMediaExtension (name) {\n  return _.toLower(name).endsWith(DEFAULT_EXT) ? name : `${name}${DEFAULT_EXT}`;\n}\n\nasync function verifyMediaProjectionRecordingIsSupported (adb) {\n  const apiLevel = await adb.getApiLevel();\n  if (apiLevel < MIN_API_LEVEL) {\n    throw new Error(`Media projection-based recording is not available on API Level ${apiLevel}. ` +\n      `Minimum required API Level is ${MIN_API_LEVEL}.`);\n  }\n}\n\n\nclass MediaProjectionRecorder {\n  constructor (adb) {\n    this.adb = adb;\n  }\n\n  async isRunning () {\n    const stdout = await this.adb.shell([\n      'dumpsys', 'activity', 'services', RECORDING_SERVICE_NAME\n    ]);\n    return stdout.includes(RECORDING_SERVICE_NAME);\n  }\n\n  async start (opts = {}) {\n    if (await this.isRunning()) {\n      return false;\n    }\n\n    await this.cleanup();\n    const {\n      filename,\n      maxDurationSec,\n      priority,\n      resolution,\n    } = opts;\n    const args = [\n      'am', 'start',\n      '-n', RECORDING_ACTIVITY_NAME,\n      '-a', RECORDING_ACTION_START,\n    ];\n    if (filename) {\n      args.push('--es', 'filename', filename);\n    }\n    if (maxDurationSec) {\n      args.push('--es', 'max_duration_sec', `${maxDurationSec}`);\n    }\n    if (priority) {\n      args.push('--es', 'priority', priority);\n    }\n    if (resolution) {\n      args.push('--es', 'resolution', resolution);\n    }\n    await this.adb.shell(args);\n    await new B((resolve, reject) => {\n      setTimeout(async () => {\n        if (!await this.isRunning()) {\n          return reject(new Error(\n            `The media projection recording is not running after ${RECORDING_STARTUP_TIMEOUT_MS}ms. ` +\n            `Please check the logcat output for more details.`\n          ));\n        }\n        resolve();\n      }, RECORDING_STARTUP_TIMEOUT_MS);\n    });\n    return true;\n  }\n\n  async cleanup () {\n    await this.adb.shell([`rm -f ${RECORDINGS_ROOT}/*`]);\n  }\n\n  async pullRecent () {\n    const recordings = await this.adb.ls(RECORDINGS_ROOT, ['-tr']);\n    if (_.isEmpty(recordings)) {\n      return null;\n    }\n\n    const dstPath = path.join(await tempDir.openDir(), recordings[0]);\n    await this.adb.pull(`${RECORDINGS_ROOT}/${recordings[0]}`, dstPath);\n    return dstPath;\n  }\n\n  async stop () {\n    if (!await this.isRunning()) {\n      return false;\n    }\n\n    await this.adb.shell([\n      'am', 'start',\n      '-n', RECORDING_ACTIVITY_NAME,\n      '-a', RECORDING_ACTION_STOP,\n    ]);\n    try {\n      await waitForCondition(async () => !(await this.isRunning()), {\n        waitMs: RECORDING_STOP_TIMEOUT_MS,\n        intervalMs: 500,\n      });\n    } catch (e) {\n      throw new Error(\n        `The attempt to stop the current media projection recording timed out after ` +\n        `${RECORDING_STOP_TIMEOUT_MS}ms`\n      );\n    }\n    return true;\n  }\n}\n\n\n/**\n * @typedef {Object} StartRecordingOptions\n *\n * @property {string?} resolution Maximum supported resolution on-device (Detected\n * automatically by the app itself), which usually equals to Full HD 1920x1080 on most\n * phones however you can change it to following supported resolutions\n * as well: \"1920x1080\", \"1280x720\", \"720x480\", \"320x240\", \"176x144\".\n * @property {number?} maxDurationSec [900] Default value: 900 seconds which means\n * maximum allowed duration is 15 minute, you can increase it if your test takes\n * longer than that.\n * @property {string?} priority [high] Means recording thread priority is maximum\n * however if you face performance drops during testing with recording enabled, you\n * can reduce recording priority to \"normal\" or \"low\".\n * @property {string?} filename You can type recording video file name as you want,\n * but recording currently supports only \"mp4\" format so your filename must end with \".mp4\".\n * An invalid file name will fail to start the recording.\n * If not provided then the current timestamp will be used as file name.\n */\n\n/**\n * Record the display of a real devices running Android 10 (API level 29) and higher.\n * The screen activity is recorded to a MPEG-4 file. Audio is also recorded by default\n * (only for apps that allow it in their manifests).\n * If another recording has been already started then the command will exit silently.\n * The previously recorded video file is deleted when a new recording session is started.\n * Recording continues it is stopped explicitly or until the timeout happens.\n *\n * @param {?StartRecordingOptions} options Available options.\n * @returns {boolean} True if a new recording has successfully started.\n * @throws {Error} If recording has failed to start or is not supported on the device under test.\n */\ncommands.mobileStartMediaProjectionRecording = async function mobileStartMediaProjectionRecording (options = {}) {\n  await verifyMediaProjectionRecordingIsSupported(this.adb);\n\n  const {resolution, priority, maxDurationSec, filename} = options;\n  const recorder = new MediaProjectionRecorder(this.adb);\n  const fname = adjustMediaExtension(filename || moment().format(DEFAULT_FILENAME_FORMAT));\n  const didStart = await recorder.start({\n    resolution,\n    priority,\n    maxDurationSec,\n    filename: fname,\n  });\n  if (didStart) {\n    this.log.info(`A new media projection recording '${fname}' has been successfully started`);\n  } else {\n    this.log.info('Another media projection recording is already in progress. There is nothing to start');\n  }\n  return didStart;\n};\n\n/**\n * Checks if a media projection-based recording is currently running.\n *\n * @returns {boolean} True if a recording is in progress.\n * @throws {Error} If a recording is not supported on the device under test.\n */\ncommands.mobileIsMediaProjectionRecordingRunning = async function mobileIsMediaProjectionRecordingRunning () {\n  await verifyMediaProjectionRecordingIsSupported(this.adb);\n\n  const recorder = new MediaProjectionRecorder(this.adb);\n  return await recorder.isRunning();\n};\n\n/**\n * @typedef {Object} StopRecordingOptions\n *\n * @property {string?} remotePath The path to the remote location, where the resulting video should be uploaded.\n * The following protocols are supported: http/https, ftp.\n * Null or empty string value (the default setting) means the content of resulting\n * file should be encoded as Base64 and passed as the endpoont response value.\n * An exception will be thrown if the generated media file is too big to\n * fit into the available process memory.\n * @property {string?} user The name of the user for the remote authentication.\n * @property {string?} pass The password for the remote authentication.\n * @property {string?} method The http multipart upload method name. The 'PUT' one is used by default.\n * @property {Object?} headers Additional headers mapping for multipart http(s) uploads\n * @property {string?} fileFieldName [file] The name of the form field, where the file content BLOB should be stored for\n * http(s) uploads\n * @property {Object|Array<Pair>?} formFields Additional form fields for multipart http(s) uploads\n */\n\n/**\n * Stop a media projection-based recording.\n * If no recording has been started before then an error is thrown.\n * If the recording has been already finished before this API has been called\n * then the most recent recorded file is returned.\n *\n * @param {?StopRecordingOptions} options Available options.\n * @returns {string} Base64-encoded content of the recorded media file if 'remotePath'\n * parameter is falsy or an empty string.\n * @throws {Error} If there was an error while stopping a recording,\n * fetching the content of the remote media file,\n * or if a recording is not supported on the device under test.\n */\ncommands.mobileStopMediaProjectionRecording = async function mobileStopMediaProjectionRecording (options = {}) {\n  await verifyMediaProjectionRecordingIsSupported(this.adb);\n\n  const recorder = new MediaProjectionRecorder(this.adb);\n  if (await recorder.stop()) {\n    this.log.info('Successfully stopped a media projection recording. Pulling the recorded media');\n  } else {\n    this.log.info('Media projection recording is not running. There is nothing to stop');\n  }\n  const recentRecordingPath = await recorder.pullRecent();\n  if (!recentRecordingPath) {\n    throw new Error(`No recent media projection recording have been found. Did you start any?`);\n  }\n\n  const {remotePath} = options;\n  if (_.isEmpty(remotePath)) {\n    const {size} = await fs.stat(recentRecordingPath);\n    this.log.debug(`The size of the resulting media projection recording is ${util.toReadableSizeString(size)}`);\n  }\n  try {\n    return await uploadRecordedMedia(recentRecordingPath, remotePath, options);\n  } finally {\n    await fs.rimraf(path.dirname(recentRecordingPath));\n  }\n};\n\n\nexport { commands };\nexport default commands;\n"],"mappings":"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"}
@@ -571,7 +571,7 @@ helpers.pushSettingsApp = async function pushSettingsApp (adb, throwError = fals
571
571
  'will raise an error if you try to use them.');
572
572
  }
573
573
 
574
- // Reinstall will stop the settings helper process anyway, so
574
+ // Reinstall would stop the settings helper process anyway, so
575
575
  // there is no need to continue if the application is still running
576
576
  if (await adb.processExists(SETTINGS_HELPER_PKG_ID)) {
577
577
  logger.debug(`${SETTINGS_HELPER_PKG_ID} is already running. ` +
@@ -579,7 +579,16 @@ helpers.pushSettingsApp = async function pushSettingsApp (adb, throwError = fals
579
579
  return;
580
580
  }
581
581
 
582
- if (await adb.getApiLevel() <= 23) { // Android 6- devices should have granted permissions
582
+ const apiLevel = await adb.getApiLevel();
583
+ if (apiLevel >= 29) {
584
+ // https://github.com/appium/io.appium.settings#internal-audio--video-recording
585
+ try {
586
+ await adb.shell(['appops', 'set', SETTINGS_HELPER_PKG_ID, 'PROJECT_MEDIA', 'allow']);
587
+ } catch (err) {
588
+ logger.debug(err);
589
+ }
590
+ }
591
+ if (apiLevel <= 23) { // Android 6- devices should have granted permissions
583
592
  // https://github.com/appium/appium/pull/11640#issuecomment-438260477
584
593
  const perms = ['SET_ANIMATION_SCALE', 'CHANGE_CONFIGURATION', 'ACCESS_FINE_LOCATION'];
585
594
  logger.info(`Granting permissions ${perms} to '${SETTINGS_HELPER_PKG_ID}'`);
@@ -5,6 +5,7 @@ import { APP_STATE } from '../android-helpers';
5
5
  import { errors } from 'appium/driver';
6
6
 
7
7
  const APP_EXTENSIONS = ['.apk', '.apks'];
8
+ const RESOLVER_ACTIVITY_NAME = 'android/com.android.internal.app.ResolverActivity';
8
9
 
9
10
  let commands = {};
10
11
 
@@ -77,6 +78,16 @@ commands.activateApp = async function activateApp (appId) {
77
78
  return;
78
79
  }
79
80
 
81
+ let activityName = await this.adb.resolveLaunchableActivity(appId);
82
+ if (activityName === RESOLVER_ACTIVITY_NAME) {
83
+ // https://github.com/appium/appium/issues/17128
84
+ this.log.debug(
85
+ `The launchable activity name of '${appId}' was resolved to '${activityName}'. ` +
86
+ `Switching the resolver to not use cmd`
87
+ );
88
+ activityName = await this.adb.resolveLaunchableActivity(appId, {preferCmd: false});
89
+ }
90
+
80
91
  const stdout = await this.adb.shell([
81
92
  'am', (apiLevel < 26) ? 'start' : 'start-activity',
82
93
  '-a', 'android.intent.action.MAIN',
@@ -85,7 +96,7 @@ commands.activateApp = async function activateApp (appId) {
85
96
  // https://developer.android.com/reference/android/content/Intent#FLAG_ACTIVITY_NEW_TASK
86
97
  // https://developer.android.com/reference/android/content/Intent#FLAG_ACTIVITY_RESET_TASK_IF_NEEDED
87
98
  '-f', '0x10200000',
88
- '-n', await this.adb.resolveLaunchableActivity(appId),
99
+ '-n', activityName,
89
100
  ]);
90
101
  this.log.debug(stdout);
91
102
  if (/^error:/mi.test(stdout)) {
@@ -60,6 +60,10 @@ extensions.executeMobile = async function executeMobile (mobileCommand, opts = {
60
60
  unlock: 'mobileUnlock',
61
61
 
62
62
  refreshGpsCache: 'mobileRefreshGpsCache',
63
+
64
+ startMediaProjectionRecording: 'mobileStartMediaProjectionRecording',
65
+ isMediaProjectionRecordingRunning: 'mobileIsMediaProjectionRecordingRunning',
66
+ stopMediaProjectionRecording: 'mobileStopMediaProjectionRecording',
63
67
  };
64
68
 
65
69
  if (!_.has(mobileCommandsMapping, mobileCommand)) {
@@ -19,6 +19,7 @@ import appManagementCmds from './app-management';
19
19
  import intentCmds from './intent';
20
20
  import systemBarsCmds from './system-bars';
21
21
  import logCmds from './log';
22
+ import mediaProjectionCmds from './media-projection';
22
23
 
23
24
 
24
25
  let commands = {};
@@ -45,6 +46,7 @@ Object.assign(
45
46
  appManagementCmds,
46
47
  fileActionsCmds,
47
48
  logCmds,
49
+ mediaProjectionCmds,
48
50
  // add other command types here
49
51
  );
50
52
 
@@ -0,0 +1,277 @@
1
+ import _ from 'lodash';
2
+ import { waitForCondition } from 'asyncbox';
3
+ import { util, fs, net, tempDir } from 'appium/support';
4
+ import path from 'path';
5
+ import B from 'bluebird';
6
+ import { SETTINGS_HELPER_PKG_ID } from '../android-helpers';
7
+ import moment from 'moment';
8
+
9
+
10
+ const commands = {};
11
+
12
+ // https://github.com/appium/io.appium.settings#internal-audio--video-recording
13
+ const DEFAULT_EXT = '.mp4';
14
+ const RECORDING_STARTUP_TIMEOUT_MS = 3 * 1000;
15
+ const RECORDING_STOP_TIMEOUT_MS = 3 * 1000;
16
+ const MIN_API_LEVEL = 29;
17
+ const RECORDING_SERVICE_NAME = `${SETTINGS_HELPER_PKG_ID}/.recorder.RecorderService`;
18
+ const RECORDING_ACTIVITY_NAME = `${SETTINGS_HELPER_PKG_ID}/io.appium.settings.Settings`;
19
+ const RECORDING_ACTION_START = `${SETTINGS_HELPER_PKG_ID}.recording.ACTION_START`;
20
+ const RECORDING_ACTION_STOP = `${SETTINGS_HELPER_PKG_ID}.recording.ACTION_STOP`;
21
+ const RECORDINGS_ROOT = `/storage/emulated/0/Android/data/${SETTINGS_HELPER_PKG_ID}/files`;
22
+ const DEFAULT_FILENAME_FORMAT = 'YYYY-MM-DDTHH-mm-ss';
23
+
24
+
25
+ async function uploadRecordedMedia (localFile, remotePath = null, uploadOptions = {}) {
26
+ if (_.isEmpty(remotePath)) {
27
+ return (await util.toInMemoryBase64(localFile)).toString();
28
+ }
29
+
30
+ const {user, pass, method, headers, fileFieldName, formFields} = uploadOptions;
31
+ const options = {
32
+ method: method || 'PUT',
33
+ headers,
34
+ fileFieldName,
35
+ formFields,
36
+ };
37
+ if (user && pass) {
38
+ options.auth = {user, pass};
39
+ }
40
+ await net.uploadFile(localFile, remotePath, options);
41
+ return '';
42
+ }
43
+
44
+ function adjustMediaExtension (name) {
45
+ return _.toLower(name).endsWith(DEFAULT_EXT) ? name : `${name}${DEFAULT_EXT}`;
46
+ }
47
+
48
+ async function verifyMediaProjectionRecordingIsSupported (adb) {
49
+ const apiLevel = await adb.getApiLevel();
50
+ if (apiLevel < MIN_API_LEVEL) {
51
+ throw new Error(`Media projection-based recording is not available on API Level ${apiLevel}. ` +
52
+ `Minimum required API Level is ${MIN_API_LEVEL}.`);
53
+ }
54
+ }
55
+
56
+
57
+ class MediaProjectionRecorder {
58
+ constructor (adb) {
59
+ this.adb = adb;
60
+ }
61
+
62
+ async isRunning () {
63
+ const stdout = await this.adb.shell([
64
+ 'dumpsys', 'activity', 'services', RECORDING_SERVICE_NAME
65
+ ]);
66
+ return stdout.includes(RECORDING_SERVICE_NAME);
67
+ }
68
+
69
+ async start (opts = {}) {
70
+ if (await this.isRunning()) {
71
+ return false;
72
+ }
73
+
74
+ await this.cleanup();
75
+ const {
76
+ filename,
77
+ maxDurationSec,
78
+ priority,
79
+ resolution,
80
+ } = opts;
81
+ const args = [
82
+ 'am', 'start',
83
+ '-n', RECORDING_ACTIVITY_NAME,
84
+ '-a', RECORDING_ACTION_START,
85
+ ];
86
+ if (filename) {
87
+ args.push('--es', 'filename', filename);
88
+ }
89
+ if (maxDurationSec) {
90
+ args.push('--es', 'max_duration_sec', `${maxDurationSec}`);
91
+ }
92
+ if (priority) {
93
+ args.push('--es', 'priority', priority);
94
+ }
95
+ if (resolution) {
96
+ args.push('--es', 'resolution', resolution);
97
+ }
98
+ await this.adb.shell(args);
99
+ await new B((resolve, reject) => {
100
+ setTimeout(async () => {
101
+ if (!await this.isRunning()) {
102
+ return reject(new Error(
103
+ `The media projection recording is not running after ${RECORDING_STARTUP_TIMEOUT_MS}ms. ` +
104
+ `Please check the logcat output for more details.`
105
+ ));
106
+ }
107
+ resolve();
108
+ }, RECORDING_STARTUP_TIMEOUT_MS);
109
+ });
110
+ return true;
111
+ }
112
+
113
+ async cleanup () {
114
+ await this.adb.shell([`rm -f ${RECORDINGS_ROOT}/*`]);
115
+ }
116
+
117
+ async pullRecent () {
118
+ const recordings = await this.adb.ls(RECORDINGS_ROOT, ['-tr']);
119
+ if (_.isEmpty(recordings)) {
120
+ return null;
121
+ }
122
+
123
+ const dstPath = path.join(await tempDir.openDir(), recordings[0]);
124
+ await this.adb.pull(`${RECORDINGS_ROOT}/${recordings[0]}`, dstPath);
125
+ return dstPath;
126
+ }
127
+
128
+ async stop () {
129
+ if (!await this.isRunning()) {
130
+ return false;
131
+ }
132
+
133
+ await this.adb.shell([
134
+ 'am', 'start',
135
+ '-n', RECORDING_ACTIVITY_NAME,
136
+ '-a', RECORDING_ACTION_STOP,
137
+ ]);
138
+ try {
139
+ await waitForCondition(async () => !(await this.isRunning()), {
140
+ waitMs: RECORDING_STOP_TIMEOUT_MS,
141
+ intervalMs: 500,
142
+ });
143
+ } catch (e) {
144
+ throw new Error(
145
+ `The attempt to stop the current media projection recording timed out after ` +
146
+ `${RECORDING_STOP_TIMEOUT_MS}ms`
147
+ );
148
+ }
149
+ return true;
150
+ }
151
+ }
152
+
153
+
154
+ /**
155
+ * @typedef {Object} StartRecordingOptions
156
+ *
157
+ * @property {string?} resolution Maximum supported resolution on-device (Detected
158
+ * automatically by the app itself), which usually equals to Full HD 1920x1080 on most
159
+ * phones however you can change it to following supported resolutions
160
+ * as well: "1920x1080", "1280x720", "720x480", "320x240", "176x144".
161
+ * @property {number?} maxDurationSec [900] Default value: 900 seconds which means
162
+ * maximum allowed duration is 15 minute, you can increase it if your test takes
163
+ * longer than that.
164
+ * @property {string?} priority [high] Means recording thread priority is maximum
165
+ * however if you face performance drops during testing with recording enabled, you
166
+ * can reduce recording priority to "normal" or "low".
167
+ * @property {string?} filename You can type recording video file name as you want,
168
+ * but recording currently supports only "mp4" format so your filename must end with ".mp4".
169
+ * An invalid file name will fail to start the recording.
170
+ * If not provided then the current timestamp will be used as file name.
171
+ */
172
+
173
+ /**
174
+ * Record the display of a real devices running Android 10 (API level 29) and higher.
175
+ * The screen activity is recorded to a MPEG-4 file. Audio is also recorded by default
176
+ * (only for apps that allow it in their manifests).
177
+ * If another recording has been already started then the command will exit silently.
178
+ * The previously recorded video file is deleted when a new recording session is started.
179
+ * Recording continues it is stopped explicitly or until the timeout happens.
180
+ *
181
+ * @param {?StartRecordingOptions} options Available options.
182
+ * @returns {boolean} True if a new recording has successfully started.
183
+ * @throws {Error} If recording has failed to start or is not supported on the device under test.
184
+ */
185
+ commands.mobileStartMediaProjectionRecording = async function mobileStartMediaProjectionRecording (options = {}) {
186
+ await verifyMediaProjectionRecordingIsSupported(this.adb);
187
+
188
+ const {resolution, priority, maxDurationSec, filename} = options;
189
+ const recorder = new MediaProjectionRecorder(this.adb);
190
+ const fname = adjustMediaExtension(filename || moment().format(DEFAULT_FILENAME_FORMAT));
191
+ const didStart = await recorder.start({
192
+ resolution,
193
+ priority,
194
+ maxDurationSec,
195
+ filename: fname,
196
+ });
197
+ if (didStart) {
198
+ this.log.info(`A new media projection recording '${fname}' has been successfully started`);
199
+ } else {
200
+ this.log.info('Another media projection recording is already in progress. There is nothing to start');
201
+ }
202
+ return didStart;
203
+ };
204
+
205
+ /**
206
+ * Checks if a media projection-based recording is currently running.
207
+ *
208
+ * @returns {boolean} True if a recording is in progress.
209
+ * @throws {Error} If a recording is not supported on the device under test.
210
+ */
211
+ commands.mobileIsMediaProjectionRecordingRunning = async function mobileIsMediaProjectionRecordingRunning () {
212
+ await verifyMediaProjectionRecordingIsSupported(this.adb);
213
+
214
+ const recorder = new MediaProjectionRecorder(this.adb);
215
+ return await recorder.isRunning();
216
+ };
217
+
218
+ /**
219
+ * @typedef {Object} StopRecordingOptions
220
+ *
221
+ * @property {string?} remotePath The path to the remote location, where the resulting video should be uploaded.
222
+ * The following protocols are supported: http/https, ftp.
223
+ * Null or empty string value (the default setting) means the content of resulting
224
+ * file should be encoded as Base64 and passed as the endpoont response value.
225
+ * An exception will be thrown if the generated media file is too big to
226
+ * fit into the available process memory.
227
+ * @property {string?} user The name of the user for the remote authentication.
228
+ * @property {string?} pass The password for the remote authentication.
229
+ * @property {string?} method The http multipart upload method name. The 'PUT' one is used by default.
230
+ * @property {Object?} headers Additional headers mapping for multipart http(s) uploads
231
+ * @property {string?} fileFieldName [file] The name of the form field, where the file content BLOB should be stored for
232
+ * http(s) uploads
233
+ * @property {Object|Array<Pair>?} formFields Additional form fields for multipart http(s) uploads
234
+ */
235
+
236
+ /**
237
+ * Stop a media projection-based recording.
238
+ * If no recording has been started before then an error is thrown.
239
+ * If the recording has been already finished before this API has been called
240
+ * then the most recent recorded file is returned.
241
+ *
242
+ * @param {?StopRecordingOptions} options Available options.
243
+ * @returns {string} Base64-encoded content of the recorded media file if 'remotePath'
244
+ * parameter is falsy or an empty string.
245
+ * @throws {Error} If there was an error while stopping a recording,
246
+ * fetching the content of the remote media file,
247
+ * or if a recording is not supported on the device under test.
248
+ */
249
+ commands.mobileStopMediaProjectionRecording = async function mobileStopMediaProjectionRecording (options = {}) {
250
+ await verifyMediaProjectionRecordingIsSupported(this.adb);
251
+
252
+ const recorder = new MediaProjectionRecorder(this.adb);
253
+ if (await recorder.stop()) {
254
+ this.log.info('Successfully stopped a media projection recording. Pulling the recorded media');
255
+ } else {
256
+ this.log.info('Media projection recording is not running. There is nothing to stop');
257
+ }
258
+ const recentRecordingPath = await recorder.pullRecent();
259
+ if (!recentRecordingPath) {
260
+ throw new Error(`No recent media projection recording have been found. Did you start any?`);
261
+ }
262
+
263
+ const {remotePath} = options;
264
+ if (_.isEmpty(remotePath)) {
265
+ const {size} = await fs.stat(recentRecordingPath);
266
+ this.log.debug(`The size of the resulting media projection recording is ${util.toReadableSizeString(size)}`);
267
+ }
268
+ try {
269
+ return await uploadRecordedMedia(recentRecordingPath, remotePath, options);
270
+ } finally {
271
+ await fs.rimraf(path.dirname(recentRecordingPath));
272
+ }
273
+ };
274
+
275
+
276
+ export { commands };
277
+ export default commands;
package/package.json CHANGED
@@ -9,7 +9,7 @@
9
9
  "mobile",
10
10
  "mobile testing"
11
11
  ],
12
- "version": "5.1.0",
12
+ "version": "5.2.2",
13
13
  "author": "appium",
14
14
  "license": "Apache-2.0",
15
15
  "repository": {
@@ -36,12 +36,12 @@
36
36
  ],
37
37
  "dependencies": {
38
38
  "@babel/runtime": "^7.0.0",
39
- "appium-adb": "^9.6.0",
39
+ "appium-adb": "^9.9.0",
40
40
  "appium-chromedriver": "^5.0.1",
41
41
  "asyncbox": "^2.8.0",
42
42
  "axios": "^0.x",
43
43
  "bluebird": "^3.4.7",
44
- "io.appium.settings": "^4.0.0",
44
+ "io.appium.settings": "^4.1.0",
45
45
  "jimp": "^0.x",
46
46
  "lodash": "^4.17.4",
47
47
  "lru-cache": "^7.3.0",
@@ -85,7 +85,6 @@
85
85
  "@babel/eslint-parser": "^7.16.3",
86
86
  "@semantic-release/git": "^10.0.1",
87
87
  "@xmldom/xmldom": "^0.x",
88
- "appium": "^2.0.0-beta.40",
89
88
  "android-apidemos": "^4.0.0",
90
89
  "chai": "^4.1.2",
91
90
  "chai-as-promised": "^7.1.1",