appium-android-driver 5.2.12 → 5.4.1

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.
@@ -19,15 +19,24 @@ var _androidHelpers = require("../android-helpers");
19
19
 
20
20
  var _driver = require("appium/driver");
21
21
 
22
+ var _utils = require("../utils");
23
+
22
24
  const APP_EXTENSIONS = ['.apk', '.apks'];
23
25
  const RESOLVER_ACTIVITY_NAME = 'android/com.android.internal.app.ResolverActivity';
24
- let commands = {};
26
+ const commands = {};
25
27
  exports.commands = commands;
26
28
 
27
29
  commands.isAppInstalled = async function isAppInstalled(appId) {
28
30
  return await this.adb.isAppInstalled(appId);
29
31
  };
30
32
 
33
+ commands.mobileIsAppInstalled = async function mobileIsAppInstalled(opts = {}) {
34
+ const {
35
+ appId
36
+ } = (0, _utils.requireArgs)('appId', opts);
37
+ return await this.isAppInstalled(appId);
38
+ };
39
+
31
40
  commands.queryAppState = async function queryAppState(appId) {
32
41
  this.log.info(`Querying the state of '${appId}'`);
33
42
 
@@ -50,6 +59,13 @@ commands.queryAppState = async function queryAppState(appId) {
50
59
  return _androidHelpers.APP_STATE.RUNNING_IN_BACKGROUND;
51
60
  };
52
61
 
62
+ commands.mobileQueryAppState = async function mobileQueryAppState(opts = {}) {
63
+ const {
64
+ appId
65
+ } = (0, _utils.requireArgs)('appId', opts);
66
+ return await this.queryAppState(appId);
67
+ };
68
+
53
69
  commands.activateApp = async function activateApp(appId) {
54
70
  this.log.debug(`Activating '${appId}'`);
55
71
  const apiLevel = await this.adb.getApiLevel();
@@ -89,10 +105,24 @@ commands.activateApp = async function activateApp(appId) {
89
105
  }
90
106
  };
91
107
 
108
+ commands.mobileActivateApp = async function mobileActivateApp(opts = {}) {
109
+ const {
110
+ appId
111
+ } = (0, _utils.requireArgs)('appId', opts);
112
+ return await this.activateApp(appId);
113
+ };
114
+
92
115
  commands.removeApp = async function removeApp(appId, options = {}) {
93
116
  return await this.adb.uninstallApk(appId, options);
94
117
  };
95
118
 
119
+ commands.mobileRemoveApp = async function mobileRemoveApp(opts = {}) {
120
+ const {
121
+ appId
122
+ } = (0, _utils.requireArgs)('appId', opts);
123
+ return await this.removeApp(appId, opts);
124
+ };
125
+
96
126
  commands.terminateApp = async function terminateApp(appId, options = {}) {
97
127
  this.log.info(`Terminating '${appId}'`);
98
128
 
@@ -117,11 +147,25 @@ commands.terminateApp = async function terminateApp(appId, options = {}) {
117
147
  return true;
118
148
  };
119
149
 
150
+ commands.mobileTerminateApp = async function mobileTerminateApp(opts = {}) {
151
+ const {
152
+ appId
153
+ } = (0, _utils.requireArgs)('appId', opts);
154
+ return await this.terminateApp(appId, opts);
155
+ };
156
+
120
157
  commands.installApp = async function installApp(appPath, options = {}) {
121
158
  const localPath = await this.helpers.configureApp(appPath, APP_EXTENSIONS);
122
159
  await this.adb.install(localPath, options);
123
160
  };
124
161
 
162
+ commands.mobileInstallApp = async function mobileInstallApp(opts = {}) {
163
+ const {
164
+ appPath
165
+ } = (0, _utils.requireArgs)('appPath', opts);
166
+ return await this.installApp(appPath, opts);
167
+ };
168
+
125
169
  commands.mobileClearApp = async function mobileClearApp(opts = {}) {
126
170
  const {
127
171
  appId
@@ -136,4 +180,4 @@ commands.mobileClearApp = async function mobileClearApp(opts = {}) {
136
180
 
137
181
  var _default = commands;
138
182
  exports.default = _default;
139
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"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"],"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":";;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AACA;;AAEA,MAAMA,cAAc,GAAG,CAAC,MAAD,EAAS,OAAT,CAAvB;AACA,MAAMC,sBAAsB,GAAG,mDAA/B;AAEA,IAAIC,QAAQ,GAAG,EAAf;;;AAQAA,QAAQ,CAACC,cAAT,GAA0B,eAAeA,cAAf,CAA+BC,KAA/B,EAAsC;EAC9D,OAAO,MAAM,KAAKC,GAAL,CAASF,cAAT,CAAwBC,KAAxB,CAAb;AACD,CAFD;;AAeAF,QAAQ,CAACI,aAAT,GAAyB,eAAeA,aAAf,CAA8BF,KAA9B,EAAqC;EAC5D,KAAKG,GAAL,CAASC,IAAT,CAAe,0BAAyBJ,KAAM,GAA9C;;EACA,IAAI,EAAC,MAAM,KAAKC,GAAL,CAASF,cAAT,CAAwBC,KAAxB,CAAP,CAAJ,EAA2C;IACzC,OAAOK,yBAAA,CAAUC,aAAjB;EACD;;EACD,IAAI,EAAC,MAAM,KAAKL,GAAL,CAASM,aAAT,CAAuBP,KAAvB,CAAP,CAAJ,EAA0C;IACxC,OAAOK,yBAAA,CAAUG,WAAjB;EACD;;EACD,MAAMC,OAAO,GAAG,IAAIC,MAAJ,CAAY,MAAKC,eAAA,CAAEC,YAAF,CAAeZ,KAAf,CAAsB,GAAvC,CAAhB;;EACA,KAAK,MAAMa,IAAX,IAAmB,CAAC,MAAM,KAAKZ,GAAL,CAASa,WAAT,EAAP,EAA+BC,KAA/B,CAAqC,IAArC,CAAnB,EAA+D;IAC7D,IAAIN,OAAO,CAACO,IAAR,CAAaH,IAAb,KAAsB,CAAC,eAAD,EAAkB,aAAlB,EAAiCI,IAAjC,CAAuCC,CAAD,IAAOL,IAAI,CAACM,QAAL,CAAcD,CAAd,CAA7C,CAA1B,EAA0F;MACxF,OAAOb,yBAAA,CAAUe,qBAAjB;IACD;EACF;;EACD,OAAOf,yBAAA,CAAUgB,qBAAjB;AACD,CAfD;;AAwBAvB,QAAQ,CAACwB,WAAT,GAAuB,eAAeA,WAAf,CAA4BtB,KAA5B,EAAmC;EACxD,KAAKG,GAAL,CAASoB,KAAT,CAAgB,eAAcvB,KAAM,GAApC;EACA,MAAMwB,QAAQ,GAAG,MAAM,KAAKvB,GAAL,CAASwB,WAAT,EAAvB;;EAEA,IAAID,QAAQ,GAAG,EAAf,EAAmB;IAGjB,MAAME,GAAG,GAAG,CAAC,QAAD,EACV,IADU,EACJ1B,KADI,EAEV,IAFU,EAEJ,kCAFI,EAGV,GAHU,CAAZ;IAIA,IAAI2B,MAAM,GAAG,EAAb;;IACA,IAAI;MACFA,MAAM,GAAG,MAAM,KAAK1B,GAAL,CAAS2B,KAAT,CAAeF,GAAf,CAAf;MACA,KAAKvB,GAAL,CAASoB,KAAT,CAAgB,mBAAkBI,MAAO,EAAzC;IACD,CAHD,CAGE,OAAOE,CAAP,EAAU;MACV,KAAK1B,GAAL,CAAS2B,aAAT,CAAwB,oBAAmB9B,KAAM,sBAAqB6B,CAAC,CAACE,OAAQ,EAAhF;IACD;;IACD,IAAIJ,MAAM,CAACR,QAAP,CAAgB,gBAAhB,CAAJ,EAAuC;MACrC,KAAKhB,GAAL,CAAS2B,aAAT,CAAwB,oBAAmB9B,KAAM,kCAAjD;IACD;;IACD;EACD;;EAED,IAAIgC,YAAY,GAAG,MAAM,KAAK/B,GAAL,CAASgC,yBAAT,CAAmCjC,KAAnC,CAAzB;;EACA,IAAIgC,YAAY,KAAKnC,sBAArB,EAA6C;IAE3C,KAAKM,GAAL,CAASoB,KAAT,CACG,oCAAmCvB,KAAM,sBAAqBgC,YAAa,KAA5E,GACC,uCAFH;IAIAA,YAAY,GAAG,MAAM,KAAK/B,GAAL,CAASgC,yBAAT,CAAmCjC,KAAnC,EAA0C;MAACkC,SAAS,EAAE;IAAZ,CAA1C,CAArB;EACD;;EAED,MAAMC,MAAM,GAAG,MAAM,KAAKlC,GAAL,CAAS2B,KAAT,CAAe,CAClC,IADkC,EAC3BJ,QAAQ,GAAG,EAAZ,GAAkB,OAAlB,GAA4B,gBADA,EAElC,IAFkC,EAE5B,4BAF4B,EAGlC,IAHkC,EAG5B,kCAH4B,EAOlC,IAPkC,EAO5B,YAP4B,EAQlC,IARkC,EAQ5BQ,YAR4B,CAAf,CAArB;EAUA,KAAK7B,GAAL,CAASoB,KAAT,CAAeY,MAAf;;EACA,IAAI,YAAYnB,IAAZ,CAAiBmB,MAAjB,CAAJ,EAA8B;IAC5B,MAAM,IAAIC,KAAJ,CAAW,oBAAmBpC,KAAM,sBAAqBmC,MAAO,EAAhE,CAAN;EACD;AACF,CAhDD;;AAmEArC,QAAQ,CAACuC,SAAT,GAAqB,eAAeA,SAAf,CAA0BrC,KAA1B,EAAiCsC,OAAO,GAAG,EAA3C,EAA+C;EAClE,OAAO,MAAM,KAAKrC,GAAL,CAASsC,YAAT,CAAsBvC,KAAtB,EAA6BsC,OAA7B,CAAb;AACD,CAFD;;AAkBAxC,QAAQ,CAAC0C,YAAT,GAAwB,eAAeA,YAAf,CAA6BxC,KAA7B,EAAoCsC,OAAO,GAAG,EAA9C,EAAkD;EACxE,KAAKnC,GAAL,CAASC,IAAT,CAAe,gBAAeJ,KAAM,GAApC;;EACA,IAAI,EAAE,MAAM,KAAKC,GAAL,CAASM,aAAT,CAAuBP,KAAvB,CAAR,CAAJ,EAA4C;IAC1C,KAAKG,GAAL,CAASC,IAAT,CAAe,YAAWJ,KAAM,kBAAhC;IACA,OAAO,KAAP;EACD;;EACD,MAAM,KAAKC,GAAL,CAASwC,SAAT,CAAmBzC,KAAnB,CAAN;EACA,MAAM0C,OAAO,GAAGC,aAAA,CAAKC,QAAL,CAAcN,OAAO,CAACI,OAAtB,KAAkC,CAACG,KAAK,CAACP,OAAO,CAACI,OAAT,CAAxC,GAA4DI,QAAQ,CAACR,OAAO,CAACI,OAAT,EAAkB,EAAlB,CAApE,GAA4F,GAA5G;;EACA,IAAI;IACF,MAAM,IAAAK,0BAAA,EAAiB,YAAY,OAAM,KAAK7C,aAAL,CAAmBF,KAAnB,CAAN,KAAmCK,yBAAA,CAAUG,WAA1E,EACJ;MAACwC,MAAM,EAAEN,OAAT;MAAkBO,UAAU,EAAE;IAA9B,CADI,CAAN;EAED,CAHD,CAGE,OAAOpB,CAAP,EAAU;IACV,KAAK1B,GAAL,CAAS2B,aAAT,CAAwB,IAAG9B,KAAM,4BAA2B0C,OAAQ,YAApE;EACD;;EACD,KAAKvC,GAAL,CAASC,IAAT,CAAe,IAAGJ,KAAM,oCAAxB;EACA,OAAO,IAAP;AACD,CAhBD;;AA0CAF,QAAQ,CAACoD,UAAT,GAAsB,eAAeA,UAAf,CAA2BC,OAA3B,EAAoCb,OAAO,GAAG,EAA9C,EAAkD;EACtE,MAAMc,SAAS,GAAG,MAAM,KAAKC,OAAL,CAAaC,YAAb,CAA0BH,OAA1B,EAAmCvD,cAAnC,CAAxB;EACA,MAAM,KAAKK,GAAL,CAASsD,OAAT,CAAiBH,SAAjB,EAA4Bd,OAA5B,CAAN;AACD,CAHD;;AAgBAxC,QAAQ,CAAC0D,cAAT,GAA0B,eAAeA,cAAf,CAA+BC,IAAI,GAAG,EAAtC,EAA0C;EAClE,MAAM;IAACzD;EAAD,IAAUyD,IAAhB;;EACA,IAAI,CAACzD,KAAL,EAAY;IACV,MAAM,IAAI0D,cAAA,CAAOC,oBAAX,CAAiC,kCAAjC,CAAN;EACD;;EACD,MAAM,KAAK1D,GAAL,CAAS2D,KAAT,CAAe5D,KAAf,CAAN;AACD,CAND;;eASeF,Q"}
183
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["APP_EXTENSIONS","RESOLVER_ACTIVITY_NAME","commands","isAppInstalled","appId","adb","mobileIsAppInstalled","opts","requireArgs","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","mobileQueryAppState","activateApp","debug","apiLevel","getApiLevel","cmd","output","shell","e","errorAndThrow","message","activityName","resolveLaunchableActivity","preferCmd","stdout","Error","mobileActivateApp","removeApp","options","uninstallApk","mobileRemoveApp","terminateApp","forceStop","timeout","util","hasValue","isNaN","parseInt","waitForCondition","waitMs","intervalMs","mobileTerminateApp","installApp","appPath","localPath","helpers","configureApp","install","mobileInstallApp","mobileClearApp","errors","InvalidArgumentError","clear"],"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';\nimport { requireArgs } from '../utils';\n\nconst APP_EXTENSIONS = ['.apk', '.apks'];\nconst RESOLVER_ACTIVITY_NAME = 'android/com.android.internal.app.ResolverActivity';\n\nconst 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 * @typedef {Object} MobileAppInstalledOptions\n * @property {string} appId - Application package identifier. Must be always provided.\n */\n\n/**\n * Verify whether an application is installed or not\n *\n * @param {MobileAppInstalledOptions} opts\n * @returns {boolean} Same as in `isAppInstalled`\n */\ncommands.mobileIsAppInstalled = async function mobileIsAppInstalled (opts = {}) {\n  const { appId } = requireArgs('appId', opts);\n  return await this.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 * @typedef {Object} MobileQueryAppStateOptions\n * @property {string} appId - Application package identifier. Must be always provided.\n */\n\n/**\n * Queries the current state of the app.\n *\n * @param {MobileQueryAppStateOptions} opts\n * @returns {number} Same as in `queryAppState`\n */\ncommands.mobileQueryAppState = async function mobileQueryAppState (opts = {}) {\n  const { appId } = requireArgs('appId', opts);\n  return await this.queryAppState(appId);\n};\n\n/**\n * Activates the given application or launches it if necessary.\n * The action literally simulates\n * clicking the corresponding application icon on the dashboard.\n *\n * @param {string} appId - Application package identifier\n * @throws {Error} If the app cannot be activated\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} MobileActivateAppOptions\n * @property {string} appId - Application package identifier. Must be always provided.\n */\n\n/**\n * Activates the given application or launches it if necessary.\n * The action literally simulates\n * clicking the corresponding application icon on the dashboard.\n *\n * @param {MobileActivateAppOptions} opts\n * @throws {Error} If the app cannot be activated\n */\ncommands.mobileActivateApp = async function mobileActivateApp (opts = {}) {\n  const { appId } = requireArgs('appId', opts);\n  return await this.activateApp(appId);\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} MobileRemoveAppOptions\n * @property {string} appId - Application package identifier. Must be always provided.\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 {MobileRemoveAppOptions} opts\n * @returns {boolean} Same as in `removeApp`\n */\ncommands.mobileRemoveApp = async function mobileRemoveApp (opts = {}) {\n  const { appId } = requireArgs('appId', opts);\n  return await this.removeApp(appId, opts);\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} MobileTerminateAppOptions\n * @property {string} appId - Application package identifier. Must be always provided.\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 {MobileTerminateAppOptions} opts\n * @returns {boolean} Same as in `terminateApp`\n * @throws {Error} if the app has not been terminated within the given timeout.\n */\ncommands.mobileTerminateApp = async function mobileTerminateApp (opts = {}) {\n  const { appId } = requireArgs('appId', opts);\n  return await this.terminateApp(appId, opts);\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} MobileInstallAppOptions\n * @property {string} appPath - The local apk path or a remote url. Must be always provided.\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 {MobileInstallAppOptions} opts\n * @throws {Error} if the given apk does not exist or is not reachable\n */\ncommands.mobileInstallApp = async function mobileInstallApp (opts = {}) {\n  const { appPath } = requireArgs('appPath', opts);\n  return await this.installApp(appPath, opts);\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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"}
@@ -48,7 +48,16 @@ extensions.executeMobile = async function executeMobile(mobileCommand, opts = {}
48
48
  stopScreenStreaming: 'mobileStopScreenStreaming',
49
49
  getNotifications: 'mobileGetNotifications',
50
50
  listSms: 'mobileListSms',
51
+ pushFile: 'mobilePushFile',
52
+ pullFile: 'mobilePullFile',
53
+ pullFolder: 'mobilePullFolder',
51
54
  deleteFile: 'mobileDeleteFile',
55
+ isAppInstalled: 'mobileIsAppInstalled',
56
+ queryAppState: 'mobileQueryAppState',
57
+ activateApp: 'mobileActivateApp',
58
+ removeApp: 'mobileRemoveApp',
59
+ terminateApp: 'mobileTerminateApp',
60
+ installApp: 'mobileInstallApp',
52
61
  clearApp: 'mobileClearApp',
53
62
  startService: 'mobileStartService',
54
63
  stopService: 'mobileStopService',
@@ -71,4 +80,4 @@ extensions.executeMobile = async function executeMobile(mobileCommand, opts = {}
71
80
 
72
81
  var _default = extensions;
73
82
  exports.default = _default;
74
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJleHRlbnNpb25zIiwiZXhlY3V0ZSIsInNjcmlwdCIsImFyZ3MiLCJtYXRjaCIsImxvZyIsImluZm8iLCJyZXBsYWNlIiwidHJpbSIsImV4ZWN1dGVNb2JpbGUiLCJfIiwiaXNBcnJheSIsImlzV2ViQ29udGV4dCIsImVycm9ycyIsIk5vdEltcGxlbWVudGVkRXJyb3IiLCJlbmRwb2ludCIsImNocm9tZWRyaXZlciIsImp3cHJveHkiLCJkb3duc3RyZWFtUHJvdG9jb2wiLCJQUk9UT0NPTFMiLCJNSlNPTldQIiwiY29tbWFuZCIsIm1vYmlsZUNvbW1hbmQiLCJvcHRzIiwibW9iaWxlQ29tbWFuZHNNYXBwaW5nIiwic2hlbGwiLCJleGVjRW11Q29uc29sZUNvbW1hbmQiLCJzdGFydExvZ3NCcm9hZGNhc3QiLCJzdG9wTG9nc0Jyb2FkY2FzdCIsImNoYW5nZVBlcm1pc3Npb25zIiwiZ2V0UGVybWlzc2lvbnMiLCJwZXJmb3JtRWRpdG9yQWN0aW9uIiwic2Vuc29yU2V0IiwiZ2V0RGV2aWNlVGltZSIsInN0YXJ0U2NyZWVuU3RyZWFtaW5nIiwic3RvcFNjcmVlblN0cmVhbWluZyIsImdldE5vdGlmaWNhdGlvbnMiLCJsaXN0U21zIiwiZGVsZXRlRmlsZSIsImNsZWFyQXBwIiwic3RhcnRTZXJ2aWNlIiwic3RvcFNlcnZpY2UiLCJzdGFydEFjdGl2aXR5IiwiYnJvYWRjYXN0IiwiZ2V0Q29udGV4dHMiLCJ1bmxvY2siLCJyZWZyZXNoR3BzQ2FjaGUiLCJzdGFydE1lZGlhUHJvamVjdGlvblJlY29yZGluZyIsImlzTWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nUnVubmluZyIsInN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmciLCJoYXMiLCJVbmtub3duQ29tbWFuZEVycm9yIiwia2V5cyJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL2xpYi9jb21tYW5kcy9leGVjdXRlLmpzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgeyBlcnJvcnMsIFBST1RPQ09MUyB9IGZyb20gJ2FwcGl1bS9kcml2ZXInO1xuXG5jb25zdCBleHRlbnNpb25zID0ge307XG5cbmV4dGVuc2lvbnMuZXhlY3V0ZSA9IGFzeW5jIGZ1bmN0aW9uIGV4ZWN1dGUgKHNjcmlwdCwgYXJncykge1xuICBpZiAoc2NyaXB0Lm1hdGNoKC9ebW9iaWxlOi8pKSB7XG4gICAgdGhpcy5sb2cuaW5mbyhgRXhlY3V0aW5nIG5hdGl2ZSBjb21tYW5kICcke3NjcmlwdH0nYCk7XG4gICAgc2NyaXB0ID0gc2NyaXB0LnJlcGxhY2UoL15tb2JpbGU6LywgJycpLnRyaW0oKTtcbiAgICByZXR1cm4gYXdhaXQgdGhpcy5leGVjdXRlTW9iaWxlKHNjcmlwdCwgXy5pc0FycmF5KGFyZ3MpID8gYXJnc1swXSA6IGFyZ3MpO1xuICB9XG4gIGlmICghdGhpcy5pc1dlYkNvbnRleHQoKSkge1xuICAgIHRocm93IG5ldyBlcnJvcnMuTm90SW1wbGVtZW50ZWRFcnJvcigpO1xuICB9XG4gIGNvbnN0IGVuZHBvaW50ID0gdGhpcy5jaHJvbWVkcml2ZXIuandwcm94eS5kb3duc3RyZWFtUHJvdG9jb2wgPT09IFBST1RPQ09MUy5NSlNPTldQXG4gICAgPyAnL2V4ZWN1dGUnXG4gICAgOiAnL2V4ZWN1dGUvc3luYyc7XG4gIHJldHVybiBhd2FpdCB0aGlzLmNocm9tZWRyaXZlci5qd3Byb3h5LmNvbW1hbmQoZW5kcG9pbnQsICdQT1NUJywge1xuICAgIHNjcmlwdCxcbiAgICBhcmdzLFxuICB9KTtcbn07XG5cbmV4dGVuc2lvbnMuZXhlY3V0ZU1vYmlsZSA9IGFzeW5jIGZ1bmN0aW9uIGV4ZWN1dGVNb2JpbGUgKG1vYmlsZUNvbW1hbmQsIG9wdHMgPSB7fSkge1xuICBjb25zdCBtb2JpbGVDb21tYW5kc01hcHBpbmcgPSB7XG4gICAgc2hlbGw6ICdtb2JpbGVTaGVsbCcsXG5cbiAgICBleGVjRW11Q29uc29sZUNvbW1hbmQ6ICdtb2JpbGVFeGVjRW11Q29uc29sZUNvbW1hbmQnLFxuXG4gICAgc3RhcnRMb2dzQnJvYWRjYXN0OiAnbW9iaWxlU3RhcnRMb2dzQnJvYWRjYXN0JyxcbiAgICBzdG9wTG9nc0Jyb2FkY2FzdDogJ21vYmlsZVN0b3BMb2dzQnJvYWRjYXN0JyxcblxuICAgIGNoYW5nZVBlcm1pc3Npb25zOiAnbW9iaWxlQ2hhbmdlUGVybWlzc2lvbnMnLFxuICAgIGdldFBlcm1pc3Npb25zOiAnbW9iaWxlR2V0UGVybWlzc2lvbnMnLFxuXG4gICAgcGVyZm9ybUVkaXRvckFjdGlvbjogJ21vYmlsZVBlcmZvcm1FZGl0b3JBY3Rpb24nLFxuXG4gICAgc2Vuc29yU2V0OiAnc2Vuc29yU2V0JyxcblxuICAgIGdldERldmljZVRpbWU6ICdtb2JpbGVHZXREZXZpY2VUaW1lJyxcblxuICAgIHN0YXJ0U2NyZWVuU3RyZWFtaW5nOiAnbW9iaWxlU3RhcnRTY3JlZW5TdHJlYW1pbmcnLFxuICAgIHN0b3BTY3JlZW5TdHJlYW1pbmc6ICdtb2JpbGVTdG9wU2NyZWVuU3RyZWFtaW5nJyxcblxuICAgIGdldE5vdGlmaWNhdGlvbnM6ICdtb2JpbGVHZXROb3RpZmljYXRpb25zJyxcblxuICAgIGxpc3RTbXM6ICdtb2JpbGVMaXN0U21zJyxcblxuICAgIGRlbGV0ZUZpbGU6ICdtb2JpbGVEZWxldGVGaWxlJyxcblxuICAgIGNsZWFyQXBwOiAnbW9iaWxlQ2xlYXJBcHAnLFxuXG4gICAgc3RhcnRTZXJ2aWNlOiAnbW9iaWxlU3RhcnRTZXJ2aWNlJyxcbiAgICBzdG9wU2VydmljZTogJ21vYmlsZVN0b3BTZXJ2aWNlJyxcbiAgICBzdGFydEFjdGl2aXR5OiAnbW9iaWxlU3RhcnRBY3Rpdml0eScsXG4gICAgYnJvYWRjYXN0OiAnbW9iaWxlQnJvYWRjYXN0JyxcblxuICAgIGdldENvbnRleHRzOiAnbW9iaWxlR2V0Q29udGV4dHMnLFxuXG4gICAgdW5sb2NrOiAnbW9iaWxlVW5sb2NrJyxcblxuICAgIHJlZnJlc2hHcHNDYWNoZTogJ21vYmlsZVJlZnJlc2hHcHNDYWNoZScsXG5cbiAgICBzdGFydE1lZGlhUHJvamVjdGlvblJlY29yZGluZzogJ21vYmlsZVN0YXJ0TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nJyxcbiAgICBpc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmc6ICdtb2JpbGVJc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmcnLFxuICAgIHN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmc6ICdtb2JpbGVTdG9wTWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nJyxcbiAgfTtcblxuICBpZiAoIV8uaGFzKG1vYmlsZUNvbW1hbmRzTWFwcGluZywgbW9iaWxlQ29tbWFuZCkpIHtcbiAgICB0aHJvdyBuZXcgZXJyb3JzLlVua25vd25Db21tYW5kRXJyb3IoYFVua25vd24gbW9iaWxlIGNvbW1hbmQgXCIke21vYmlsZUNvbW1hbmR9XCIuIGAgK1xuICAgICAgYE9ubHkgJHtfLmtleXMobW9iaWxlQ29tbWFuZHNNYXBwaW5nKX0gY29tbWFuZHMgYXJlIHN1cHBvcnRlZC5gKTtcbiAgfVxuICByZXR1cm4gYXdhaXQgdGhpc1ttb2JpbGVDb21tYW5kc01hcHBpbmdbbW9iaWxlQ29tbWFuZF1dKG9wdHMpO1xufTtcblxuZXhwb3J0IGRlZmF1bHQgZXh0ZW5zaW9ucztcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFFQSxNQUFNQSxVQUFVLEdBQUcsRUFBbkI7O0FBRUFBLFVBQVUsQ0FBQ0MsT0FBWCxHQUFxQixlQUFlQSxPQUFmLENBQXdCQyxNQUF4QixFQUFnQ0MsSUFBaEMsRUFBc0M7RUFDekQsSUFBSUQsTUFBTSxDQUFDRSxLQUFQLENBQWEsVUFBYixDQUFKLEVBQThCO0lBQzVCLEtBQUtDLEdBQUwsQ0FBU0MsSUFBVCxDQUFlLDZCQUE0QkosTUFBTyxHQUFsRDtJQUNBQSxNQUFNLEdBQUdBLE1BQU0sQ0FBQ0ssT0FBUCxDQUFlLFVBQWYsRUFBMkIsRUFBM0IsRUFBK0JDLElBQS9CLEVBQVQ7SUFDQSxPQUFPLE1BQU0sS0FBS0MsYUFBTCxDQUFtQlAsTUFBbkIsRUFBMkJRLGVBQUEsQ0FBRUMsT0FBRixDQUFVUixJQUFWLElBQWtCQSxJQUFJLENBQUMsQ0FBRCxDQUF0QixHQUE0QkEsSUFBdkQsQ0FBYjtFQUNEOztFQUNELElBQUksQ0FBQyxLQUFLUyxZQUFMLEVBQUwsRUFBMEI7SUFDeEIsTUFBTSxJQUFJQyxjQUFBLENBQU9DLG1CQUFYLEVBQU47RUFDRDs7RUFDRCxNQUFNQyxRQUFRLEdBQUcsS0FBS0MsWUFBTCxDQUFrQkMsT0FBbEIsQ0FBMEJDLGtCQUExQixLQUFpREMsaUJBQUEsQ0FBVUMsT0FBM0QsR0FDYixVQURhLEdBRWIsZUFGSjtFQUdBLE9BQU8sTUFBTSxLQUFLSixZQUFMLENBQWtCQyxPQUFsQixDQUEwQkksT0FBMUIsQ0FBa0NOLFFBQWxDLEVBQTRDLE1BQTVDLEVBQW9EO0lBQy9EYixNQUQrRDtJQUUvREM7RUFGK0QsQ0FBcEQsQ0FBYjtBQUlELENBaEJEOztBQWtCQUgsVUFBVSxDQUFDUyxhQUFYLEdBQTJCLGVBQWVBLGFBQWYsQ0FBOEJhLGFBQTlCLEVBQTZDQyxJQUFJLEdBQUcsRUFBcEQsRUFBd0Q7RUFDakYsTUFBTUMscUJBQXFCLEdBQUc7SUFDNUJDLEtBQUssRUFBRSxhQURxQjtJQUc1QkMscUJBQXFCLEVBQUUsNkJBSEs7SUFLNUJDLGtCQUFrQixFQUFFLDBCQUxRO0lBTTVCQyxpQkFBaUIsRUFBRSx5QkFOUztJQVE1QkMsaUJBQWlCLEVBQUUseUJBUlM7SUFTNUJDLGNBQWMsRUFBRSxzQkFUWTtJQVc1QkMsbUJBQW1CLEVBQUUsMkJBWE87SUFhNUJDLFNBQVMsRUFBRSxXQWJpQjtJQWU1QkMsYUFBYSxFQUFFLHFCQWZhO0lBaUI1QkMsb0JBQW9CLEVBQUUsNEJBakJNO0lBa0I1QkMsbUJBQW1CLEVBQUUsMkJBbEJPO0lBb0I1QkMsZ0JBQWdCLEVBQUUsd0JBcEJVO0lBc0I1QkMsT0FBTyxFQUFFLGVBdEJtQjtJQXdCNUJDLFVBQVUsRUFBRSxrQkF4QmdCO0lBMEI1QkMsUUFBUSxFQUFFLGdCQTFCa0I7SUE0QjVCQyxZQUFZLEVBQUUsb0JBNUJjO0lBNkI1QkMsV0FBVyxFQUFFLG1CQTdCZTtJQThCNUJDLGFBQWEsRUFBRSxxQkE5QmE7SUErQjVCQyxTQUFTLEVBQUUsaUJBL0JpQjtJQWlDNUJDLFdBQVcsRUFBRSxtQkFqQ2U7SUFtQzVCQyxNQUFNLEVBQUUsY0FuQ29CO0lBcUM1QkMsZUFBZSxFQUFFLHVCQXJDVztJQXVDNUJDLDZCQUE2QixFQUFFLHFDQXZDSDtJQXdDNUJDLGlDQUFpQyxFQUFFLHlDQXhDUDtJQXlDNUJDLDRCQUE0QixFQUFFO0VBekNGLENBQTlCOztFQTRDQSxJQUFJLENBQUN2QyxlQUFBLENBQUV3QyxHQUFGLENBQU0xQixxQkFBTixFQUE2QkYsYUFBN0IsQ0FBTCxFQUFrRDtJQUNoRCxNQUFNLElBQUlULGNBQUEsQ0FBT3NDLG1CQUFYLENBQWdDLDJCQUEwQjdCLGFBQWMsS0FBekMsR0FDbEMsUUFBT1osZUFBQSxDQUFFMEMsSUFBRixDQUFPNUIscUJBQVAsQ0FBOEIsMEJBRGxDLENBQU47RUFFRDs7RUFDRCxPQUFPLE1BQU0sS0FBS0EscUJBQXFCLENBQUNGLGFBQUQsQ0FBMUIsRUFBMkNDLElBQTNDLENBQWI7QUFDRCxDQWxERDs7ZUFvRGV2QixVIn0=
83
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJleHRlbnNpb25zIiwiZXhlY3V0ZSIsInNjcmlwdCIsImFyZ3MiLCJtYXRjaCIsImxvZyIsImluZm8iLCJyZXBsYWNlIiwidHJpbSIsImV4ZWN1dGVNb2JpbGUiLCJfIiwiaXNBcnJheSIsImlzV2ViQ29udGV4dCIsImVycm9ycyIsIk5vdEltcGxlbWVudGVkRXJyb3IiLCJlbmRwb2ludCIsImNocm9tZWRyaXZlciIsImp3cHJveHkiLCJkb3duc3RyZWFtUHJvdG9jb2wiLCJQUk9UT0NPTFMiLCJNSlNPTldQIiwiY29tbWFuZCIsIm1vYmlsZUNvbW1hbmQiLCJvcHRzIiwibW9iaWxlQ29tbWFuZHNNYXBwaW5nIiwic2hlbGwiLCJleGVjRW11Q29uc29sZUNvbW1hbmQiLCJzdGFydExvZ3NCcm9hZGNhc3QiLCJzdG9wTG9nc0Jyb2FkY2FzdCIsImNoYW5nZVBlcm1pc3Npb25zIiwiZ2V0UGVybWlzc2lvbnMiLCJwZXJmb3JtRWRpdG9yQWN0aW9uIiwic2Vuc29yU2V0IiwiZ2V0RGV2aWNlVGltZSIsInN0YXJ0U2NyZWVuU3RyZWFtaW5nIiwic3RvcFNjcmVlblN0cmVhbWluZyIsImdldE5vdGlmaWNhdGlvbnMiLCJsaXN0U21zIiwicHVzaEZpbGUiLCJwdWxsRmlsZSIsInB1bGxGb2xkZXIiLCJkZWxldGVGaWxlIiwiaXNBcHBJbnN0YWxsZWQiLCJxdWVyeUFwcFN0YXRlIiwiYWN0aXZhdGVBcHAiLCJyZW1vdmVBcHAiLCJ0ZXJtaW5hdGVBcHAiLCJpbnN0YWxsQXBwIiwiY2xlYXJBcHAiLCJzdGFydFNlcnZpY2UiLCJzdG9wU2VydmljZSIsInN0YXJ0QWN0aXZpdHkiLCJicm9hZGNhc3QiLCJnZXRDb250ZXh0cyIsInVubG9jayIsInJlZnJlc2hHcHNDYWNoZSIsInN0YXJ0TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nIiwiaXNNZWRpYVByb2plY3Rpb25SZWNvcmRpbmdSdW5uaW5nIiwic3RvcE1lZGlhUHJvamVjdGlvblJlY29yZGluZyIsImhhcyIsIlVua25vd25Db21tYW5kRXJyb3IiLCJrZXlzIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vbGliL2NvbW1hbmRzL2V4ZWN1dGUuanMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IF8gZnJvbSAnbG9kYXNoJztcbmltcG9ydCB7IGVycm9ycywgUFJPVE9DT0xTIH0gZnJvbSAnYXBwaXVtL2RyaXZlcic7XG5cbmNvbnN0IGV4dGVuc2lvbnMgPSB7fTtcblxuZXh0ZW5zaW9ucy5leGVjdXRlID0gYXN5bmMgZnVuY3Rpb24gZXhlY3V0ZSAoc2NyaXB0LCBhcmdzKSB7XG4gIGlmIChzY3JpcHQubWF0Y2goL15tb2JpbGU6LykpIHtcbiAgICB0aGlzLmxvZy5pbmZvKGBFeGVjdXRpbmcgbmF0aXZlIGNvbW1hbmQgJyR7c2NyaXB0fSdgKTtcbiAgICBzY3JpcHQgPSBzY3JpcHQucmVwbGFjZSgvXm1vYmlsZTovLCAnJykudHJpbSgpO1xuICAgIHJldHVybiBhd2FpdCB0aGlzLmV4ZWN1dGVNb2JpbGUoc2NyaXB0LCBfLmlzQXJyYXkoYXJncykgPyBhcmdzWzBdIDogYXJncyk7XG4gIH1cbiAgaWYgKCF0aGlzLmlzV2ViQ29udGV4dCgpKSB7XG4gICAgdGhyb3cgbmV3IGVycm9ycy5Ob3RJbXBsZW1lbnRlZEVycm9yKCk7XG4gIH1cbiAgY29uc3QgZW5kcG9pbnQgPSB0aGlzLmNocm9tZWRyaXZlci5qd3Byb3h5LmRvd25zdHJlYW1Qcm90b2NvbCA9PT0gUFJPVE9DT0xTLk1KU09OV1BcbiAgICA/ICcvZXhlY3V0ZSdcbiAgICA6ICcvZXhlY3V0ZS9zeW5jJztcbiAgcmV0dXJuIGF3YWl0IHRoaXMuY2hyb21lZHJpdmVyLmp3cHJveHkuY29tbWFuZChlbmRwb2ludCwgJ1BPU1QnLCB7XG4gICAgc2NyaXB0LFxuICAgIGFyZ3MsXG4gIH0pO1xufTtcblxuZXh0ZW5zaW9ucy5leGVjdXRlTW9iaWxlID0gYXN5bmMgZnVuY3Rpb24gZXhlY3V0ZU1vYmlsZSAobW9iaWxlQ29tbWFuZCwgb3B0cyA9IHt9KSB7XG4gIGNvbnN0IG1vYmlsZUNvbW1hbmRzTWFwcGluZyA9IHtcbiAgICBzaGVsbDogJ21vYmlsZVNoZWxsJyxcblxuICAgIGV4ZWNFbXVDb25zb2xlQ29tbWFuZDogJ21vYmlsZUV4ZWNFbXVDb25zb2xlQ29tbWFuZCcsXG5cbiAgICBzdGFydExvZ3NCcm9hZGNhc3Q6ICdtb2JpbGVTdGFydExvZ3NCcm9hZGNhc3QnLFxuICAgIHN0b3BMb2dzQnJvYWRjYXN0OiAnbW9iaWxlU3RvcExvZ3NCcm9hZGNhc3QnLFxuXG4gICAgY2hhbmdlUGVybWlzc2lvbnM6ICdtb2JpbGVDaGFuZ2VQZXJtaXNzaW9ucycsXG4gICAgZ2V0UGVybWlzc2lvbnM6ICdtb2JpbGVHZXRQZXJtaXNzaW9ucycsXG5cbiAgICBwZXJmb3JtRWRpdG9yQWN0aW9uOiAnbW9iaWxlUGVyZm9ybUVkaXRvckFjdGlvbicsXG5cbiAgICBzZW5zb3JTZXQ6ICdzZW5zb3JTZXQnLFxuXG4gICAgZ2V0RGV2aWNlVGltZTogJ21vYmlsZUdldERldmljZVRpbWUnLFxuXG4gICAgc3RhcnRTY3JlZW5TdHJlYW1pbmc6ICdtb2JpbGVTdGFydFNjcmVlblN0cmVhbWluZycsXG4gICAgc3RvcFNjcmVlblN0cmVhbWluZzogJ21vYmlsZVN0b3BTY3JlZW5TdHJlYW1pbmcnLFxuXG4gICAgZ2V0Tm90aWZpY2F0aW9uczogJ21vYmlsZUdldE5vdGlmaWNhdGlvbnMnLFxuXG4gICAgbGlzdFNtczogJ21vYmlsZUxpc3RTbXMnLFxuXG4gICAgcHVzaEZpbGU6ICdtb2JpbGVQdXNoRmlsZScsXG4gICAgcHVsbEZpbGU6ICdtb2JpbGVQdWxsRmlsZScsXG4gICAgcHVsbEZvbGRlcjogJ21vYmlsZVB1bGxGb2xkZXInLFxuICAgIGRlbGV0ZUZpbGU6ICdtb2JpbGVEZWxldGVGaWxlJyxcblxuICAgIGlzQXBwSW5zdGFsbGVkOiAnbW9iaWxlSXNBcHBJbnN0YWxsZWQnLFxuICAgIHF1ZXJ5QXBwU3RhdGU6ICdtb2JpbGVRdWVyeUFwcFN0YXRlJyxcbiAgICBhY3RpdmF0ZUFwcDogJ21vYmlsZUFjdGl2YXRlQXBwJyxcbiAgICByZW1vdmVBcHA6ICdtb2JpbGVSZW1vdmVBcHAnLFxuICAgIHRlcm1pbmF0ZUFwcDogJ21vYmlsZVRlcm1pbmF0ZUFwcCcsXG4gICAgaW5zdGFsbEFwcDogJ21vYmlsZUluc3RhbGxBcHAnLFxuICAgIGNsZWFyQXBwOiAnbW9iaWxlQ2xlYXJBcHAnLFxuXG4gICAgc3RhcnRTZXJ2aWNlOiAnbW9iaWxlU3RhcnRTZXJ2aWNlJyxcbiAgICBzdG9wU2VydmljZTogJ21vYmlsZVN0b3BTZXJ2aWNlJyxcbiAgICBzdGFydEFjdGl2aXR5OiAnbW9iaWxlU3RhcnRBY3Rpdml0eScsXG4gICAgYnJvYWRjYXN0OiAnbW9iaWxlQnJvYWRjYXN0JyxcblxuICAgIGdldENvbnRleHRzOiAnbW9iaWxlR2V0Q29udGV4dHMnLFxuXG4gICAgdW5sb2NrOiAnbW9iaWxlVW5sb2NrJyxcblxuICAgIHJlZnJlc2hHcHNDYWNoZTogJ21vYmlsZVJlZnJlc2hHcHNDYWNoZScsXG5cbiAgICBzdGFydE1lZGlhUHJvamVjdGlvblJlY29yZGluZzogJ21vYmlsZVN0YXJ0TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nJyxcbiAgICBpc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmc6ICdtb2JpbGVJc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmcnLFxuICAgIHN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmc6ICdtb2JpbGVTdG9wTWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nJyxcbiAgfTtcblxuICBpZiAoIV8uaGFzKG1vYmlsZUNvbW1hbmRzTWFwcGluZywgbW9iaWxlQ29tbWFuZCkpIHtcbiAgICB0aHJvdyBuZXcgZXJyb3JzLlVua25vd25Db21tYW5kRXJyb3IoYFVua25vd24gbW9iaWxlIGNvbW1hbmQgXCIke21vYmlsZUNvbW1hbmR9XCIuIGAgK1xuICAgICAgYE9ubHkgJHtfLmtleXMobW9iaWxlQ29tbWFuZHNNYXBwaW5nKX0gY29tbWFuZHMgYXJlIHN1cHBvcnRlZC5gKTtcbiAgfVxuICByZXR1cm4gYXdhaXQgdGhpc1ttb2JpbGVDb21tYW5kc01hcHBpbmdbbW9iaWxlQ29tbWFuZF1dKG9wdHMpO1xufTtcblxuZXhwb3J0IGRlZmF1bHQgZXh0ZW5zaW9ucztcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFFQSxNQUFNQSxVQUFVLEdBQUcsRUFBbkI7O0FBRUFBLFVBQVUsQ0FBQ0MsT0FBWCxHQUFxQixlQUFlQSxPQUFmLENBQXdCQyxNQUF4QixFQUFnQ0MsSUFBaEMsRUFBc0M7RUFDekQsSUFBSUQsTUFBTSxDQUFDRSxLQUFQLENBQWEsVUFBYixDQUFKLEVBQThCO0lBQzVCLEtBQUtDLEdBQUwsQ0FBU0MsSUFBVCxDQUFlLDZCQUE0QkosTUFBTyxHQUFsRDtJQUNBQSxNQUFNLEdBQUdBLE1BQU0sQ0FBQ0ssT0FBUCxDQUFlLFVBQWYsRUFBMkIsRUFBM0IsRUFBK0JDLElBQS9CLEVBQVQ7SUFDQSxPQUFPLE1BQU0sS0FBS0MsYUFBTCxDQUFtQlAsTUFBbkIsRUFBMkJRLGVBQUEsQ0FBRUMsT0FBRixDQUFVUixJQUFWLElBQWtCQSxJQUFJLENBQUMsQ0FBRCxDQUF0QixHQUE0QkEsSUFBdkQsQ0FBYjtFQUNEOztFQUNELElBQUksQ0FBQyxLQUFLUyxZQUFMLEVBQUwsRUFBMEI7SUFDeEIsTUFBTSxJQUFJQyxjQUFBLENBQU9DLG1CQUFYLEVBQU47RUFDRDs7RUFDRCxNQUFNQyxRQUFRLEdBQUcsS0FBS0MsWUFBTCxDQUFrQkMsT0FBbEIsQ0FBMEJDLGtCQUExQixLQUFpREMsaUJBQUEsQ0FBVUMsT0FBM0QsR0FDYixVQURhLEdBRWIsZUFGSjtFQUdBLE9BQU8sTUFBTSxLQUFLSixZQUFMLENBQWtCQyxPQUFsQixDQUEwQkksT0FBMUIsQ0FBa0NOLFFBQWxDLEVBQTRDLE1BQTVDLEVBQW9EO0lBQy9EYixNQUQrRDtJQUUvREM7RUFGK0QsQ0FBcEQsQ0FBYjtBQUlELENBaEJEOztBQWtCQUgsVUFBVSxDQUFDUyxhQUFYLEdBQTJCLGVBQWVBLGFBQWYsQ0FBOEJhLGFBQTlCLEVBQTZDQyxJQUFJLEdBQUcsRUFBcEQsRUFBd0Q7RUFDakYsTUFBTUMscUJBQXFCLEdBQUc7SUFDNUJDLEtBQUssRUFBRSxhQURxQjtJQUc1QkMscUJBQXFCLEVBQUUsNkJBSEs7SUFLNUJDLGtCQUFrQixFQUFFLDBCQUxRO0lBTTVCQyxpQkFBaUIsRUFBRSx5QkFOUztJQVE1QkMsaUJBQWlCLEVBQUUseUJBUlM7SUFTNUJDLGNBQWMsRUFBRSxzQkFUWTtJQVc1QkMsbUJBQW1CLEVBQUUsMkJBWE87SUFhNUJDLFNBQVMsRUFBRSxXQWJpQjtJQWU1QkMsYUFBYSxFQUFFLHFCQWZhO0lBaUI1QkMsb0JBQW9CLEVBQUUsNEJBakJNO0lBa0I1QkMsbUJBQW1CLEVBQUUsMkJBbEJPO0lBb0I1QkMsZ0JBQWdCLEVBQUUsd0JBcEJVO0lBc0I1QkMsT0FBTyxFQUFFLGVBdEJtQjtJQXdCNUJDLFFBQVEsRUFBRSxnQkF4QmtCO0lBeUI1QkMsUUFBUSxFQUFFLGdCQXpCa0I7SUEwQjVCQyxVQUFVLEVBQUUsa0JBMUJnQjtJQTJCNUJDLFVBQVUsRUFBRSxrQkEzQmdCO0lBNkI1QkMsY0FBYyxFQUFFLHNCQTdCWTtJQThCNUJDLGFBQWEsRUFBRSxxQkE5QmE7SUErQjVCQyxXQUFXLEVBQUUsbUJBL0JlO0lBZ0M1QkMsU0FBUyxFQUFFLGlCQWhDaUI7SUFpQzVCQyxZQUFZLEVBQUUsb0JBakNjO0lBa0M1QkMsVUFBVSxFQUFFLGtCQWxDZ0I7SUFtQzVCQyxRQUFRLEVBQUUsZ0JBbkNrQjtJQXFDNUJDLFlBQVksRUFBRSxvQkFyQ2M7SUFzQzVCQyxXQUFXLEVBQUUsbUJBdENlO0lBdUM1QkMsYUFBYSxFQUFFLHFCQXZDYTtJQXdDNUJDLFNBQVMsRUFBRSxpQkF4Q2lCO0lBMEM1QkMsV0FBVyxFQUFFLG1CQTFDZTtJQTRDNUJDLE1BQU0sRUFBRSxjQTVDb0I7SUE4QzVCQyxlQUFlLEVBQUUsdUJBOUNXO0lBZ0Q1QkMsNkJBQTZCLEVBQUUscUNBaERIO0lBaUQ1QkMsaUNBQWlDLEVBQUUseUNBakRQO0lBa0Q1QkMsNEJBQTRCLEVBQUU7RUFsREYsQ0FBOUI7O0VBcURBLElBQUksQ0FBQ2hELGVBQUEsQ0FBRWlELEdBQUYsQ0FBTW5DLHFCQUFOLEVBQTZCRixhQUE3QixDQUFMLEVBQWtEO0lBQ2hELE1BQU0sSUFBSVQsY0FBQSxDQUFPK0MsbUJBQVgsQ0FBZ0MsMkJBQTBCdEMsYUFBYyxLQUF6QyxHQUNsQyxRQUFPWixlQUFBLENBQUVtRCxJQUFGLENBQU9yQyxxQkFBUCxDQUE4QiwwQkFEbEMsQ0FBTjtFQUVEOztFQUNELE9BQU8sTUFBTSxLQUFLQSxxQkFBcUIsQ0FBQ0YsYUFBRCxDQUExQixFQUEyQ0MsSUFBM0MsQ0FBYjtBQUNELENBM0REOztlQTZEZXZCLFUifQ==
@@ -17,6 +17,8 @@ var _path = _interopRequireDefault(require("path"));
17
17
 
18
18
  var _driver = require("appium/driver");
19
19
 
20
+ var _utils = require("../utils");
21
+
20
22
  const CONTAINER_PATH_MARKER = '@';
21
23
  const CONTAINER_PATH_PATTERN = new RegExp(`^${CONTAINER_PATH_MARKER}([^/]+)/(.+)`);
22
24
  const ANDROID_MEDIA_RESCAN_INTENT = 'android.intent.action.MEDIA_SCANNER_SCAN_FILE';
@@ -90,6 +92,13 @@ commands.pullFile = async function pullFile(remotePath) {
90
92
  }
91
93
  };
92
94
 
95
+ commands.mobilePullFile = async function mobilePullFile(opts = {}) {
96
+ const {
97
+ remotePath
98
+ } = (0, _utils.requireArgs)('remotePath', opts);
99
+ return await this.pullFile(remotePath);
100
+ };
101
+
93
102
  commands.pushFile = async function pushFile(remotePath, base64Data) {
94
103
  if (remotePath.endsWith('/')) {
95
104
  throw new _driver.errors.InvalidArgumentError(`It is expected that remote path points to a file and not to a folder. ` + `'${remotePath}' is given instead`);
@@ -139,14 +148,32 @@ commands.pushFile = async function pushFile(remotePath, base64Data) {
139
148
  }
140
149
  };
141
150
 
151
+ commands.mobilePushFile = async function mobilePushFile(opts = {}) {
152
+ const {
153
+ remotePath,
154
+ payload
155
+ } = (0, _utils.requireArgs)(['remotePath', 'payload'], opts);
156
+ return await this.pushFile(remotePath, payload);
157
+ };
158
+
142
159
  commands.pullFolder = async function pullFolder(remotePath) {
143
- let localFolder = await _support.tempDir.path({
144
- prefix: 'appium'
145
- });
146
- await this.adb.pull(remotePath, localFolder);
147
- return (await _support.zip.toInMemoryZip(localFolder, {
148
- encodeToBase64: true
149
- })).toString();
160
+ const tmpRoot = await _support.tempDir.openDir();
161
+
162
+ try {
163
+ await this.adb.pull(remotePath, tmpRoot);
164
+ return (await _support.zip.toInMemoryZip(tmpRoot, {
165
+ encodeToBase64: true
166
+ })).toString();
167
+ } finally {
168
+ await _support.fs.rimraf(tmpRoot);
169
+ }
170
+ };
171
+
172
+ commands.mobilePullFolder = async function mobilePullFolder(opts = {}) {
173
+ const {
174
+ remotePath
175
+ } = (0, _utils.requireArgs)('remotePath', opts);
176
+ return await this.pullFolder(remotePath);
150
177
  };
151
178
 
152
179
  async function deleteFileOrFolder(adb, remotePath) {
@@ -230,4 +257,4 @@ commands.mobileDeleteFile = async function mobileDeleteFile(opts = {}) {
230
257
 
231
258
  var _default = commands;
232
259
  exports.default = _default;
233
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["CONTAINER_PATH_MARKER","CONTAINER_PATH_PATTERN","RegExp","ANDROID_MEDIA_RESCAN_INTENT","commands","parseContainerPath","remotePath","match","exec","Error","path","posix","resolve","scanMedia","adb","log","debug","getApiLevel","shell","e","warn","stderr","message","escapePath","p","replace","pullFile","endsWith","errors","InvalidArgumentError","tmpDestination","startsWith","packageId","pathInContainer","basename","errorAndThrow","localFile","tempDir","prefix","suffix","pull","util","toInMemoryBase64","toString","fs","exists","unlink","pushFile","base64Data","_","isArray","Buffer","from","content","writeFile","dirname","push","pullFolder","localFolder","zip","toInMemoryZip","encodeToBase64","deleteFileOrFolder","performRemoteFsCheck","op","runAs","passFlag","checkCmd","fullCmd","includes","ign","isFile","isDir","isPresent","dstPath","pkgId","info","expectsFile","mobileDeleteFile","opts"],"sources":["../../../lib/commands/file-actions.js"],"sourcesContent":["import _ from 'lodash';\nimport { fs, util, zip, tempDir } from 'appium/support';\nimport path from 'path';\nimport { errors } from 'appium/driver';\n\n\nconst CONTAINER_PATH_MARKER = '@';\n// https://regex101.com/r/PLdB0G/2\nconst CONTAINER_PATH_PATTERN = new RegExp(`^${CONTAINER_PATH_MARKER}([^/]+)/(.+)`);\nconst ANDROID_MEDIA_RESCAN_INTENT = 'android.intent.action.MEDIA_SCANNER_SCAN_FILE';\n\n\nconst commands = {};\n\n/**\n * Parses the actual destination path from the given value\n *\n * @param {string} remotePath The preformatted remote path, which looks like\n * `@my.app.id/my/path`\n * @returns {Array<string>} An array, where the first item is the parsed package\n * identifier and the second one is the actual destination path inside the package.\n * @throws {Error} If the given string cannot be parsed\n */\nfunction parseContainerPath (remotePath) {\n  const match = CONTAINER_PATH_PATTERN.exec(remotePath);\n  if (!match) {\n    throw new Error(`It is expected that package identifier is separated from the relative path with a single slash. ` +\n      `'${remotePath}' is given instead`);\n  }\n  return [match[1], path.posix.resolve(`/data/data/${match[1]}`, match[2])];\n}\n\n/**\n * Scans the given file/folder on the remote device\n * and adds matching items to the device's media library.\n * Exceptions are ignored and written into the log.\n *\n * @param {ADB} adb ADB instance\n * @param {Object?} log Logger instance\n * @param {string} remotePath The file/folder path on the remote device\n */\nasync function scanMedia (adb, remotePath, log = null) {\n  log?.debug(`Performing media scan of '${remotePath}'`);\n  try {\n    // https://github.com/appium/appium/issues/16184\n    if (await adb.getApiLevel() >= 29) {\n      await adb.scanMedia(remotePath);\n    } else {\n      await adb.shell([\n        'am', 'broadcast',\n        '-a', ANDROID_MEDIA_RESCAN_INTENT,\n        '-d', `file://${remotePath}`\n      ]);\n    }\n  } catch (e) {\n    log?.warn(`Ignoring an unexpected error upon media scanning of '${remotePath}': ${e.stderr || e.message}`);\n  }\n}\n\n/**\n * A small helper, which escapes single quotes in paths,\n * so they are safe to be passed as arguments of shell commands\n *\n * @param {string} p The initial remote path\n * @returns {string} The escaped path value\n */\nfunction escapePath (p) {\n  return p.replace(/'/g, `\\\\'`);\n}\n\n/**\n * Pulls a remote file from the device.\n * It is required, that a package has debugging flag enabled\n * in order to access its files.\n *\n * @param {string} remotePath The full path to the remote file\n * or a specially formatted path, which points to an item inside app bundle\n * @returns {string} Base64 encoded content of the pulled file\n * @throws {Error} If the pull operation failed\n */\ncommands.pullFile = async function pullFile (remotePath) {\n  if (remotePath.endsWith('/')) {\n    throw new errors.InvalidArgumentError(`It is expected that remote path points to a file and not to a folder. ` +\n      `'${remotePath}' is given instead`);\n  }\n  let tmpDestination = null;\n  if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n    const [packageId, pathInContainer] = parseContainerPath(remotePath);\n    this.log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'. Will get the data from '${pathInContainer}'`);\n    tmpDestination = `/data/local/tmp/${path.posix.basename(pathInContainer)}`;\n    try {\n      await this.adb.shell(['run-as', packageId, `chmod 777 '${escapePath(pathInContainer)}'`]);\n      await this.adb.shell([\n        'run-as', packageId,\n        `cp -f '${escapePath(pathInContainer)}' '${escapePath(tmpDestination)}'`\n      ]);\n    } catch (e) {\n      this.log.errorAndThrow(`Cannot access the container of '${packageId}' application. ` +\n                        `Is the application installed and has 'debuggable' build option set to true? ` +\n                        `Original error: ${e.message}`);\n    }\n  }\n  const localFile = await tempDir.path({prefix: 'appium', suffix: '.tmp'});\n  try {\n    await this.adb.pull(tmpDestination || remotePath, localFile);\n    return (await util.toInMemoryBase64(localFile)).toString();\n  } finally {\n    if (await fs.exists(localFile)) {\n      await fs.unlink(localFile);\n    }\n    if (tmpDestination) {\n      await this.adb.shell(['rm', '-f', tmpDestination]);\n    }\n  }\n};\n\n/**\n * Pushed the given data to a file on the remote device\n * It is required, that a package has debugging flag enabled\n * in order to access its files.\n *\n * @param {string} remotePath The full path to the remote file or\n * a file inside a package bundle\n * @param {string} base64Data Base64 encoded data to be written to the\n * remote file. The remote file will be silently overridden if it already exists.\n * @throws {Error} If there was an error while pushing the data\n */\ncommands.pushFile = async function pushFile (remotePath, base64Data) {\n  if (remotePath.endsWith('/')) {\n    throw new errors.InvalidArgumentError(\n      `It is expected that remote path points to a file and not to a folder. ` +\n      `'${remotePath}' is given instead`\n    );\n  }\n  const localFile = await tempDir.path({prefix: 'appium', suffix: '.tmp'});\n  if (_.isArray(base64Data)) {\n    // some clients (ahem) java, send a byte array encoding utf8 characters\n    // instead of a string, which would be infinitely better!\n    base64Data = Buffer.from(base64Data).toString('utf8');\n  }\n  const content = Buffer.from(base64Data, 'base64');\n  let tmpDestination = null;\n  try {\n    await fs.writeFile(localFile, content.toString('binary'), 'binary');\n    if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n      const [packageId, pathInContainer] = parseContainerPath(remotePath);\n      this.log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'. ` +\n        `Will put the data into '${pathInContainer}'`);\n      tmpDestination = `/data/local/tmp/${path.posix.basename(pathInContainer)}`;\n      try {\n        await this.adb.shell(\n          ['run-as', packageId, `mkdir -p '${escapePath(path.posix.dirname(pathInContainer))}'`]\n        );\n        await this.adb.shell(['run-as', packageId, `touch '${escapePath(pathInContainer)}'`]);\n        await this.adb.shell(['run-as', packageId, `chmod 777 '${escapePath(pathInContainer)}'`]);\n        await this.adb.push(localFile, tmpDestination);\n        await this.adb.shell([\n          'run-as', packageId,\n          `cp -f '${escapePath(tmpDestination)}' '${escapePath(pathInContainer)}'`\n        ]);\n      } catch (e) {\n        this.log.errorAndThrow(`Cannot access the container of '${packageId}' application. ` +\n                          `Is the application installed and has 'debuggable' build option set to true? ` +\n                          `Original error: ${e.message}`);\n      }\n    } else {\n      // adb push creates folders and overwrites existing files.\n      await this.adb.push(localFile, remotePath);\n\n      // if we have pushed a file, it might be a media file, so ensure that\n      // apps know about it\n      await scanMedia(this.adb, remotePath, this.log);\n    }\n  } finally {\n    if (await fs.exists(localFile)) {\n      await fs.unlink(localFile);\n    }\n    if (tmpDestination) {\n      await this.adb.shell(['rm', '-f', tmpDestination]);\n    }\n  }\n};\n\n/**\n * Pulls the whole folder from the remote device\n *\n * @param {string} remotePath The full path to a folder on the\n * remote device or a folder inside an application bundle\n * @returns {string} Base64-encoded and zipped content of the folder\n * @throws {Error} If there was a failure while getting the folder content\n */\ncommands.pullFolder = async function pullFolder (remotePath) {\n  let localFolder = await tempDir.path({prefix: 'appium'});\n  await this.adb.pull(remotePath, localFolder);\n  return (await zip.toInMemoryZip(localFolder, {\n    encodeToBase64: true,\n  })).toString();\n};\n\n/**\n * Deletes the given folder or file from the remote device\n *\n * @param {ADB} adb\n * @param {string} remotePath The full path to the remote folder\n * or file (folder names must end with a single slash)\n * @throws {Error} If the provided remote path is invalid or\n * the package content cannot be accessed\n * @returns {boolean} `true` if the remote item has been successfully deleted.\n * If the remote path is valid, but the remote path does not exist\n * this function return `false`.\n */\nasync function deleteFileOrFolder (adb, remotePath) {\n  const performRemoteFsCheck = async (p, op, runAs = null) => {\n    const passFlag = '__PASS__';\n    const checkCmd = `[ -${op} '${escapePath(p)}' ] && echo ${passFlag}`;\n    const fullCmd = runAs ? `run-as ${runAs} ${checkCmd}` : checkCmd;\n    try {\n      return _.includes(await adb.shell([fullCmd]), passFlag);\n    } catch (ign) {\n      return false;\n    }\n  };\n  const isFile = async (p, runAs = null) => await performRemoteFsCheck(p, 'f', runAs);\n  const isDir = async (p, runAs = null) => await performRemoteFsCheck(p, 'd', runAs);\n  const isPresent = async (p, runAs = null) => await performRemoteFsCheck(p, 'e', runAs);\n\n  let dstPath = remotePath;\n  let pkgId = null;\n  if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n    const [packageId, pathInContainer] = parseContainerPath(remotePath);\n    this.log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'`);\n    dstPath = pathInContainer;\n    pkgId = packageId;\n  }\n\n  if (pkgId) {\n    try {\n      await adb.shell(['run-as', pkgId, 'ls']);\n    } catch (e) {\n      this.log.errorAndThrow(`Cannot access the container of '${pkgId}' application. ` +\n        `Is the application installed and has 'debuggable' build option set to true? ` +\n        `Original error: ${e.message}`);\n    }\n  }\n\n  if (!await isPresent(dstPath, pkgId)) {\n    this.log.info(`The item at '${dstPath}' does not exist. Perhaps, already deleted?`);\n    return false;\n  }\n\n  const expectsFile = !remotePath.endsWith('/');\n  if (expectsFile && !await isFile(dstPath, pkgId)) {\n    this.log.errorAndThrow(`The item at '${dstPath}' is not a file`);\n  } else if (!expectsFile && !await isDir(dstPath, pkgId)) {\n    this.log.errorAndThrow(`The item at '${dstPath}' is not a folder`);\n  }\n\n  if (pkgId) {\n    await adb.shell(\n      ['run-as', pkgId, `rm -f${expectsFile ? '' : 'r'} '${escapePath(dstPath)}'`]);\n  } else {\n    await adb.shell(['rm', `-f${expectsFile ? '' : 'r'}`, dstPath]);\n  }\n  if (await isPresent(dstPath, pkgId)) {\n    this.log.errorAndThrow(`The item at '${dstPath}' still exists after being deleted. ` +\n      `Is it writable?`);\n  }\n  return true;\n}\n\n/**\n * @typedef {Object} DeleteFileOpts\n * @property {!string} remotePath The full path to the remote file\n * or a file inside an application bundle (for example `@my.app.id/path/in/bundle`)\n */\n\n/**\n * Deletes a file on the remote device\n *\n * @param {DeleteFileOpts} opts\n * @returns {boolean} `true` if the remote file has been successfully deleted.\n * If the path to a remote file is valid, but the file itself does not exist\n * then `false` is returned.\n * @throws {Error} If the argument is invalid or there was an error while\n * deleting the file\n */\ncommands.mobileDeleteFile = async function mobileDeleteFile (opts = {}) {\n  const {remotePath} = opts;\n  if (!remotePath) {\n    throw new errors.InvalidArgumentError(`The 'remotePath' argument is mandatory`);\n  }\n  if (remotePath.endsWith('/')) {\n    throw new errors.InvalidArgumentError(\n      `It is expected that remote path points to a folder and not to a file. ` +\n      `'${remotePath}' is given instead`\n    );\n  }\n  return await deleteFileOrFolder(this.adb, remotePath);\n};\n\nexport { commands };\nexport default commands;\n"],"mappings":"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"}
260
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["CONTAINER_PATH_MARKER","CONTAINER_PATH_PATTERN","RegExp","ANDROID_MEDIA_RESCAN_INTENT","commands","parseContainerPath","remotePath","match","exec","Error","path","posix","resolve","scanMedia","adb","log","debug","getApiLevel","shell","e","warn","stderr","message","escapePath","p","replace","pullFile","endsWith","errors","InvalidArgumentError","tmpDestination","startsWith","packageId","pathInContainer","basename","errorAndThrow","localFile","tempDir","prefix","suffix","pull","util","toInMemoryBase64","toString","fs","exists","unlink","mobilePullFile","opts","requireArgs","pushFile","base64Data","_","isArray","Buffer","from","content","writeFile","dirname","push","mobilePushFile","payload","pullFolder","tmpRoot","openDir","zip","toInMemoryZip","encodeToBase64","rimraf","mobilePullFolder","deleteFileOrFolder","performRemoteFsCheck","op","runAs","passFlag","checkCmd","fullCmd","includes","ign","isFile","isDir","isPresent","dstPath","pkgId","info","expectsFile","mobileDeleteFile"],"sources":["../../../lib/commands/file-actions.js"],"sourcesContent":["import _ from 'lodash';\nimport { fs, util, zip, tempDir } from 'appium/support';\nimport path from 'path';\nimport { errors } from 'appium/driver';\nimport { requireArgs } from '../utils';\n\n\nconst CONTAINER_PATH_MARKER = '@';\n// https://regex101.com/r/PLdB0G/2\nconst CONTAINER_PATH_PATTERN = new RegExp(`^${CONTAINER_PATH_MARKER}([^/]+)/(.+)`);\nconst ANDROID_MEDIA_RESCAN_INTENT = 'android.intent.action.MEDIA_SCANNER_SCAN_FILE';\n\n\nconst commands = {};\n\n/**\n * Parses the actual destination path from the given value\n *\n * @param {string} remotePath The preformatted remote path, which looks like\n * `@my.app.id/my/path`\n * @returns {Array<string>} An array, where the first item is the parsed package\n * identifier and the second one is the actual destination path inside the package.\n * @throws {Error} If the given string cannot be parsed\n */\nfunction parseContainerPath (remotePath) {\n  const match = CONTAINER_PATH_PATTERN.exec(remotePath);\n  if (!match) {\n    throw new Error(`It is expected that package identifier is separated from the relative path with a single slash. ` +\n      `'${remotePath}' is given instead`);\n  }\n  return [match[1], path.posix.resolve(`/data/data/${match[1]}`, match[2])];\n}\n\n/**\n * Scans the given file/folder on the remote device\n * and adds matching items to the device's media library.\n * Exceptions are ignored and written into the log.\n *\n * @param {ADB} adb ADB instance\n * @param {Object?} log Logger instance\n * @param {string} remotePath The file/folder path on the remote device\n */\nasync function scanMedia (adb, remotePath, log = null) {\n  log?.debug(`Performing media scan of '${remotePath}'`);\n  try {\n    // https://github.com/appium/appium/issues/16184\n    if (await adb.getApiLevel() >= 29) {\n      await adb.scanMedia(remotePath);\n    } else {\n      await adb.shell([\n        'am', 'broadcast',\n        '-a', ANDROID_MEDIA_RESCAN_INTENT,\n        '-d', `file://${remotePath}`\n      ]);\n    }\n  } catch (e) {\n    log?.warn(`Ignoring an unexpected error upon media scanning of '${remotePath}': ${e.stderr || e.message}`);\n  }\n}\n\n/**\n * A small helper, which escapes single quotes in paths,\n * so they are safe to be passed as arguments of shell commands\n *\n * @param {string} p The initial remote path\n * @returns {string} The escaped path value\n */\nfunction escapePath (p) {\n  return p.replace(/'/g, `\\\\'`);\n}\n\n/**\n * Pulls a remote file from the device.\n * It is required, that a package has debugging flag enabled\n * in order to access its files.\n *\n * @param {string} remotePath The full path to the remote file\n * or a specially formatted path, which points to an item inside app bundle\n * @returns {string} Base64 encoded content of the pulled file\n * @throws {Error} If the pull operation failed\n */\ncommands.pullFile = async function pullFile (remotePath) {\n  if (remotePath.endsWith('/')) {\n    throw new errors.InvalidArgumentError(`It is expected that remote path points to a file and not to a folder. ` +\n      `'${remotePath}' is given instead`);\n  }\n  let tmpDestination = null;\n  if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n    const [packageId, pathInContainer] = parseContainerPath(remotePath);\n    this.log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'. Will get the data from '${pathInContainer}'`);\n    tmpDestination = `/data/local/tmp/${path.posix.basename(pathInContainer)}`;\n    try {\n      await this.adb.shell(['run-as', packageId, `chmod 777 '${escapePath(pathInContainer)}'`]);\n      await this.adb.shell([\n        'run-as', packageId,\n        `cp -f '${escapePath(pathInContainer)}' '${escapePath(tmpDestination)}'`\n      ]);\n    } catch (e) {\n      this.log.errorAndThrow(`Cannot access the container of '${packageId}' application. ` +\n                        `Is the application installed and has 'debuggable' build option set to true? ` +\n                        `Original error: ${e.message}`);\n    }\n  }\n  const localFile = await tempDir.path({prefix: 'appium', suffix: '.tmp'});\n  try {\n    await this.adb.pull(tmpDestination || remotePath, localFile);\n    return (await util.toInMemoryBase64(localFile)).toString();\n  } finally {\n    if (await fs.exists(localFile)) {\n      await fs.unlink(localFile);\n    }\n    if (tmpDestination) {\n      await this.adb.shell(['rm', '-f', tmpDestination]);\n    }\n  }\n};\n\n/**\n * @typedef {Object} PullFileOptions\n * @property {string} remotePath The full path to the remote file\n * or a specially formatted path, which points to an item inside an app bundle,\n * for example `@my.app.id/my/path`. It is mandatory for the app bundle to have\n * debugging enabled in order to use the latter remotePath format.\n */\n\n/**\n * Pulls a remote file from the device.\n *\n * @param {PullFileOptions} opts\n * @returns {string} The same as `pullFile`\n */\ncommands.mobilePullFile = async function mobilePullFile (opts = {}) {\n  const { remotePath } = requireArgs('remotePath', opts);\n  return await this.pullFile(remotePath);\n};\n\n/**\n * Pushes the given data to a file on the remote device\n * It is required, that a package has debugging flag enabled\n * in order to access its files.\n * After a file is pushed it gets automatically scanned for possible\n * media occurrences. If the scan succeeds then the file is added to the\n * media library.\n *\n * @param {string} remotePath The full path to the remote file or\n * a file inside a package bundle\n * @param {string} base64Data Base64 encoded data to be written to the\n * remote file. The remote file will be silently overridden if it already exists.\n * @throws {Error} If there was an error while pushing the data\n */\ncommands.pushFile = async function pushFile (remotePath, base64Data) {\n  if (remotePath.endsWith('/')) {\n    throw new errors.InvalidArgumentError(\n      `It is expected that remote path points to a file and not to a folder. ` +\n      `'${remotePath}' is given instead`\n    );\n  }\n  const localFile = await tempDir.path({prefix: 'appium', suffix: '.tmp'});\n  if (_.isArray(base64Data)) {\n    // some clients (ahem) java, send a byte array encoding utf8 characters\n    // instead of a string, which would be infinitely better!\n    base64Data = Buffer.from(base64Data).toString('utf8');\n  }\n  const content = Buffer.from(base64Data, 'base64');\n  let tmpDestination = null;\n  try {\n    await fs.writeFile(localFile, content.toString('binary'), 'binary');\n    if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n      const [packageId, pathInContainer] = parseContainerPath(remotePath);\n      this.log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'. ` +\n        `Will put the data into '${pathInContainer}'`);\n      tmpDestination = `/data/local/tmp/${path.posix.basename(pathInContainer)}`;\n      try {\n        await this.adb.shell(\n          ['run-as', packageId, `mkdir -p '${escapePath(path.posix.dirname(pathInContainer))}'`]\n        );\n        await this.adb.shell(['run-as', packageId, `touch '${escapePath(pathInContainer)}'`]);\n        await this.adb.shell(['run-as', packageId, `chmod 777 '${escapePath(pathInContainer)}'`]);\n        await this.adb.push(localFile, tmpDestination);\n        await this.adb.shell([\n          'run-as', packageId,\n          `cp -f '${escapePath(tmpDestination)}' '${escapePath(pathInContainer)}'`\n        ]);\n      } catch (e) {\n        this.log.errorAndThrow(`Cannot access the container of '${packageId}' application. ` +\n                          `Is the application installed and has 'debuggable' build option set to true? ` +\n                          `Original error: ${e.message}`);\n      }\n    } else {\n      // adb push creates folders and overwrites existing files.\n      await this.adb.push(localFile, remotePath);\n\n      // if we have pushed a file, it might be a media file, so ensure that\n      // apps know about it\n      await scanMedia(this.adb, remotePath, this.log);\n    }\n  } finally {\n    if (await fs.exists(localFile)) {\n      await fs.unlink(localFile);\n    }\n    if (tmpDestination) {\n      await this.adb.shell(['rm', '-f', tmpDestination]);\n    }\n  }\n};\n\n/**\n * @typedef {Object} PushFileOptions\n * @property {string} remotePath The full path to the remote file\n * or a specially formatted path, which points to an item inside an app bundle,\n * for example `@my.app.id/my/path`. It is mandatory for the app bundle to have\n * debugging enabled in order to use the latter remotePath format.\n * @property {string} payload Base64-encoded content of the file to be pushed.\n */\n\n/**\n * Pushes the given data to a file on the remote device.\n *\n * @param {PushFileOptions} opts\n */\ncommands.mobilePushFile = async function mobilePushFile (opts = {}) {\n  const { remotePath, payload } = requireArgs(['remotePath', 'payload'], opts);\n  return await this.pushFile(remotePath, payload);\n};\n\n/**\n * Pulls the whole folder from the remote device\n *\n * @param {string} remotePath The full path to a folder on the\n * remote device or a folder inside an application bundle\n * @returns {string} Base64-encoded and zipped content of the folder\n * @throws {Error} If there was a failure while getting the folder content\n */\ncommands.pullFolder = async function pullFolder (remotePath) {\n  const tmpRoot = await tempDir.openDir();\n  try {\n    await this.adb.pull(remotePath, tmpRoot);\n    return (await zip.toInMemoryZip(tmpRoot, {\n      encodeToBase64: true,\n    })).toString();\n  } finally {\n    await fs.rimraf(tmpRoot);\n  }\n};\n\n/**\n * @typedef {Object} PullFolderOptions\n * @property {string} remotePath The full path to the remote folder.\n */\n\n/**\n * Pulls the whole folder from the device under test.\n *\n * @param {PullFolderOptions} opts\n * @returns {string} The same as `pullFolder`\n */\ncommands.mobilePullFolder = async function mobilePullFolder (opts = {}) {\n  const { remotePath } = requireArgs('remotePath', opts);\n  return await this.pullFolder(remotePath);\n};\n\n/**\n * Deletes the given folder or file from the remote device\n *\n * @param {ADB} adb\n * @param {string} remotePath The full path to the remote folder\n * or file (folder names must end with a single slash)\n * @throws {Error} If the provided remote path is invalid or\n * the package content cannot be accessed\n * @returns {boolean} `true` if the remote item has been successfully deleted.\n * If the remote path is valid, but the remote path does not exist\n * this function return `false`.\n */\nasync function deleteFileOrFolder (adb, remotePath) {\n  const performRemoteFsCheck = async (p, op, runAs = null) => {\n    const passFlag = '__PASS__';\n    const checkCmd = `[ -${op} '${escapePath(p)}' ] && echo ${passFlag}`;\n    const fullCmd = runAs ? `run-as ${runAs} ${checkCmd}` : checkCmd;\n    try {\n      return _.includes(await adb.shell([fullCmd]), passFlag);\n    } catch (ign) {\n      return false;\n    }\n  };\n  const isFile = async (p, runAs = null) => await performRemoteFsCheck(p, 'f', runAs);\n  const isDir = async (p, runAs = null) => await performRemoteFsCheck(p, 'd', runAs);\n  const isPresent = async (p, runAs = null) => await performRemoteFsCheck(p, 'e', runAs);\n\n  let dstPath = remotePath;\n  let pkgId = null;\n  if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n    const [packageId, pathInContainer] = parseContainerPath(remotePath);\n    this.log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'`);\n    dstPath = pathInContainer;\n    pkgId = packageId;\n  }\n\n  if (pkgId) {\n    try {\n      await adb.shell(['run-as', pkgId, 'ls']);\n    } catch (e) {\n      this.log.errorAndThrow(`Cannot access the container of '${pkgId}' application. ` +\n        `Is the application installed and has 'debuggable' build option set to true? ` +\n        `Original error: ${e.message}`);\n    }\n  }\n\n  if (!await isPresent(dstPath, pkgId)) {\n    this.log.info(`The item at '${dstPath}' does not exist. Perhaps, already deleted?`);\n    return false;\n  }\n\n  const expectsFile = !remotePath.endsWith('/');\n  if (expectsFile && !await isFile(dstPath, pkgId)) {\n    this.log.errorAndThrow(`The item at '${dstPath}' is not a file`);\n  } else if (!expectsFile && !await isDir(dstPath, pkgId)) {\n    this.log.errorAndThrow(`The item at '${dstPath}' is not a folder`);\n  }\n\n  if (pkgId) {\n    await adb.shell(\n      ['run-as', pkgId, `rm -f${expectsFile ? '' : 'r'} '${escapePath(dstPath)}'`]);\n  } else {\n    await adb.shell(['rm', `-f${expectsFile ? '' : 'r'}`, dstPath]);\n  }\n  if (await isPresent(dstPath, pkgId)) {\n    this.log.errorAndThrow(`The item at '${dstPath}' still exists after being deleted. ` +\n      `Is it writable?`);\n  }\n  return true;\n}\n\n/**\n * @typedef {Object} DeleteFileOpts\n * @property {!string} remotePath The full path to the remote file\n * or a file inside an application bundle (for example `@my.app.id/path/in/bundle`)\n */\n\n/**\n * Deletes a file on the remote device\n *\n * @param {DeleteFileOpts} opts\n * @returns {boolean} `true` if the remote file has been successfully deleted.\n * If the path to a remote file is valid, but the file itself does not exist\n * then `false` is returned.\n * @throws {Error} If the argument is invalid or there was an error while\n * deleting the file\n */\ncommands.mobileDeleteFile = async function mobileDeleteFile (opts = {}) {\n  const {remotePath} = opts;\n  if (!remotePath) {\n    throw new errors.InvalidArgumentError(`The 'remotePath' argument is mandatory`);\n  }\n  if (remotePath.endsWith('/')) {\n    throw new errors.InvalidArgumentError(\n      `It is expected that remote path points to a folder and not to a file. ` +\n      `'${remotePath}' is given instead`\n    );\n  }\n  return await deleteFileOrFolder(this.adb, remotePath);\n};\n\nexport { commands };\nexport default commands;\n"],"mappings":"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"}