appium-android-driver 5.2.12 → 5.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/lib/commands/app-management.js +54 -2
- package/build/lib/commands/execute.js +7 -1
- package/build/lib/commands/recordscreen.js +10 -5
- package/build/lib/driver.js +10 -5
- package/lib/commands/app-management.js +134 -2
- package/lib/commands/execute.js +6 -0
- package/lib/commands/recordscreen.js +6 -3
- package/lib/driver.js +9 -4
- package/package.json +1 -1
|
@@ -21,13 +21,30 @@ var _driver = require("appium/driver");
|
|
|
21
21
|
|
|
22
22
|
const APP_EXTENSIONS = ['.apk', '.apks'];
|
|
23
23
|
const RESOLVER_ACTIVITY_NAME = 'android/com.android.internal.app.ResolverActivity';
|
|
24
|
-
|
|
24
|
+
const commands = {};
|
|
25
25
|
exports.commands = commands;
|
|
26
26
|
|
|
27
|
+
function requireArgs(argNames, opts = {}) {
|
|
28
|
+
for (const argName of _lodash.default.isArray(argNames) ? argNames : [argNames]) {
|
|
29
|
+
if (!_lodash.default.has(opts, argName)) {
|
|
30
|
+
throw new _driver.errors.InvalidArgumentError(`'${argName}' argument must be provided`);
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
return opts;
|
|
35
|
+
}
|
|
36
|
+
|
|
27
37
|
commands.isAppInstalled = async function isAppInstalled(appId) {
|
|
28
38
|
return await this.adb.isAppInstalled(appId);
|
|
29
39
|
};
|
|
30
40
|
|
|
41
|
+
commands.mobileIsAppInstalled = async function mobileIsAppInstalled(opts = {}) {
|
|
42
|
+
const {
|
|
43
|
+
appId
|
|
44
|
+
} = requireArgs('appId', opts);
|
|
45
|
+
return await this.isAppInstalled(appId);
|
|
46
|
+
};
|
|
47
|
+
|
|
31
48
|
commands.queryAppState = async function queryAppState(appId) {
|
|
32
49
|
this.log.info(`Querying the state of '${appId}'`);
|
|
33
50
|
|
|
@@ -50,6 +67,13 @@ commands.queryAppState = async function queryAppState(appId) {
|
|
|
50
67
|
return _androidHelpers.APP_STATE.RUNNING_IN_BACKGROUND;
|
|
51
68
|
};
|
|
52
69
|
|
|
70
|
+
commands.mobileQueryAppState = async function mobileQueryAppState(opts = {}) {
|
|
71
|
+
const {
|
|
72
|
+
appId
|
|
73
|
+
} = requireArgs('appId', opts);
|
|
74
|
+
return await this.queryAppState(appId);
|
|
75
|
+
};
|
|
76
|
+
|
|
53
77
|
commands.activateApp = async function activateApp(appId) {
|
|
54
78
|
this.log.debug(`Activating '${appId}'`);
|
|
55
79
|
const apiLevel = await this.adb.getApiLevel();
|
|
@@ -89,10 +113,24 @@ commands.activateApp = async function activateApp(appId) {
|
|
|
89
113
|
}
|
|
90
114
|
};
|
|
91
115
|
|
|
116
|
+
commands.mobileActivateApp = async function mobileActivateApp(opts = {}) {
|
|
117
|
+
const {
|
|
118
|
+
appId
|
|
119
|
+
} = requireArgs('appId', opts);
|
|
120
|
+
return await this.activateApp(appId);
|
|
121
|
+
};
|
|
122
|
+
|
|
92
123
|
commands.removeApp = async function removeApp(appId, options = {}) {
|
|
93
124
|
return await this.adb.uninstallApk(appId, options);
|
|
94
125
|
};
|
|
95
126
|
|
|
127
|
+
commands.mobileRemoveApp = async function mobileRemoveApp(opts = {}) {
|
|
128
|
+
const {
|
|
129
|
+
appId
|
|
130
|
+
} = requireArgs('appId', opts);
|
|
131
|
+
return await this.removeApp(appId, opts);
|
|
132
|
+
};
|
|
133
|
+
|
|
96
134
|
commands.terminateApp = async function terminateApp(appId, options = {}) {
|
|
97
135
|
this.log.info(`Terminating '${appId}'`);
|
|
98
136
|
|
|
@@ -117,11 +155,25 @@ commands.terminateApp = async function terminateApp(appId, options = {}) {
|
|
|
117
155
|
return true;
|
|
118
156
|
};
|
|
119
157
|
|
|
158
|
+
commands.mobileTerminateApp = async function mobileTerminateApp(opts = {}) {
|
|
159
|
+
const {
|
|
160
|
+
appId
|
|
161
|
+
} = requireArgs('appId', opts);
|
|
162
|
+
return await this.terminateApp(appId, opts);
|
|
163
|
+
};
|
|
164
|
+
|
|
120
165
|
commands.installApp = async function installApp(appPath, options = {}) {
|
|
121
166
|
const localPath = await this.helpers.configureApp(appPath, APP_EXTENSIONS);
|
|
122
167
|
await this.adb.install(localPath, options);
|
|
123
168
|
};
|
|
124
169
|
|
|
170
|
+
commands.mobileInstallApp = async function mobileInstallApp(opts = {}) {
|
|
171
|
+
const {
|
|
172
|
+
appPath
|
|
173
|
+
} = requireArgs('appPath', opts);
|
|
174
|
+
return await this.installApp(appPath, opts);
|
|
175
|
+
};
|
|
176
|
+
|
|
125
177
|
commands.mobileClearApp = async function mobileClearApp(opts = {}) {
|
|
126
178
|
const {
|
|
127
179
|
appId
|
|
@@ -136,4 +188,4 @@ commands.mobileClearApp = async function mobileClearApp(opts = {}) {
|
|
|
136
188
|
|
|
137
189
|
var _default = commands;
|
|
138
190
|
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":"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"}
|
|
191
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["APP_EXTENSIONS","RESOLVER_ACTIVITY_NAME","commands","requireArgs","argNames","opts","argName","_","isArray","has","errors","InvalidArgumentError","isAppInstalled","appId","adb","mobileIsAppInstalled","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","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\nconst commands = {};\n\n/**\n * Assert the presence of particular keys in the given object\n *\n * @param {string|Array<string>} argNames one or more key names\n * @param {Object} opts the object to check\n * @returns {Object} the same given object\n */\nfunction requireArgs (argNames, opts = {}) {\n  for (const argName of (_.isArray(argNames) ? argNames : [argNames])) {\n    if (!_.has(opts, argName)) {\n      throw new errors.InvalidArgumentError(`'${argName}' argument must be provided`);\n    }\n  }\n  return opts;\n}\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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"}
|
|
@@ -49,6 +49,12 @@ extensions.executeMobile = async function executeMobile(mobileCommand, opts = {}
|
|
|
49
49
|
getNotifications: 'mobileGetNotifications',
|
|
50
50
|
listSms: 'mobileListSms',
|
|
51
51
|
deleteFile: 'mobileDeleteFile',
|
|
52
|
+
isAppInstalled: 'mobileIsAppInstalled',
|
|
53
|
+
queryAppState: 'mobileQueryAppState',
|
|
54
|
+
activateApp: 'mobileActivateApp',
|
|
55
|
+
removeApp: 'mobileRemoveApp',
|
|
56
|
+
terminateApp: 'mobileTerminateApp',
|
|
57
|
+
installApp: 'mobileInstallApp',
|
|
52
58
|
clearApp: 'mobileClearApp',
|
|
53
59
|
startService: 'mobileStartService',
|
|
54
60
|
stopService: 'mobileStopService',
|
|
@@ -71,4 +77,4 @@ extensions.executeMobile = async function executeMobile(mobileCommand, opts = {}
|
|
|
71
77
|
|
|
72
78
|
var _default = extensions;
|
|
73
79
|
exports.default = _default;
|
|
74
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
80
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJleHRlbnNpb25zIiwiZXhlY3V0ZSIsInNjcmlwdCIsImFyZ3MiLCJtYXRjaCIsImxvZyIsImluZm8iLCJyZXBsYWNlIiwidHJpbSIsImV4ZWN1dGVNb2JpbGUiLCJfIiwiaXNBcnJheSIsImlzV2ViQ29udGV4dCIsImVycm9ycyIsIk5vdEltcGxlbWVudGVkRXJyb3IiLCJlbmRwb2ludCIsImNocm9tZWRyaXZlciIsImp3cHJveHkiLCJkb3duc3RyZWFtUHJvdG9jb2wiLCJQUk9UT0NPTFMiLCJNSlNPTldQIiwiY29tbWFuZCIsIm1vYmlsZUNvbW1hbmQiLCJvcHRzIiwibW9iaWxlQ29tbWFuZHNNYXBwaW5nIiwic2hlbGwiLCJleGVjRW11Q29uc29sZUNvbW1hbmQiLCJzdGFydExvZ3NCcm9hZGNhc3QiLCJzdG9wTG9nc0Jyb2FkY2FzdCIsImNoYW5nZVBlcm1pc3Npb25zIiwiZ2V0UGVybWlzc2lvbnMiLCJwZXJmb3JtRWRpdG9yQWN0aW9uIiwic2Vuc29yU2V0IiwiZ2V0RGV2aWNlVGltZSIsInN0YXJ0U2NyZWVuU3RyZWFtaW5nIiwic3RvcFNjcmVlblN0cmVhbWluZyIsImdldE5vdGlmaWNhdGlvbnMiLCJsaXN0U21zIiwiZGVsZXRlRmlsZSIsImlzQXBwSW5zdGFsbGVkIiwicXVlcnlBcHBTdGF0ZSIsImFjdGl2YXRlQXBwIiwicmVtb3ZlQXBwIiwidGVybWluYXRlQXBwIiwiaW5zdGFsbEFwcCIsImNsZWFyQXBwIiwic3RhcnRTZXJ2aWNlIiwic3RvcFNlcnZpY2UiLCJzdGFydEFjdGl2aXR5IiwiYnJvYWRjYXN0IiwiZ2V0Q29udGV4dHMiLCJ1bmxvY2siLCJyZWZyZXNoR3BzQ2FjaGUiLCJzdGFydE1lZGlhUHJvamVjdGlvblJlY29yZGluZyIsImlzTWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nUnVubmluZyIsInN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmciLCJoYXMiLCJVbmtub3duQ29tbWFuZEVycm9yIiwia2V5cyJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL2xpYi9jb21tYW5kcy9leGVjdXRlLmpzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgeyBlcnJvcnMsIFBST1RPQ09MUyB9IGZyb20gJ2FwcGl1bS9kcml2ZXInO1xuXG5jb25zdCBleHRlbnNpb25zID0ge307XG5cbmV4dGVuc2lvbnMuZXhlY3V0ZSA9IGFzeW5jIGZ1bmN0aW9uIGV4ZWN1dGUgKHNjcmlwdCwgYXJncykge1xuICBpZiAoc2NyaXB0Lm1hdGNoKC9ebW9iaWxlOi8pKSB7XG4gICAgdGhpcy5sb2cuaW5mbyhgRXhlY3V0aW5nIG5hdGl2ZSBjb21tYW5kICcke3NjcmlwdH0nYCk7XG4gICAgc2NyaXB0ID0gc2NyaXB0LnJlcGxhY2UoL15tb2JpbGU6LywgJycpLnRyaW0oKTtcbiAgICByZXR1cm4gYXdhaXQgdGhpcy5leGVjdXRlTW9iaWxlKHNjcmlwdCwgXy5pc0FycmF5KGFyZ3MpID8gYXJnc1swXSA6IGFyZ3MpO1xuICB9XG4gIGlmICghdGhpcy5pc1dlYkNvbnRleHQoKSkge1xuICAgIHRocm93IG5ldyBlcnJvcnMuTm90SW1wbGVtZW50ZWRFcnJvcigpO1xuICB9XG4gIGNvbnN0IGVuZHBvaW50ID0gdGhpcy5jaHJvbWVkcml2ZXIuandwcm94eS5kb3duc3RyZWFtUHJvdG9jb2wgPT09IFBST1RPQ09MUy5NSlNPTldQXG4gICAgPyAnL2V4ZWN1dGUnXG4gICAgOiAnL2V4ZWN1dGUvc3luYyc7XG4gIHJldHVybiBhd2FpdCB0aGlzLmNocm9tZWRyaXZlci5qd3Byb3h5LmNvbW1hbmQoZW5kcG9pbnQsICdQT1NUJywge1xuICAgIHNjcmlwdCxcbiAgICBhcmdzLFxuICB9KTtcbn07XG5cbmV4dGVuc2lvbnMuZXhlY3V0ZU1vYmlsZSA9IGFzeW5jIGZ1bmN0aW9uIGV4ZWN1dGVNb2JpbGUgKG1vYmlsZUNvbW1hbmQsIG9wdHMgPSB7fSkge1xuICBjb25zdCBtb2JpbGVDb21tYW5kc01hcHBpbmcgPSB7XG4gICAgc2hlbGw6ICdtb2JpbGVTaGVsbCcsXG5cbiAgICBleGVjRW11Q29uc29sZUNvbW1hbmQ6ICdtb2JpbGVFeGVjRW11Q29uc29sZUNvbW1hbmQnLFxuXG4gICAgc3RhcnRMb2dzQnJvYWRjYXN0OiAnbW9iaWxlU3RhcnRMb2dzQnJvYWRjYXN0JyxcbiAgICBzdG9wTG9nc0Jyb2FkY2FzdDogJ21vYmlsZVN0b3BMb2dzQnJvYWRjYXN0JyxcblxuICAgIGNoYW5nZVBlcm1pc3Npb25zOiAnbW9iaWxlQ2hhbmdlUGVybWlzc2lvbnMnLFxuICAgIGdldFBlcm1pc3Npb25zOiAnbW9iaWxlR2V0UGVybWlzc2lvbnMnLFxuXG4gICAgcGVyZm9ybUVkaXRvckFjdGlvbjogJ21vYmlsZVBlcmZvcm1FZGl0b3JBY3Rpb24nLFxuXG4gICAgc2Vuc29yU2V0OiAnc2Vuc29yU2V0JyxcblxuICAgIGdldERldmljZVRpbWU6ICdtb2JpbGVHZXREZXZpY2VUaW1lJyxcblxuICAgIHN0YXJ0U2NyZWVuU3RyZWFtaW5nOiAnbW9iaWxlU3RhcnRTY3JlZW5TdHJlYW1pbmcnLFxuICAgIHN0b3BTY3JlZW5TdHJlYW1pbmc6ICdtb2JpbGVTdG9wU2NyZWVuU3RyZWFtaW5nJyxcblxuICAgIGdldE5vdGlmaWNhdGlvbnM6ICdtb2JpbGVHZXROb3RpZmljYXRpb25zJyxcblxuICAgIGxpc3RTbXM6ICdtb2JpbGVMaXN0U21zJyxcblxuICAgIGRlbGV0ZUZpbGU6ICdtb2JpbGVEZWxldGVGaWxlJyxcblxuICAgIGlzQXBwSW5zdGFsbGVkOiAnbW9iaWxlSXNBcHBJbnN0YWxsZWQnLFxuICAgIHF1ZXJ5QXBwU3RhdGU6ICdtb2JpbGVRdWVyeUFwcFN0YXRlJyxcbiAgICBhY3RpdmF0ZUFwcDogJ21vYmlsZUFjdGl2YXRlQXBwJyxcbiAgICByZW1vdmVBcHA6ICdtb2JpbGVSZW1vdmVBcHAnLFxuICAgIHRlcm1pbmF0ZUFwcDogJ21vYmlsZVRlcm1pbmF0ZUFwcCcsXG4gICAgaW5zdGFsbEFwcDogJ21vYmlsZUluc3RhbGxBcHAnLFxuICAgIGNsZWFyQXBwOiAnbW9iaWxlQ2xlYXJBcHAnLFxuXG4gICAgc3RhcnRTZXJ2aWNlOiAnbW9iaWxlU3RhcnRTZXJ2aWNlJyxcbiAgICBzdG9wU2VydmljZTogJ21vYmlsZVN0b3BTZXJ2aWNlJyxcbiAgICBzdGFydEFjdGl2aXR5OiAnbW9iaWxlU3RhcnRBY3Rpdml0eScsXG4gICAgYnJvYWRjYXN0OiAnbW9iaWxlQnJvYWRjYXN0JyxcblxuICAgIGdldENvbnRleHRzOiAnbW9iaWxlR2V0Q29udGV4dHMnLFxuXG4gICAgdW5sb2NrOiAnbW9iaWxlVW5sb2NrJyxcblxuICAgIHJlZnJlc2hHcHNDYWNoZTogJ21vYmlsZVJlZnJlc2hHcHNDYWNoZScsXG5cbiAgICBzdGFydE1lZGlhUHJvamVjdGlvblJlY29yZGluZzogJ21vYmlsZVN0YXJ0TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nJyxcbiAgICBpc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmc6ICdtb2JpbGVJc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmcnLFxuICAgIHN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmc6ICdtb2JpbGVTdG9wTWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nJyxcbiAgfTtcblxuICBpZiAoIV8uaGFzKG1vYmlsZUNvbW1hbmRzTWFwcGluZywgbW9iaWxlQ29tbWFuZCkpIHtcbiAgICB0aHJvdyBuZXcgZXJyb3JzLlVua25vd25Db21tYW5kRXJyb3IoYFVua25vd24gbW9iaWxlIGNvbW1hbmQgXCIke21vYmlsZUNvbW1hbmR9XCIuIGAgK1xuICAgICAgYE9ubHkgJHtfLmtleXMobW9iaWxlQ29tbWFuZHNNYXBwaW5nKX0gY29tbWFuZHMgYXJlIHN1cHBvcnRlZC5gKTtcbiAgfVxuICByZXR1cm4gYXdhaXQgdGhpc1ttb2JpbGVDb21tYW5kc01hcHBpbmdbbW9iaWxlQ29tbWFuZF1dKG9wdHMpO1xufTtcblxuZXhwb3J0IGRlZmF1bHQgZXh0ZW5zaW9ucztcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7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
|
|
@@ -121,10 +121,15 @@ async function scheduleScreenRecord(adb, recordingProperties, log = null) {
|
|
|
121
121
|
recordingProperties.currentTimeLimit = timeLimitInt - currentDuration;
|
|
122
122
|
const chunkDuration = recordingProperties.currentTimeLimit < MAX_RECORDING_TIME_SEC ? recordingProperties.currentTimeLimit : MAX_RECORDING_TIME_SEC;
|
|
123
123
|
log === null || log === void 0 ? void 0 : log.debug(`Starting the next ${chunkDuration}s-chunk ` + `of screen recording in order to achieve ${timeLimitInt}s total duration`);
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
124
|
+
|
|
125
|
+
(async () => {
|
|
126
|
+
try {
|
|
127
|
+
await scheduleScreenRecord(adb, recordingProperties, log);
|
|
128
|
+
} catch (e) {
|
|
129
|
+
log === null || log === void 0 ? void 0 : log.error(e.stack);
|
|
130
|
+
recordingProperties.stopped = true;
|
|
131
|
+
}
|
|
132
|
+
})();
|
|
128
133
|
});
|
|
129
134
|
await recordingProc.start(0);
|
|
130
135
|
|
|
@@ -305,4 +310,4 @@ commands.stopRecordingScreen = async function stopRecordingScreen(options = {})
|
|
|
305
310
|
|
|
306
311
|
var _default = commands;
|
|
307
312
|
exports.default = _default;
|
|
308
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["commands","RETRY_PAUSE","RETRY_TIMEOUT","MAX_RECORDING_TIME_SEC","MAX_TIME_SEC","DEFAULT_RECORDING_TIME_SEC","PROCESS_SHUTDOWN_TIMEOUT","SCREENRECORD_BINARY","DEFAULT_EXT","MIN_EMULATOR_API_LEVEL","FFMPEG_BINARY","system","isWindows","uploadRecordedMedia","localFile","remotePath","uploadOptions","_","isEmpty","util","toInMemoryBase64","toString","user","pass","method","headers","fileFieldName","formFields","options","auth","net","uploadFile","verifyScreenRecordIsSupported","adb","isEmulator","apiLevel","getApiLevel","Error","scheduleScreenRecord","recordingProperties","log","stopped","timer","videoSize","bitRate","timeLimit","bugReport","currentTimeLimit","hasValue","currentTimeLimitInt","parseInt","isNaN","pathOnDevice","uuidV4","substring","recordingProc","screenrecord","on","currentDuration","getDuration","asSeconds","toFixed","debug","timeLimitInt","chunkDuration","catch","e","error","stack","start","waitForCondition","fileExists","waitMs","intervalMs","records","push","recordingProcess","mergeScreenRecords","mediaFiles","fs","which","configContent","map","x","join","configFile","path","resolve","dirname","writeFile","result","Math","floor","Date","args","info","exec","terminateBackgroundScreenRecording","force","pids","getPIDsByName","p","shell","err","message","startRecordingScreen","forceRestart","stopRecordingScreen","warn","_screenRecordingProperties","record","rimraf","timeout","parseFloat","timing","Timer","isRunning","stop","errorAndThrow","tmpRoot","tempDir","openDir","localRecords","posix","basename","pull","last","resultFilePath","length","size","stat","toReadableSizeString"],"sources":["../../../lib/commands/recordscreen.js"],"sourcesContent":["import _ from 'lodash';\nimport { waitForCondition } from 'asyncbox';\nimport { util, fs, net, tempDir, system, timing } from 'appium/support';\nimport { exec } from 'teen_process';\nimport path from 'path';\n\n\nconst commands = {};\n\nconst RETRY_PAUSE = 300;\nconst RETRY_TIMEOUT = 5000;\nconst MAX_RECORDING_TIME_SEC = 60 * 3;\nconst MAX_TIME_SEC = 60 * 30;\nconst DEFAULT_RECORDING_TIME_SEC = MAX_RECORDING_TIME_SEC;\nconst PROCESS_SHUTDOWN_TIMEOUT = 10 * 1000;\nconst SCREENRECORD_BINARY = 'screenrecord';\nconst DEFAULT_EXT = '.mp4';\nconst MIN_EMULATOR_API_LEVEL = 27;\nconst FFMPEG_BINARY = `ffmpeg${system.isWindows() ? '.exe' : ''}`;\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\nasync function verifyScreenRecordIsSupported (adb, isEmulator) {\n  const apiLevel = await adb.getApiLevel();\n  if (isEmulator && apiLevel < MIN_EMULATOR_API_LEVEL) {\n    throw new Error(`Screen recording does not work on emulators running Android API level less than ${MIN_EMULATOR_API_LEVEL}`);\n  }\n  if (apiLevel < 19) {\n    throw new Error(`Screen recording not available on API Level ${apiLevel}. Minimum API Level is 19.`);\n  }\n}\n\nasync function scheduleScreenRecord (adb, recordingProperties, log = null) {\n  if (recordingProperties.stopped) {\n    return;\n  }\n\n  const {\n    timer,\n    videoSize,\n    bitRate,\n    timeLimit,\n    bugReport,\n  } = recordingProperties;\n\n  let currentTimeLimit = MAX_RECORDING_TIME_SEC;\n  if (util.hasValue(recordingProperties.currentTimeLimit)) {\n    const currentTimeLimitInt = parseInt(recordingProperties.currentTimeLimit, 10);\n    if (!isNaN(currentTimeLimitInt) && currentTimeLimitInt < MAX_RECORDING_TIME_SEC) {\n      currentTimeLimit = currentTimeLimitInt;\n    }\n  }\n  const pathOnDevice = `/sdcard/${util.uuidV4().substring(0, 8)}${DEFAULT_EXT}`;\n  const recordingProc = adb.screenrecord(pathOnDevice, {\n    videoSize,\n    bitRate,\n    timeLimit: currentTimeLimit,\n    bugReport,\n  });\n\n  recordingProc.on('end', () => {\n    if (recordingProperties.stopped || !util.hasValue(timeLimit)) {\n      return;\n    }\n    const currentDuration = timer.getDuration().asSeconds.toFixed(0);\n    log?.debug(`The overall screen recording duration is ${currentDuration}s so far`);\n    const timeLimitInt = parseInt(timeLimit, 10);\n    if (isNaN(timeLimitInt) || currentDuration >= timeLimitInt) {\n      log?.debug('There is no need to start the next recording chunk');\n      return;\n    }\n\n    recordingProperties.currentTimeLimit = timeLimitInt - currentDuration;\n    const chunkDuration = recordingProperties.currentTimeLimit < MAX_RECORDING_TIME_SEC\n      ? recordingProperties.currentTimeLimit\n      : MAX_RECORDING_TIME_SEC;\n    log?.debug(`Starting the next ${chunkDuration}s-chunk ` +\n      `of screen recording in order to achieve ${timeLimitInt}s total duration`);\n    scheduleScreenRecord(adb, recordingProperties, log)\n      .catch((e) => {\n        log?.error(e.stack);\n        recordingProperties.stopped = true;\n      });\n  });\n\n  await recordingProc.start(0);\n  try {\n    await waitForCondition(async () => await adb.fileExists(pathOnDevice),\n      {waitMs: RETRY_TIMEOUT, intervalMs: RETRY_PAUSE});\n  } catch (e) {\n    throw new Error(`The expected screen record file '${pathOnDevice}' does not exist after ${RETRY_TIMEOUT}ms. ` +\n      `Is ${SCREENRECORD_BINARY} utility available and operational on the device under test?`);\n  }\n\n  recordingProperties.records.push(pathOnDevice);\n  recordingProperties.recordingProcess = recordingProc;\n}\n\nasync function mergeScreenRecords (mediaFiles, log = null) {\n  try {\n    await fs.which(FFMPEG_BINARY);\n  } catch (e) {\n    throw new Error(`${FFMPEG_BINARY} utility is not available in PATH. Please install it from https://www.ffmpeg.org/`);\n  }\n  const configContent = mediaFiles\n    .map((x) => `file '${x}'`)\n    .join('\\n');\n  const configFile = path.resolve(path.dirname(mediaFiles[0]), 'config.txt');\n  await fs.writeFile(configFile, configContent, 'utf8');\n  log?.debug(`Generated ffmpeg merging config '${configFile}' with items:\\n${configContent}`);\n  const result = path.resolve(path.dirname(mediaFiles[0]), `merge_${Math.floor(new Date())}${DEFAULT_EXT}`);\n  const args = ['-safe', '0', '-f', 'concat', '-i', configFile, '-c', 'copy', result];\n  log?.info(`Initiating screen records merging using the command '${FFMPEG_BINARY} ${args.join(' ')}'`);\n  await exec(FFMPEG_BINARY, args);\n  return result;\n}\n\nasync function terminateBackgroundScreenRecording (adb, force = true) {\n  const pids = (await adb.getPIDsByName(SCREENRECORD_BINARY))\n    .map((p) => `${p}`);\n  if (_.isEmpty(pids)) {\n    return false;\n  }\n\n  try {\n    await adb.shell(['kill', force ? '-15' : '-2', ...pids]);\n    await waitForCondition(async () => _.isEmpty(await adb.getPIDsByName(SCREENRECORD_BINARY)), {\n      waitMs: PROCESS_SHUTDOWN_TIMEOUT,\n      intervalMs: 500,\n    });\n    return true;\n  } catch (err) {\n    throw new Error(`Unable to stop the background screen recording: ${err.message}`);\n  }\n}\n\n\n/**\n * @typedef {Object} StartRecordingOptions\n *\n * @property {?string} remotePath - The path to the remote location, where the captured 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 endpount 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 *                                  This option only has an effect if there is screen recording process in progreess\n *                                  and `forceRestart` parameter is not set to `true`.\n * @property {?string} user - The name of the user for the remote authentication. Only works if `remotePath` is provided.\n * @property {?string} pass - The password for the remote authentication. Only works if `remotePath` is provided.\n * @property {?string} method [PUT] - The http multipart upload method name. Only works if `remotePath` is provided.\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 * @property {?string} videoSize - The format is widthxheight.\n *                  The default value is the device's native display resolution (if supported),\n *                  1280x720 if not. For best results,\n *                  use a size supported by your device's Advanced Video Coding (AVC) encoder.\n *                  For example, \"1280x720\"\n * @property {?boolean} bugReport - Set it to `true` in order to display additional information on the video overlay,\n *                                  such as a timestamp, that is helpful in videos captured to illustrate bugs.\n *                                  This option is only supported since API level 27 (Android P).\n * @property {?string|number} timeLimit - The maximum recording time, in seconds. The default value is 180 (3 minutes).\n *                                        The maximum value is 1800 (30 minutes). If the passed value is greater than 180 then\n *                                        the algorithm will try to schedule multiple screen recording chunks and merge the\n *                                        resulting videos into a single media file using `ffmpeg` utility.\n *                                        If the utility is not available in PATH then the most recent screen recording chunk is\n *                                        going to be returned.\n * @property {?string|number} bitRate - The video bit rate for the video, in bits per second.\n *                The default value is 4000000 (4 Mbit/s). You can increase the bit rate to improve video quality,\n *                but doing so results in larger movie files.\n * @property {?boolean} forceRestart - Whether to try to catch and upload/return the currently running screen recording\n *                                     (`false`, the default setting) or ignore the result of it and start a new recording\n *                                     immediately (`true`).\n */\n\n/**\n * Record the display of a real devices running Android 4.4 (API level 19) and higher.\n * Emulators are supported since API level 27 (Android P).\n * It records screen activity to an MPEG-4 file. Audio is not recorded with the video file.\n * If screen recording has been already started then the command will stop it forcefully and start a new one.\n * The previously recorded video file will be deleted.\n *\n * @param {?StartRecordingOptions} options - The available options.\n * @returns {string} Base64-encoded content of the recorded media file if\n *                   any screen recording is currently running or an empty string.\n * @throws {Error} If screen recording has failed to start or is not supported on the device under test.\n */\ncommands.startRecordingScreen = async function startRecordingScreen (options = {}) {\n  await verifyScreenRecordIsSupported(this.adb, this.isEmulator());\n\n  let result = '';\n  const {videoSize, timeLimit = DEFAULT_RECORDING_TIME_SEC, bugReport, bitRate, forceRestart} = options;\n  if (!forceRestart) {\n    result = await this.stopRecordingScreen(options);\n  }\n\n  if (await terminateBackgroundScreenRecording(this.adb, true)) {\n    this.log.warn(`There were some ${SCREENRECORD_BINARY} process leftovers running ` +\n      `in the background. Make sure you stop screen recording each time after it is started, ` +\n      `otherwise the recorded media might quickly exceed all the free space on the device under test.`);\n  }\n\n  if (!_.isEmpty(this._screenRecordingProperties)) {\n    for (const record of (this._screenRecordingProperties.records || [])) {\n      await this.adb.rimraf(record);\n    }\n    this._screenRecordingProperties = null;\n  }\n\n  const timeout = parseFloat(timeLimit);\n  if (isNaN(timeout) || timeout > MAX_TIME_SEC || timeout <= 0) {\n    throw new Error(`The timeLimit value must be in range [1, ${MAX_TIME_SEC}] seconds. ` +\n      `The value of '${timeLimit}' has been passed instead.`);\n  }\n\n  this._screenRecordingProperties = {\n    timer: new timing.Timer().start(),\n    videoSize,\n    timeLimit,\n    currentTimeLimit: timeLimit,\n    bitRate,\n    bugReport,\n    records: [],\n    recordingProcess: null,\n    stopped: false,\n  };\n  await scheduleScreenRecord(this.adb, this._screenRecordingProperties, this.log);\n  return result;\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 endpount 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 recording the screen.\n * If no screen recording has been started before then the method returns an empty string.\n *\n * @param {?StopRecordingOptions} options - The 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 getting the name of a media file\n *                 or the file content cannot be uploaded to the remote location\n *                 or screen recording is not supported on the device under test.\n */\ncommands.stopRecordingScreen = async function stopRecordingScreen (options = {}) {\n  await verifyScreenRecordIsSupported(this.adb, this.isEmulator());\n\n  if (!_.isEmpty(this._screenRecordingProperties)) {\n    this._screenRecordingProperties.stopped = true;\n  }\n\n  try {\n    await terminateBackgroundScreenRecording(this.adb, false);\n  } catch (err) {\n    this.log.warn(err.message);\n    if (!_.isEmpty(this._screenRecordingProperties)) {\n      this.log.warn('The resulting video might be corrupted');\n    }\n  }\n\n  if (_.isEmpty(this._screenRecordingProperties)) {\n    this.log.info(`Screen recording has not been previously started by Appium. There is nothing to stop`);\n    return '';\n  }\n\n  if (this._screenRecordingProperties.recordingProcess && this._screenRecordingProperties.recordingProcess.isRunning) {\n    try {\n      await this._screenRecordingProperties.recordingProcess.stop('SIGINT', PROCESS_SHUTDOWN_TIMEOUT);\n    } catch (e) {\n      this.log.errorAndThrow(`Unable to stop screen recording within ${PROCESS_SHUTDOWN_TIMEOUT}ms`);\n    }\n    this._screenRecordingProperties.recordingProcess = null;\n  }\n\n  if (_.isEmpty(this._screenRecordingProperties.records)) {\n    this.log.errorAndThrow(`No screen recordings have been stored on the device so far. ` +\n      `Are you sure the ${SCREENRECORD_BINARY} utility works as expected?`);\n  }\n\n  const tmpRoot = await tempDir.openDir();\n  try {\n    const localRecords = [];\n    for (const pathOnDevice of this._screenRecordingProperties.records) {\n      localRecords.push(path.resolve(tmpRoot, path.posix.basename(pathOnDevice)));\n      await this.adb.pull(pathOnDevice, _.last(localRecords));\n      await this.adb.rimraf(pathOnDevice);\n    }\n    let resultFilePath = _.last(localRecords);\n    if (localRecords.length > 1) {\n      this.log.info(`Got ${localRecords.length} screen recordings. Trying to merge them`);\n      try {\n        resultFilePath = await mergeScreenRecords(localRecords, this.log);\n      } catch (e) {\n        this.log.warn(`Cannot merge the recorded files. The most recent screen recording is going to be returned as the result. ` +\n          `Original error: ${e.message}`);\n      }\n    }\n    if (_.isEmpty(options.remotePath)) {\n      const {size} = await fs.stat(resultFilePath);\n      this.log.debug(`The size of the resulting screen recording is ${util.toReadableSizeString(size)}`);\n    }\n    return await uploadRecordedMedia(resultFilePath, options.remotePath, options);\n  } finally {\n    await fs.rimraf(tmpRoot);\n    this._screenRecordingProperties = null;\n  }\n};\n\n\nexport { commands };\nexport default commands;\n"],"mappings":"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"}
|
|
313
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["commands","RETRY_PAUSE","RETRY_TIMEOUT","MAX_RECORDING_TIME_SEC","MAX_TIME_SEC","DEFAULT_RECORDING_TIME_SEC","PROCESS_SHUTDOWN_TIMEOUT","SCREENRECORD_BINARY","DEFAULT_EXT","MIN_EMULATOR_API_LEVEL","FFMPEG_BINARY","system","isWindows","uploadRecordedMedia","localFile","remotePath","uploadOptions","_","isEmpty","util","toInMemoryBase64","toString","user","pass","method","headers","fileFieldName","formFields","options","auth","net","uploadFile","verifyScreenRecordIsSupported","adb","isEmulator","apiLevel","getApiLevel","Error","scheduleScreenRecord","recordingProperties","log","stopped","timer","videoSize","bitRate","timeLimit","bugReport","currentTimeLimit","hasValue","currentTimeLimitInt","parseInt","isNaN","pathOnDevice","uuidV4","substring","recordingProc","screenrecord","on","currentDuration","getDuration","asSeconds","toFixed","debug","timeLimitInt","chunkDuration","e","error","stack","start","waitForCondition","fileExists","waitMs","intervalMs","records","push","recordingProcess","mergeScreenRecords","mediaFiles","fs","which","configContent","map","x","join","configFile","path","resolve","dirname","writeFile","result","Math","floor","Date","args","info","exec","terminateBackgroundScreenRecording","force","pids","getPIDsByName","p","shell","err","message","startRecordingScreen","forceRestart","stopRecordingScreen","warn","_screenRecordingProperties","record","rimraf","timeout","parseFloat","timing","Timer","isRunning","stop","errorAndThrow","tmpRoot","tempDir","openDir","localRecords","posix","basename","pull","last","resultFilePath","length","size","stat","toReadableSizeString"],"sources":["../../../lib/commands/recordscreen.js"],"sourcesContent":["import _ from 'lodash';\nimport { waitForCondition } from 'asyncbox';\nimport { util, fs, net, tempDir, system, timing } from 'appium/support';\nimport { exec } from 'teen_process';\nimport path from 'path';\n\n\nconst commands = {};\n\nconst RETRY_PAUSE = 300;\nconst RETRY_TIMEOUT = 5000;\nconst MAX_RECORDING_TIME_SEC = 60 * 3;\nconst MAX_TIME_SEC = 60 * 30;\nconst DEFAULT_RECORDING_TIME_SEC = MAX_RECORDING_TIME_SEC;\nconst PROCESS_SHUTDOWN_TIMEOUT = 10 * 1000;\nconst SCREENRECORD_BINARY = 'screenrecord';\nconst DEFAULT_EXT = '.mp4';\nconst MIN_EMULATOR_API_LEVEL = 27;\nconst FFMPEG_BINARY = `ffmpeg${system.isWindows() ? '.exe' : ''}`;\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\nasync function verifyScreenRecordIsSupported (adb, isEmulator) {\n  const apiLevel = await adb.getApiLevel();\n  if (isEmulator && apiLevel < MIN_EMULATOR_API_LEVEL) {\n    throw new Error(`Screen recording does not work on emulators running Android API level less than ${MIN_EMULATOR_API_LEVEL}`);\n  }\n  if (apiLevel < 19) {\n    throw new Error(`Screen recording not available on API Level ${apiLevel}. Minimum API Level is 19.`);\n  }\n}\n\nasync function scheduleScreenRecord (adb, recordingProperties, log = null) {\n  if (recordingProperties.stopped) {\n    return;\n  }\n\n  const {\n    timer,\n    videoSize,\n    bitRate,\n    timeLimit,\n    bugReport,\n  } = recordingProperties;\n\n  let currentTimeLimit = MAX_RECORDING_TIME_SEC;\n  if (util.hasValue(recordingProperties.currentTimeLimit)) {\n    const currentTimeLimitInt = parseInt(recordingProperties.currentTimeLimit, 10);\n    if (!isNaN(currentTimeLimitInt) && currentTimeLimitInt < MAX_RECORDING_TIME_SEC) {\n      currentTimeLimit = currentTimeLimitInt;\n    }\n  }\n  const pathOnDevice = `/sdcard/${util.uuidV4().substring(0, 8)}${DEFAULT_EXT}`;\n  const recordingProc = adb.screenrecord(pathOnDevice, {\n    videoSize,\n    bitRate,\n    timeLimit: currentTimeLimit,\n    bugReport,\n  });\n\n  recordingProc.on('end', () => {\n    if (recordingProperties.stopped || !util.hasValue(timeLimit)) {\n      return;\n    }\n    const currentDuration = timer.getDuration().asSeconds.toFixed(0);\n    log?.debug(`The overall screen recording duration is ${currentDuration}s so far`);\n    const timeLimitInt = parseInt(timeLimit, 10);\n    if (isNaN(timeLimitInt) || currentDuration >= timeLimitInt) {\n      log?.debug('There is no need to start the next recording chunk');\n      return;\n    }\n\n    recordingProperties.currentTimeLimit = timeLimitInt - currentDuration;\n    const chunkDuration = recordingProperties.currentTimeLimit < MAX_RECORDING_TIME_SEC\n      ? recordingProperties.currentTimeLimit\n      : MAX_RECORDING_TIME_SEC;\n    log?.debug(`Starting the next ${chunkDuration}s-chunk ` +\n      `of screen recording in order to achieve ${timeLimitInt}s total duration`);\n    (async () => {\n      try {\n        await scheduleScreenRecord(adb, recordingProperties, log);\n      } catch (e) {\n        log?.error(e.stack);\n        recordingProperties.stopped = true;\n      }\n    })();\n  });\n\n  await recordingProc.start(0);\n  try {\n    await waitForCondition(async () => await adb.fileExists(pathOnDevice),\n      {waitMs: RETRY_TIMEOUT, intervalMs: RETRY_PAUSE});\n  } catch (e) {\n    throw new Error(`The expected screen record file '${pathOnDevice}' does not exist after ${RETRY_TIMEOUT}ms. ` +\n      `Is ${SCREENRECORD_BINARY} utility available and operational on the device under test?`);\n  }\n\n  recordingProperties.records.push(pathOnDevice);\n  recordingProperties.recordingProcess = recordingProc;\n}\n\nasync function mergeScreenRecords (mediaFiles, log = null) {\n  try {\n    await fs.which(FFMPEG_BINARY);\n  } catch (e) {\n    throw new Error(`${FFMPEG_BINARY} utility is not available in PATH. Please install it from https://www.ffmpeg.org/`);\n  }\n  const configContent = mediaFiles\n    .map((x) => `file '${x}'`)\n    .join('\\n');\n  const configFile = path.resolve(path.dirname(mediaFiles[0]), 'config.txt');\n  await fs.writeFile(configFile, configContent, 'utf8');\n  log?.debug(`Generated ffmpeg merging config '${configFile}' with items:\\n${configContent}`);\n  const result = path.resolve(path.dirname(mediaFiles[0]), `merge_${Math.floor(new Date())}${DEFAULT_EXT}`);\n  const args = ['-safe', '0', '-f', 'concat', '-i', configFile, '-c', 'copy', result];\n  log?.info(`Initiating screen records merging using the command '${FFMPEG_BINARY} ${args.join(' ')}'`);\n  await exec(FFMPEG_BINARY, args);\n  return result;\n}\n\nasync function terminateBackgroundScreenRecording (adb, force = true) {\n  const pids = (await adb.getPIDsByName(SCREENRECORD_BINARY))\n    .map((p) => `${p}`);\n  if (_.isEmpty(pids)) {\n    return false;\n  }\n\n  try {\n    await adb.shell(['kill', force ? '-15' : '-2', ...pids]);\n    await waitForCondition(async () => _.isEmpty(await adb.getPIDsByName(SCREENRECORD_BINARY)), {\n      waitMs: PROCESS_SHUTDOWN_TIMEOUT,\n      intervalMs: 500,\n    });\n    return true;\n  } catch (err) {\n    throw new Error(`Unable to stop the background screen recording: ${err.message}`);\n  }\n}\n\n\n/**\n * @typedef {Object} StartRecordingOptions\n *\n * @property {?string} remotePath - The path to the remote location, where the captured 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 endpount 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 *                                  This option only has an effect if there is screen recording process in progreess\n *                                  and `forceRestart` parameter is not set to `true`.\n * @property {?string} user - The name of the user for the remote authentication. Only works if `remotePath` is provided.\n * @property {?string} pass - The password for the remote authentication. Only works if `remotePath` is provided.\n * @property {?string} method [PUT] - The http multipart upload method name. Only works if `remotePath` is provided.\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 * @property {?string} videoSize - The format is widthxheight.\n *                  The default value is the device's native display resolution (if supported),\n *                  1280x720 if not. For best results,\n *                  use a size supported by your device's Advanced Video Coding (AVC) encoder.\n *                  For example, \"1280x720\"\n * @property {?boolean} bugReport - Set it to `true` in order to display additional information on the video overlay,\n *                                  such as a timestamp, that is helpful in videos captured to illustrate bugs.\n *                                  This option is only supported since API level 27 (Android P).\n * @property {?string|number} timeLimit - The maximum recording time, in seconds. The default value is 180 (3 minutes).\n *                                        The maximum value is 1800 (30 minutes). If the passed value is greater than 180 then\n *                                        the algorithm will try to schedule multiple screen recording chunks and merge the\n *                                        resulting videos into a single media file using `ffmpeg` utility.\n *                                        If the utility is not available in PATH then the most recent screen recording chunk is\n *                                        going to be returned.\n * @property {?string|number} bitRate - The video bit rate for the video, in bits per second.\n *                The default value is 4000000 (4 Mbit/s). You can increase the bit rate to improve video quality,\n *                but doing so results in larger movie files.\n * @property {?boolean} forceRestart - Whether to try to catch and upload/return the currently running screen recording\n *                                     (`false`, the default setting) or ignore the result of it and start a new recording\n *                                     immediately (`true`).\n */\n\n/**\n * Record the display of a real devices running Android 4.4 (API level 19) and higher.\n * Emulators are supported since API level 27 (Android P).\n * It records screen activity to an MPEG-4 file. Audio is not recorded with the video file.\n * If screen recording has been already started then the command will stop it forcefully and start a new one.\n * The previously recorded video file will be deleted.\n *\n * @param {?StartRecordingOptions} options - The available options.\n * @returns {string} Base64-encoded content of the recorded media file if\n *                   any screen recording is currently running or an empty string.\n * @throws {Error} If screen recording has failed to start or is not supported on the device under test.\n */\ncommands.startRecordingScreen = async function startRecordingScreen (options = {}) {\n  await verifyScreenRecordIsSupported(this.adb, this.isEmulator());\n\n  let result = '';\n  const {videoSize, timeLimit = DEFAULT_RECORDING_TIME_SEC, bugReport, bitRate, forceRestart} = options;\n  if (!forceRestart) {\n    result = await this.stopRecordingScreen(options);\n  }\n\n  if (await terminateBackgroundScreenRecording(this.adb, true)) {\n    this.log.warn(`There were some ${SCREENRECORD_BINARY} process leftovers running ` +\n      `in the background. Make sure you stop screen recording each time after it is started, ` +\n      `otherwise the recorded media might quickly exceed all the free space on the device under test.`);\n  }\n\n  if (!_.isEmpty(this._screenRecordingProperties)) {\n    for (const record of (this._screenRecordingProperties.records || [])) {\n      await this.adb.rimraf(record);\n    }\n    this._screenRecordingProperties = null;\n  }\n\n  const timeout = parseFloat(timeLimit);\n  if (isNaN(timeout) || timeout > MAX_TIME_SEC || timeout <= 0) {\n    throw new Error(`The timeLimit value must be in range [1, ${MAX_TIME_SEC}] seconds. ` +\n      `The value of '${timeLimit}' has been passed instead.`);\n  }\n\n  this._screenRecordingProperties = {\n    timer: new timing.Timer().start(),\n    videoSize,\n    timeLimit,\n    currentTimeLimit: timeLimit,\n    bitRate,\n    bugReport,\n    records: [],\n    recordingProcess: null,\n    stopped: false,\n  };\n  await scheduleScreenRecord(this.adb, this._screenRecordingProperties, this.log);\n  return result;\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 endpount 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 recording the screen.\n * If no screen recording has been started before then the method returns an empty string.\n *\n * @param {?StopRecordingOptions} options - The 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 getting the name of a media file\n *                 or the file content cannot be uploaded to the remote location\n *                 or screen recording is not supported on the device under test.\n */\ncommands.stopRecordingScreen = async function stopRecordingScreen (options = {}) {\n  await verifyScreenRecordIsSupported(this.adb, this.isEmulator());\n\n  if (!_.isEmpty(this._screenRecordingProperties)) {\n    this._screenRecordingProperties.stopped = true;\n  }\n\n  try {\n    await terminateBackgroundScreenRecording(this.adb, false);\n  } catch (err) {\n    this.log.warn(err.message);\n    if (!_.isEmpty(this._screenRecordingProperties)) {\n      this.log.warn('The resulting video might be corrupted');\n    }\n  }\n\n  if (_.isEmpty(this._screenRecordingProperties)) {\n    this.log.info(`Screen recording has not been previously started by Appium. There is nothing to stop`);\n    return '';\n  }\n\n  if (this._screenRecordingProperties.recordingProcess && this._screenRecordingProperties.recordingProcess.isRunning) {\n    try {\n      await this._screenRecordingProperties.recordingProcess.stop('SIGINT', PROCESS_SHUTDOWN_TIMEOUT);\n    } catch (e) {\n      this.log.errorAndThrow(`Unable to stop screen recording within ${PROCESS_SHUTDOWN_TIMEOUT}ms`);\n    }\n    this._screenRecordingProperties.recordingProcess = null;\n  }\n\n  if (_.isEmpty(this._screenRecordingProperties.records)) {\n    this.log.errorAndThrow(`No screen recordings have been stored on the device so far. ` +\n      `Are you sure the ${SCREENRECORD_BINARY} utility works as expected?`);\n  }\n\n  const tmpRoot = await tempDir.openDir();\n  try {\n    const localRecords = [];\n    for (const pathOnDevice of this._screenRecordingProperties.records) {\n      localRecords.push(path.resolve(tmpRoot, path.posix.basename(pathOnDevice)));\n      await this.adb.pull(pathOnDevice, _.last(localRecords));\n      await this.adb.rimraf(pathOnDevice);\n    }\n    let resultFilePath = _.last(localRecords);\n    if (localRecords.length > 1) {\n      this.log.info(`Got ${localRecords.length} screen recordings. Trying to merge them`);\n      try {\n        resultFilePath = await mergeScreenRecords(localRecords, this.log);\n      } catch (e) {\n        this.log.warn(`Cannot merge the recorded files. The most recent screen recording is going to be returned as the result. ` +\n          `Original error: ${e.message}`);\n      }\n    }\n    if (_.isEmpty(options.remotePath)) {\n      const {size} = await fs.stat(resultFilePath);\n      this.log.debug(`The size of the resulting screen recording is ${util.toReadableSizeString(size)}`);\n    }\n    return await uploadRecordedMedia(resultFilePath, options.remotePath, options);\n  } finally {\n    await fs.rimraf(tmpRoot);\n    this._screenRecordingProperties = null;\n  }\n};\n\n\nexport { commands };\nexport default commands;\n"],"mappings":"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"}
|
package/build/lib/driver.js
CHANGED
|
@@ -236,11 +236,16 @@ class AndroidDriver extends _driver.BaseDriver {
|
|
|
236
236
|
await this.initAUT();
|
|
237
237
|
this.bootstrap = new _androidHelpers.helpers.bootstrap(this.adb, this.opts.bootstrapPort, this.opts.websocket);
|
|
238
238
|
await this.bootstrap.start(this.opts.appPackage, this.opts.disableAndroidWatchers, this.opts.acceptSslCerts);
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
239
|
+
|
|
240
|
+
(async () => {
|
|
241
|
+
try {
|
|
242
|
+
await this.bootstrap.onUnexpectedShutdown;
|
|
243
|
+
} catch (err) {
|
|
244
|
+
if (!this.bootstrap.ignoreUnexpectedShutdown) {
|
|
245
|
+
await this.startUnexpectedShutdown(err);
|
|
246
|
+
}
|
|
242
247
|
}
|
|
243
|
-
});
|
|
248
|
+
})();
|
|
244
249
|
|
|
245
250
|
if (!this.opts.skipUnlock) {
|
|
246
251
|
await _androidHelpers.helpers.unlock(this, this.adb, this.caps);
|
|
@@ -504,4 +509,4 @@ class AndroidDriver extends _driver.BaseDriver {
|
|
|
504
509
|
exports.AndroidDriver = AndroidDriver;
|
|
505
510
|
var _default = AndroidDriver;
|
|
506
511
|
exports.default = _default;
|
|
507
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["APP_EXTENSION","DEVICE_PORT","NO_PROXY","RegExp","AndroidDriver","BaseDriver","constructor","opts","shouldValidateCaps","locatorStrategies","desiredCapConstraints","desiredConstraints","sessionChromedrivers","jwpProxyActive","jwpProxyAvoid","_","clone","settings","DeviceSettings","ignoreUnimportantViews","onSettingsUpdate","bind","chromedriver","apkStrings","unlocker","helpers","cmd","fn","toPairs","commands","prototype","curContext","defaultContextName","createSession","args","sessionId","caps","serverDetails","platform","webStorageEnabled","takesScreenshot","javascriptEnabled","databaseEnabled","networkConnectionEnabled","locationContextEnabled","warnings","desired","Object","assign","defaultOpts","action","category","flags","disableAndroidWatchers","tmpDir","tempDir","staticDir","fullReset","autoLaunch","adbPort","DEFAULT_ADB_PORT","bootstrapPort","androidInstallTimeout","defaults","useUnlockHelperApp","isUndefined","unlockType","noReset","fastReset","skipUninstall","isChromeSession","adjustBrowserSessionCaps","nativeWebScreenshot","push","reboot","setAvdFromCapabilities","udid","emPort","getDeviceInfoFromCaps","adb","createADB","suppressKillServer","remoteAdbHost","clearDeviceLogsOnStart","adbExecTimeout","allowOfflineDevices","getApiLevel","log","warn","isPackageOrBundle","app","appPackage","configureApp","checkAppPresent","appOnDevice","info","checkPackagePresent","util","hasValue","networkSpeed","isEmulator","ensureNetworkSpeed","gpsEnabled","toggleGPSLocationProvider","startAndroidSession","e","deleteSession","ign","avd","deviceName","errorAndThrow","platformVersion","avdDevice","replace","isChromeBrowser","browserName","key","value","setCompressedLayoutHierarchy","defaultIME","initDevice","curDeviceId","deviceUDID","getPlatformVersion","deviceScreenSize","getScreenSize","deviceModel","getModel","deviceManufacturer","getManufacturer","disableWindowAnimation","isAnimationOn","setHiddenApiPolicy","ignoreHiddenApiPolicyError","setAnimationState","_wasWindowAnimationDisabled","initAUT","bootstrap","websocket","start","acceptSslCerts","onUnexpectedShutdown","catch","err","ignoreUnexpectedShutdown","startUnexpectedShutdown","skipUnlock","unlock","update","startChromeSession","startAUT","orientation","debug","setOrientation","initAutoWebview","autoWebview","viewName","defaultWebviewName","timeout","autoWebviewTimeout","retryInterval","setContext","launchInfo","getLaunchInfo","uninstallOtherPackages","validateDesiredCaps","parseArray","SETTINGS_HELPER_PKG_ID","otherApps","message","B","all","map","installOtherApks","resetApp","uninstallApk","installApk","apkStringsForLanguage","pushStrings","language","sharedPreferences","setSharedPreferences","fs","exists","shell","compress","sendAction","compressLayout","isEmpty","_screenRecordingProperties","stopRecordingScreen","removeAllSessionWebSocketHandlers","server","mobileStopScreenStreaming","stopChromedriverProxies","unicodeKeyboard","resetKeyboard","setIME","dontStopAppOnReset","forceStop","goToHome","shutdown","stopLogcat","setDefaultHiddenApiPolicy","avdName","killEmulator","sharedPrefs","name","JSON","stringify","remotePath","remoteFile","localPath","builder","getPrefsBuilder","build","prefs","toFile","unlink","SharedPrefsBuilder","proxyActive","getProxyAvoidList","canProxy","isFunction","proxyReqRes"],"sources":["../../lib/driver.js"],"sourcesContent":["import { BaseDriver, DeviceSettings } from 'appium/driver';\nimport desiredConstraints from './desired-caps';\nimport commands from './commands/index';\nimport {\n  helpers, ensureNetworkSpeed,\n  SETTINGS_HELPER_PKG_ID,\n} from './android-helpers';\nimport _ from 'lodash';\nimport { DEFAULT_ADB_PORT } from 'appium-adb';\nimport { fs, tempDir, util } from 'appium/support';\nimport { retryInterval } from 'asyncbox';\nimport { SharedPrefsBuilder } from 'shared-preferences-builder';\nimport B from 'bluebird';\n\nconst APP_EXTENSION = '.apk';\nconst DEVICE_PORT = 4724;\n\n// This is a set of methods and paths that we never want to proxy to\n// Chromedriver\nconst NO_PROXY = [\n  ['POST', new RegExp('^/session/[^/]+/context')],\n  ['GET', new RegExp('^/session/[^/]+/context')],\n  ['POST', new RegExp('^/session/[^/]+/appium')],\n  ['GET', new RegExp('^/session/[^/]+/appium')],\n  ['POST', new RegExp('^/session/[^/]+/touch/perform')],\n  ['POST', new RegExp('^/session/[^/]+/touch/multi/perform')],\n  ['POST', new RegExp('^/session/[^/]+/orientation')],\n  ['GET', new RegExp('^/session/[^/]+/orientation')],\n  ['POST', new RegExp('^/session/[^/]+/execute')],\n  ['POST', new RegExp('^/session/[^/]+/execute/sync')],\n  ['GET', new RegExp('^/session/[^/]+/network_connection')],\n  ['POST', new RegExp('^/session/[^/]+/network_connection')],\n];\n\nclass AndroidDriver extends BaseDriver {\n  constructor (opts = {}, shouldValidateCaps = true) {\n    super(opts, shouldValidateCaps);\n\n    this.locatorStrategies = [\n      'xpath',\n      'id',\n      'class name',\n      'accessibility id',\n      '-android uiautomator'\n    ];\n    this.desiredCapConstraints = desiredConstraints;\n    this.sessionChromedrivers = {};\n    this.jwpProxyActive = false;\n    this.jwpProxyAvoid = _.clone(NO_PROXY);\n    this.settings = new DeviceSettings({ignoreUnimportantViews: false},\n                                       this.onSettingsUpdate.bind(this));\n    this.chromedriver = null;\n    this.apkStrings = {};\n    this.unlocker = helpers.unlocker;\n\n    for (let [cmd, fn] of _.toPairs(commands)) {\n      AndroidDriver.prototype[cmd] = fn;\n    }\n\n    // needs to be after the line which assigns commands to AndroidDriver.prototype, so that `this.defaultContextName` is defined.\n    this.curContext = this.defaultContextName();\n  }\n\n  async createSession (...args) {\n    // the whole createSession flow is surrounded in a try-catch statement\n    // if creating a session fails at any point, we teardown everything we\n    // set up before throwing the error.\n    try {\n      let [sessionId, caps] = await super.createSession(...args);\n\n      let serverDetails = {\n        platform: 'LINUX',\n        webStorageEnabled: false,\n        takesScreenshot: true,\n        javascriptEnabled: true,\n        databaseEnabled: false,\n        networkConnectionEnabled: true,\n        locationContextEnabled: false,\n        warnings: {},\n        desired: this.caps\n      };\n\n      this.caps = Object.assign(serverDetails, this.caps);\n\n      // assigning defaults\n      let defaultOpts = {\n        action: 'android.intent.action.MAIN',\n        category: 'android.intent.category.LAUNCHER',\n        flags: '0x10200000',\n        disableAndroidWatchers: false,\n        tmpDir: await tempDir.staticDir(),\n        fullReset: false,\n        autoLaunch: true,\n        adbPort: DEFAULT_ADB_PORT,\n        bootstrapPort: DEVICE_PORT,\n        androidInstallTimeout: 90000,\n      };\n      _.defaults(this.opts, defaultOpts);\n      this.useUnlockHelperApp = _.isUndefined(this.caps.unlockType);\n\n      // not user visible via caps\n      if (this.opts.noReset === true) {\n        this.opts.fullReset = false;\n      }\n      if (this.opts.fullReset === true) {\n        this.opts.noReset = false;\n      }\n      this.opts.fastReset = !this.opts.fullReset && !this.opts.noReset;\n      this.opts.skipUninstall = this.opts.fastReset || this.opts.noReset;\n\n      if (this.isChromeSession) {\n        helpers.adjustBrowserSessionCaps(this.opts);\n      }\n\n      if (this.opts.nativeWebScreenshot) {\n        this.jwpProxyAvoid.push(['GET', new RegExp('^/session/[^/]+/screenshot')]);\n      }\n\n      if (this.opts.reboot) {\n        this.setAvdFromCapabilities(caps);\n      }\n\n      // get device udid for this session\n      let {udid, emPort} = await helpers.getDeviceInfoFromCaps(this.opts);\n      this.opts.udid = udid;\n      this.opts.emPort = emPort;\n\n      // set up an instance of ADB\n      this.adb = await helpers.createADB({\n        udid: this.opts.udid,\n        emPort: this.opts.emPort,\n        adbPort: this.opts.adbPort,\n        suppressKillServer: this.opts.suppressKillServer,\n        remoteAdbHost: this.opts.remoteAdbHost,\n        clearDeviceLogsOnStart: this.opts.clearDeviceLogsOnStart,\n        adbExecTimeout: this.opts.adbExecTimeout,\n        allowOfflineDevices: this.opts.allowOfflineDevices,\n      });\n\n      if (await this.adb.getApiLevel() >= 23) {\n        this.log.warn(\"Consider setting 'automationName' capability to \" +\n          \"'uiautomator2' on Android >= 6, since UIAutomator framework \" +\n          'is not maintained anymore by the OS vendor.');\n      }\n\n      if (this.helpers.isPackageOrBundle(this.opts.app)) {\n        // user provided package instead of app for 'app' capability, massage options\n        this.opts.appPackage = this.opts.app;\n        this.opts.app = null;\n      }\n\n      if (this.opts.app) {\n        // find and copy, or download and unzip an app url or path\n        this.opts.app = await this.helpers.configureApp(this.opts.app, APP_EXTENSION);\n        await this.checkAppPresent();\n      } else if (this.appOnDevice) {\n        // the app isn't an actual app file but rather something we want to\n        // assume is on the device and just launch via the appPackage\n        this.log.info(`App file was not listed, instead we're going to run ` +\n          `${this.opts.appPackage} directly on the device`);\n        await this.checkPackagePresent();\n      }\n\n      // Some cloud services using appium launch the avd themselves, so we ensure netspeed\n      // is set for emulators by calling adb.networkSpeed before running the app\n      if (util.hasValue(this.opts.networkSpeed)) {\n        if (!this.isEmulator()) {\n          this.log.warn('Sorry, networkSpeed capability is only available for emulators');\n        } else {\n          const networkSpeed = ensureNetworkSpeed(this.adb, this.opts.networkSpeed);\n          await this.adb.networkSpeed(networkSpeed);\n        }\n      }\n      // check if we have to enable/disable gps before running the application\n      if (util.hasValue(this.opts.gpsEnabled)) {\n        if (this.isEmulator()) {\n          this.log.info(`Trying to ${this.opts.gpsEnabled ? 'enable' : 'disable'} gps location provider`);\n          await this.adb.toggleGPSLocationProvider(this.opts.gpsEnabled);\n        } else {\n          this.log.warn('Sorry! gpsEnabled capability is only available for emulators');\n        }\n      }\n\n      await this.startAndroidSession(this.opts);\n      return [sessionId, this.caps];\n    } catch (e) {\n      // ignoring delete session exception if any and throw the real error\n      // that happened while creating the session.\n      try {\n        await this.deleteSession();\n      } catch (ign) {}\n      throw e;\n    }\n  }\n\n  isEmulator () {\n    return helpers.isEmulator(this.adb, this.opts);\n  }\n\n  setAvdFromCapabilities (caps) {\n    if (this.opts.avd) {\n      this.log.info('avd name defined, ignoring device name and platform version');\n    } else {\n      if (!caps.deviceName) {\n        this.log.errorAndThrow('avd or deviceName should be specified when reboot option is enables');\n      }\n      if (!caps.platformVersion) {\n        this.log.errorAndThrow('avd or platformVersion should be specified when reboot option is enabled');\n      }\n      let avdDevice = caps.deviceName.replace(/[^a-zA-Z0-9_.]/g, '-');\n      this.opts.avd = `${avdDevice}__${caps.platformVersion}`;\n    }\n  }\n\n  get appOnDevice () {\n    return this.helpers.isPackageOrBundle(this.opts.app) || (!this.opts.app &&\n           this.helpers.isPackageOrBundle(this.opts.appPackage));\n  }\n\n  get isChromeSession () {\n    return helpers.isChromeBrowser(this.opts.browserName);\n  }\n\n  async onSettingsUpdate (key, value) {\n    if (key === 'ignoreUnimportantViews') {\n      await this.setCompressedLayoutHierarchy(value);\n    }\n  }\n\n  async startAndroidSession () {\n    this.log.info(`Starting Android session`);\n    // set up the device to run on (real or emulator, etc)\n    this.defaultIME = await helpers.initDevice(this.adb, this.opts);\n\n    // set actual device name, udid, platform version, screen size, model and manufacturer details.\n    this.caps.deviceName = this.adb.curDeviceId;\n    this.caps.deviceUDID = this.opts.udid;\n    this.caps.platformVersion = await this.adb.getPlatformVersion();\n    this.caps.deviceScreenSize = await this.adb.getScreenSize();\n    this.caps.deviceModel = await this.adb.getModel();\n    this.caps.deviceManufacturer = await this.adb.getManufacturer();\n\n    if (this.opts.disableWindowAnimation) {\n      if (await this.adb.isAnimationOn()) {\n        if (await this.adb.getApiLevel() >= 28) { // API level 28 is Android P\n          // Don't forget to reset the relaxing in delete session\n          this.log.warn('Relaxing hidden api policy to manage animation scale');\n          await this.adb.setHiddenApiPolicy('1', !!this.opts.ignoreHiddenApiPolicyError);\n        }\n\n        this.log.info('Disabling window animation as it is requested by \"disableWindowAnimation\" capability');\n        await this.adb.setAnimationState(false);\n        this._wasWindowAnimationDisabled = true;\n      } else {\n        this.log.info('Window animation is already disabled');\n      }\n    }\n\n    // set up app under test\n    await this.initAUT();\n\n    // start UiAutomator\n    this.bootstrap = new helpers.bootstrap(this.adb, this.opts.bootstrapPort, this.opts.websocket);\n    await this.bootstrap.start(this.opts.appPackage, this.opts.disableAndroidWatchers, this.opts.acceptSslCerts);\n    // handling unexpected shutdown\n    this.bootstrap.onUnexpectedShutdown.catch(async (err) => { // eslint-disable-line promise/prefer-await-to-callbacks\n      if (!this.bootstrap.ignoreUnexpectedShutdown) {\n        await this.startUnexpectedShutdown(err);\n      }\n    });\n\n    if (!this.opts.skipUnlock) {\n      // Let's try to unlock the device\n      await helpers.unlock(this, this.adb, this.caps);\n    }\n\n    // Set CompressedLayoutHierarchy on the device based on current settings object\n    // this has to happen _after_ bootstrap is initialized\n    if (this.opts.ignoreUnimportantViews) {\n      await this.settings.update({ignoreUnimportantViews: this.opts.ignoreUnimportantViews});\n    }\n\n    if (this.isChromeSession) {\n      // start a chromedriver session and proxy to it\n      await this.startChromeSession();\n    } else {\n      if (this.opts.autoLaunch) {\n        // start app\n        await this.startAUT();\n      }\n    }\n\n    if (util.hasValue(this.opts.orientation)) {\n      this.log.debug(`Setting initial orientation to '${this.opts.orientation}'`);\n      await this.setOrientation(this.opts.orientation);\n    }\n\n    await this.initAutoWebview();\n  }\n\n  async initAutoWebview () {\n    if (this.opts.autoWebview) {\n      let viewName = this.defaultWebviewName();\n      let timeout = (this.opts.autoWebviewTimeout) || 2000;\n\n      this.log.info(`Setting auto webview to context '${viewName}' with timeout ${timeout}ms`);\n\n      // try every 500ms until timeout is over\n      await retryInterval(timeout / 500, 500, async () => {\n        await this.setContext(viewName);\n      });\n    }\n  }\n\n  async initAUT () {\n    // populate appPackage, appActivity, appWaitPackage, appWaitActivity,\n    // and the device being used\n    // in the opts and caps (so it gets back to the user on session creation)\n    let launchInfo = await helpers.getLaunchInfo(this.adb, this.opts);\n    Object.assign(this.opts, launchInfo);\n    Object.assign(this.caps, launchInfo);\n\n    // Uninstall any uninstallOtherPackages which were specified in caps\n    if (this.opts.uninstallOtherPackages) {\n      helpers.validateDesiredCaps(this.opts);\n      // Only SETTINGS_HELPER_PKG_ID package is used by UIA1\n      await helpers.uninstallOtherPackages(\n        this.adb,\n        helpers.parseArray(this.opts.uninstallOtherPackages),\n        [SETTINGS_HELPER_PKG_ID]\n      );\n    }\n\n    // Install any \"otherApps\" that were specified in caps\n    if (this.opts.otherApps) {\n      let otherApps;\n      try {\n        otherApps = helpers.parseArray(this.opts.otherApps);\n      } catch (e) {\n        this.log.errorAndThrow(`Could not parse \"otherApps\" capability: ${e.message}`);\n      }\n      otherApps = await B.all(otherApps.map((app) => this.helpers.configureApp(app, APP_EXTENSION)));\n      await helpers.installOtherApks(otherApps, this.adb, this.opts);\n    }\n\n    // install app\n    if (!this.opts.app) {\n      if (this.opts.fullReset) {\n        this.log.errorAndThrow('Full reset requires an app capability, use fastReset if app is not provided');\n      }\n      this.log.debug('No app capability. Assuming it is already on the device');\n      if (this.opts.fastReset) {\n        await helpers.resetApp(this.adb, this.opts);\n      }\n      return;\n    }\n    if (!this.opts.skipUninstall) {\n      await this.adb.uninstallApk(this.opts.appPackage);\n    }\n    await helpers.installApk(this.adb, this.opts);\n    const apkStringsForLanguage = await helpers.pushStrings(this.opts.language, this.adb, this.opts);\n    if (this.opts.language) {\n      this.apkStrings[this.opts.language] = apkStringsForLanguage;\n    }\n\n    // This must run after installing the apk, otherwise it would cause the\n    // install to fail. And before running the app.\n    if (!_.isUndefined(this.opts.sharedPreferences)) {\n      await this.setSharedPreferences(this.opts);\n    }\n  }\n\n  async checkAppPresent () {\n    this.log.debug('Checking whether app is actually present');\n    if (!(await fs.exists(this.opts.app))) {\n      this.log.errorAndThrow(`Could not find app apk at ${this.opts.app}`);\n    }\n  }\n\n  async checkPackagePresent () {\n    this.log.debug('Checking whether package is present on the device');\n    if (!(await this.adb.shell(['pm', 'list', 'packages', this.opts.appPackage]))) {\n      this.log.errorAndThrow(`Could not find package ${this.opts.appPackage} on the device`);\n    }\n  }\n\n  // Set CompressedLayoutHierarchy on the device\n  async setCompressedLayoutHierarchy (compress) {\n    await this.bootstrap.sendAction('compressedLayoutHierarchy', {compressLayout: compress});\n  }\n\n  async deleteSession () {\n    this.log.debug('Shutting down Android driver');\n\n    try {\n      if (!_.isEmpty(this._screenRecordingProperties)) {\n        await this.stopRecordingScreen();\n      }\n    } catch (ign) {}\n\n    await helpers.removeAllSessionWebSocketHandlers(this.server, this.sessionId);\n\n    await this.mobileStopScreenStreaming();\n\n    await super.deleteSession();\n\n    if (this.bootstrap) {\n      // certain cleanup we only care to do if the bootstrap was ever run\n      await this.stopChromedriverProxies();\n      if (this.opts.unicodeKeyboard && this.opts.resetKeyboard && this.defaultIME) {\n        this.log.debug(`Resetting IME to ${this.defaultIME}`);\n        await this.adb?.setIME(this.defaultIME);\n      }\n      if (!this.isChromeSession && !this.opts.dontStopAppOnReset) {\n        await this.adb?.forceStop(this.opts.appPackage);\n      }\n      await this.adb?.goToHome();\n      if (this.opts.fullReset && !this.opts.skipUninstall && !this.appOnDevice) {\n        await this.adb?.uninstallApk(this.opts.appPackage);\n      }\n      await this.bootstrap.shutdown();\n      this.bootstrap = null;\n    } else {\n      this.log.debug(\"Called deleteSession but bootstrap wasn't active\");\n    }\n    // some cleanup we want to do regardless, in case we are shutting down\n    // mid-startup\n    await this.adb?.stopLogcat();\n    if (this.useUnlockHelperApp) {\n      await this.adb?.forceStop('io.appium.unlock');\n    }\n    if (this._wasWindowAnimationDisabled) {\n      this.log.info('Restoring window animation state');\n      await this.adb?.setAnimationState(true);\n\n      // This was necessary to change animation scale over Android P. We must reset the policy for the security.\n      if (await this.adb?.getApiLevel() >= 28) {\n        this.log.info('Restoring hidden api policy to the device default configuration');\n        await this.adb?.setDefaultHiddenApiPolicy(!!this.opts.ignoreHiddenApiPolicyError);\n      }\n    }\n\n    if (this.opts.reboot) {\n      let avdName = this.opts.avd.replace('@', '');\n      this.log.debug(`closing emulator '${avdName}'`);\n      await this.adb?.killEmulator(avdName);\n    }\n  }\n\n  async setSharedPreferences () {\n    let sharedPrefs = this.opts.sharedPreferences;\n    this.log.info('Trying to set shared preferences');\n    let name = sharedPrefs.name;\n    if (_.isUndefined(name)) {\n      this.log.warn(`Skipping setting Shared preferences, name is undefined: ${JSON.stringify(sharedPrefs)}`);\n      return false;\n    }\n    let remotePath = `/data/data/${this.opts.appPackage}/shared_prefs`;\n    let remoteFile = `${remotePath}/${name}.xml`;\n    let localPath = `/tmp/${name}.xml`;\n    let builder = this.getPrefsBuilder();\n    builder.build(sharedPrefs.prefs);\n    this.log.info(`Creating temporary shared preferences: ${localPath}`);\n    builder.toFile(localPath);\n    this.log.info(`Creating shared_prefs remote folder: ${remotePath}`);\n    await this.adb.shell(['mkdir', '-p', remotePath]);\n    this.log.info(`Pushing shared_prefs to ${remoteFile}`);\n    await this.adb.push(localPath, remoteFile);\n    try {\n      this.log.info(`Trying to remove shared preferences temporary file`);\n      if (await fs.exists(localPath)) {\n        await fs.unlink(localPath);\n      }\n    } catch (e) {\n      this.log.warn(`Error trying to remove temporary file ${localPath}`);\n    }\n    return true;\n  }\n\n  getPrefsBuilder () {\n    /* Add this method to create a new SharedPrefsBuilder instead of\n     * directly creating the object on setSharedPreferences for testing purposes\n    */\n    return new SharedPrefsBuilder();\n  }\n\n  validateDesiredCaps (caps) {\n    if (!super.validateDesiredCaps(caps)) {\n      return false;\n    }\n    if ((!caps.browserName || !helpers.isChromeBrowser(caps.browserName)) && !caps.app && !caps.appPackage) {\n      this.log.errorAndThrow('The desired capabilities must include either an app, appPackage or browserName');\n    }\n    return helpers.validateDesiredCaps(caps);\n  }\n\n  proxyActive (sessionId) {\n    super.proxyActive(sessionId);\n\n    return this.jwpProxyActive;\n  }\n\n  getProxyAvoidList (sessionId) {\n    super.getProxyAvoidList(sessionId);\n\n    return this.jwpProxyAvoid;\n  }\n\n  canProxy (sessionId) {\n    super.canProxy(sessionId);\n\n    // this will change depending on ChromeDriver status\n    return _.isFunction(this.proxyReqRes);\n  }\n}\n\nexport { AndroidDriver };\nexport default AndroidDriver;\n"],"mappings":"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"}
|
|
512
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["APP_EXTENSION","DEVICE_PORT","NO_PROXY","RegExp","AndroidDriver","BaseDriver","constructor","opts","shouldValidateCaps","locatorStrategies","desiredCapConstraints","desiredConstraints","sessionChromedrivers","jwpProxyActive","jwpProxyAvoid","_","clone","settings","DeviceSettings","ignoreUnimportantViews","onSettingsUpdate","bind","chromedriver","apkStrings","unlocker","helpers","cmd","fn","toPairs","commands","prototype","curContext","defaultContextName","createSession","args","sessionId","caps","serverDetails","platform","webStorageEnabled","takesScreenshot","javascriptEnabled","databaseEnabled","networkConnectionEnabled","locationContextEnabled","warnings","desired","Object","assign","defaultOpts","action","category","flags","disableAndroidWatchers","tmpDir","tempDir","staticDir","fullReset","autoLaunch","adbPort","DEFAULT_ADB_PORT","bootstrapPort","androidInstallTimeout","defaults","useUnlockHelperApp","isUndefined","unlockType","noReset","fastReset","skipUninstall","isChromeSession","adjustBrowserSessionCaps","nativeWebScreenshot","push","reboot","setAvdFromCapabilities","udid","emPort","getDeviceInfoFromCaps","adb","createADB","suppressKillServer","remoteAdbHost","clearDeviceLogsOnStart","adbExecTimeout","allowOfflineDevices","getApiLevel","log","warn","isPackageOrBundle","app","appPackage","configureApp","checkAppPresent","appOnDevice","info","checkPackagePresent","util","hasValue","networkSpeed","isEmulator","ensureNetworkSpeed","gpsEnabled","toggleGPSLocationProvider","startAndroidSession","e","deleteSession","ign","avd","deviceName","errorAndThrow","platformVersion","avdDevice","replace","isChromeBrowser","browserName","key","value","setCompressedLayoutHierarchy","defaultIME","initDevice","curDeviceId","deviceUDID","getPlatformVersion","deviceScreenSize","getScreenSize","deviceModel","getModel","deviceManufacturer","getManufacturer","disableWindowAnimation","isAnimationOn","setHiddenApiPolicy","ignoreHiddenApiPolicyError","setAnimationState","_wasWindowAnimationDisabled","initAUT","bootstrap","websocket","start","acceptSslCerts","onUnexpectedShutdown","err","ignoreUnexpectedShutdown","startUnexpectedShutdown","skipUnlock","unlock","update","startChromeSession","startAUT","orientation","debug","setOrientation","initAutoWebview","autoWebview","viewName","defaultWebviewName","timeout","autoWebviewTimeout","retryInterval","setContext","launchInfo","getLaunchInfo","uninstallOtherPackages","validateDesiredCaps","parseArray","SETTINGS_HELPER_PKG_ID","otherApps","message","B","all","map","installOtherApks","resetApp","uninstallApk","installApk","apkStringsForLanguage","pushStrings","language","sharedPreferences","setSharedPreferences","fs","exists","shell","compress","sendAction","compressLayout","isEmpty","_screenRecordingProperties","stopRecordingScreen","removeAllSessionWebSocketHandlers","server","mobileStopScreenStreaming","stopChromedriverProxies","unicodeKeyboard","resetKeyboard","setIME","dontStopAppOnReset","forceStop","goToHome","shutdown","stopLogcat","setDefaultHiddenApiPolicy","avdName","killEmulator","sharedPrefs","name","JSON","stringify","remotePath","remoteFile","localPath","builder","getPrefsBuilder","build","prefs","toFile","unlink","SharedPrefsBuilder","proxyActive","getProxyAvoidList","canProxy","isFunction","proxyReqRes"],"sources":["../../lib/driver.js"],"sourcesContent":["import { BaseDriver, DeviceSettings } from 'appium/driver';\nimport desiredConstraints from './desired-caps';\nimport commands from './commands/index';\nimport {\n  helpers, ensureNetworkSpeed,\n  SETTINGS_HELPER_PKG_ID,\n} from './android-helpers';\nimport _ from 'lodash';\nimport { DEFAULT_ADB_PORT } from 'appium-adb';\nimport { fs, tempDir, util } from 'appium/support';\nimport { retryInterval } from 'asyncbox';\nimport { SharedPrefsBuilder } from 'shared-preferences-builder';\nimport B from 'bluebird';\n\nconst APP_EXTENSION = '.apk';\nconst DEVICE_PORT = 4724;\n\n// This is a set of methods and paths that we never want to proxy to\n// Chromedriver\nconst NO_PROXY = [\n  ['POST', new RegExp('^/session/[^/]+/context')],\n  ['GET', new RegExp('^/session/[^/]+/context')],\n  ['POST', new RegExp('^/session/[^/]+/appium')],\n  ['GET', new RegExp('^/session/[^/]+/appium')],\n  ['POST', new RegExp('^/session/[^/]+/touch/perform')],\n  ['POST', new RegExp('^/session/[^/]+/touch/multi/perform')],\n  ['POST', new RegExp('^/session/[^/]+/orientation')],\n  ['GET', new RegExp('^/session/[^/]+/orientation')],\n  ['POST', new RegExp('^/session/[^/]+/execute')],\n  ['POST', new RegExp('^/session/[^/]+/execute/sync')],\n  ['GET', new RegExp('^/session/[^/]+/network_connection')],\n  ['POST', new RegExp('^/session/[^/]+/network_connection')],\n];\n\nclass AndroidDriver extends BaseDriver {\n  constructor (opts = {}, shouldValidateCaps = true) {\n    super(opts, shouldValidateCaps);\n\n    this.locatorStrategies = [\n      'xpath',\n      'id',\n      'class name',\n      'accessibility id',\n      '-android uiautomator'\n    ];\n    this.desiredCapConstraints = desiredConstraints;\n    this.sessionChromedrivers = {};\n    this.jwpProxyActive = false;\n    this.jwpProxyAvoid = _.clone(NO_PROXY);\n    this.settings = new DeviceSettings({ignoreUnimportantViews: false},\n                                       this.onSettingsUpdate.bind(this));\n    this.chromedriver = null;\n    this.apkStrings = {};\n    this.unlocker = helpers.unlocker;\n\n    for (let [cmd, fn] of _.toPairs(commands)) {\n      AndroidDriver.prototype[cmd] = fn;\n    }\n\n    // needs to be after the line which assigns commands to AndroidDriver.prototype, so that `this.defaultContextName` is defined.\n    this.curContext = this.defaultContextName();\n  }\n\n  async createSession (...args) {\n    // the whole createSession flow is surrounded in a try-catch statement\n    // if creating a session fails at any point, we teardown everything we\n    // set up before throwing the error.\n    try {\n      let [sessionId, caps] = await super.createSession(...args);\n\n      let serverDetails = {\n        platform: 'LINUX',\n        webStorageEnabled: false,\n        takesScreenshot: true,\n        javascriptEnabled: true,\n        databaseEnabled: false,\n        networkConnectionEnabled: true,\n        locationContextEnabled: false,\n        warnings: {},\n        desired: this.caps\n      };\n\n      this.caps = Object.assign(serverDetails, this.caps);\n\n      // assigning defaults\n      let defaultOpts = {\n        action: 'android.intent.action.MAIN',\n        category: 'android.intent.category.LAUNCHER',\n        flags: '0x10200000',\n        disableAndroidWatchers: false,\n        tmpDir: await tempDir.staticDir(),\n        fullReset: false,\n        autoLaunch: true,\n        adbPort: DEFAULT_ADB_PORT,\n        bootstrapPort: DEVICE_PORT,\n        androidInstallTimeout: 90000,\n      };\n      _.defaults(this.opts, defaultOpts);\n      this.useUnlockHelperApp = _.isUndefined(this.caps.unlockType);\n\n      // not user visible via caps\n      if (this.opts.noReset === true) {\n        this.opts.fullReset = false;\n      }\n      if (this.opts.fullReset === true) {\n        this.opts.noReset = false;\n      }\n      this.opts.fastReset = !this.opts.fullReset && !this.opts.noReset;\n      this.opts.skipUninstall = this.opts.fastReset || this.opts.noReset;\n\n      if (this.isChromeSession) {\n        helpers.adjustBrowserSessionCaps(this.opts);\n      }\n\n      if (this.opts.nativeWebScreenshot) {\n        this.jwpProxyAvoid.push(['GET', new RegExp('^/session/[^/]+/screenshot')]);\n      }\n\n      if (this.opts.reboot) {\n        this.setAvdFromCapabilities(caps);\n      }\n\n      // get device udid for this session\n      let {udid, emPort} = await helpers.getDeviceInfoFromCaps(this.opts);\n      this.opts.udid = udid;\n      this.opts.emPort = emPort;\n\n      // set up an instance of ADB\n      this.adb = await helpers.createADB({\n        udid: this.opts.udid,\n        emPort: this.opts.emPort,\n        adbPort: this.opts.adbPort,\n        suppressKillServer: this.opts.suppressKillServer,\n        remoteAdbHost: this.opts.remoteAdbHost,\n        clearDeviceLogsOnStart: this.opts.clearDeviceLogsOnStart,\n        adbExecTimeout: this.opts.adbExecTimeout,\n        allowOfflineDevices: this.opts.allowOfflineDevices,\n      });\n\n      if (await this.adb.getApiLevel() >= 23) {\n        this.log.warn(\"Consider setting 'automationName' capability to \" +\n          \"'uiautomator2' on Android >= 6, since UIAutomator framework \" +\n          'is not maintained anymore by the OS vendor.');\n      }\n\n      if (this.helpers.isPackageOrBundle(this.opts.app)) {\n        // user provided package instead of app for 'app' capability, massage options\n        this.opts.appPackage = this.opts.app;\n        this.opts.app = null;\n      }\n\n      if (this.opts.app) {\n        // find and copy, or download and unzip an app url or path\n        this.opts.app = await this.helpers.configureApp(this.opts.app, APP_EXTENSION);\n        await this.checkAppPresent();\n      } else if (this.appOnDevice) {\n        // the app isn't an actual app file but rather something we want to\n        // assume is on the device and just launch via the appPackage\n        this.log.info(`App file was not listed, instead we're going to run ` +\n          `${this.opts.appPackage} directly on the device`);\n        await this.checkPackagePresent();\n      }\n\n      // Some cloud services using appium launch the avd themselves, so we ensure netspeed\n      // is set for emulators by calling adb.networkSpeed before running the app\n      if (util.hasValue(this.opts.networkSpeed)) {\n        if (!this.isEmulator()) {\n          this.log.warn('Sorry, networkSpeed capability is only available for emulators');\n        } else {\n          const networkSpeed = ensureNetworkSpeed(this.adb, this.opts.networkSpeed);\n          await this.adb.networkSpeed(networkSpeed);\n        }\n      }\n      // check if we have to enable/disable gps before running the application\n      if (util.hasValue(this.opts.gpsEnabled)) {\n        if (this.isEmulator()) {\n          this.log.info(`Trying to ${this.opts.gpsEnabled ? 'enable' : 'disable'} gps location provider`);\n          await this.adb.toggleGPSLocationProvider(this.opts.gpsEnabled);\n        } else {\n          this.log.warn('Sorry! gpsEnabled capability is only available for emulators');\n        }\n      }\n\n      await this.startAndroidSession(this.opts);\n      return [sessionId, this.caps];\n    } catch (e) {\n      // ignoring delete session exception if any and throw the real error\n      // that happened while creating the session.\n      try {\n        await this.deleteSession();\n      } catch (ign) {}\n      throw e;\n    }\n  }\n\n  isEmulator () {\n    return helpers.isEmulator(this.adb, this.opts);\n  }\n\n  setAvdFromCapabilities (caps) {\n    if (this.opts.avd) {\n      this.log.info('avd name defined, ignoring device name and platform version');\n    } else {\n      if (!caps.deviceName) {\n        this.log.errorAndThrow('avd or deviceName should be specified when reboot option is enables');\n      }\n      if (!caps.platformVersion) {\n        this.log.errorAndThrow('avd or platformVersion should be specified when reboot option is enabled');\n      }\n      let avdDevice = caps.deviceName.replace(/[^a-zA-Z0-9_.]/g, '-');\n      this.opts.avd = `${avdDevice}__${caps.platformVersion}`;\n    }\n  }\n\n  get appOnDevice () {\n    return this.helpers.isPackageOrBundle(this.opts.app) || (!this.opts.app &&\n           this.helpers.isPackageOrBundle(this.opts.appPackage));\n  }\n\n  get isChromeSession () {\n    return helpers.isChromeBrowser(this.opts.browserName);\n  }\n\n  async onSettingsUpdate (key, value) {\n    if (key === 'ignoreUnimportantViews') {\n      await this.setCompressedLayoutHierarchy(value);\n    }\n  }\n\n  async startAndroidSession () {\n    this.log.info(`Starting Android session`);\n    // set up the device to run on (real or emulator, etc)\n    this.defaultIME = await helpers.initDevice(this.adb, this.opts);\n\n    // set actual device name, udid, platform version, screen size, model and manufacturer details.\n    this.caps.deviceName = this.adb.curDeviceId;\n    this.caps.deviceUDID = this.opts.udid;\n    this.caps.platformVersion = await this.adb.getPlatformVersion();\n    this.caps.deviceScreenSize = await this.adb.getScreenSize();\n    this.caps.deviceModel = await this.adb.getModel();\n    this.caps.deviceManufacturer = await this.adb.getManufacturer();\n\n    if (this.opts.disableWindowAnimation) {\n      if (await this.adb.isAnimationOn()) {\n        if (await this.adb.getApiLevel() >= 28) { // API level 28 is Android P\n          // Don't forget to reset the relaxing in delete session\n          this.log.warn('Relaxing hidden api policy to manage animation scale');\n          await this.adb.setHiddenApiPolicy('1', !!this.opts.ignoreHiddenApiPolicyError);\n        }\n\n        this.log.info('Disabling window animation as it is requested by \"disableWindowAnimation\" capability');\n        await this.adb.setAnimationState(false);\n        this._wasWindowAnimationDisabled = true;\n      } else {\n        this.log.info('Window animation is already disabled');\n      }\n    }\n\n    // set up app under test\n    await this.initAUT();\n\n    // start UiAutomator\n    this.bootstrap = new helpers.bootstrap(this.adb, this.opts.bootstrapPort, this.opts.websocket);\n    await this.bootstrap.start(this.opts.appPackage, this.opts.disableAndroidWatchers, this.opts.acceptSslCerts);\n    // handling unexpected shutdown\n    (async () => {\n      try {\n        await this.bootstrap.onUnexpectedShutdown;\n      } catch (err) {\n        if (!this.bootstrap.ignoreUnexpectedShutdown) {\n          await this.startUnexpectedShutdown(err);\n        }\n      }\n    })();\n\n\n    if (!this.opts.skipUnlock) {\n      // Let's try to unlock the device\n      await helpers.unlock(this, this.adb, this.caps);\n    }\n\n    // Set CompressedLayoutHierarchy on the device based on current settings object\n    // this has to happen _after_ bootstrap is initialized\n    if (this.opts.ignoreUnimportantViews) {\n      await this.settings.update({ignoreUnimportantViews: this.opts.ignoreUnimportantViews});\n    }\n\n    if (this.isChromeSession) {\n      // start a chromedriver session and proxy to it\n      await this.startChromeSession();\n    } else {\n      if (this.opts.autoLaunch) {\n        // start app\n        await this.startAUT();\n      }\n    }\n\n    if (util.hasValue(this.opts.orientation)) {\n      this.log.debug(`Setting initial orientation to '${this.opts.orientation}'`);\n      await this.setOrientation(this.opts.orientation);\n    }\n\n    await this.initAutoWebview();\n  }\n\n  async initAutoWebview () {\n    if (this.opts.autoWebview) {\n      let viewName = this.defaultWebviewName();\n      let timeout = (this.opts.autoWebviewTimeout) || 2000;\n\n      this.log.info(`Setting auto webview to context '${viewName}' with timeout ${timeout}ms`);\n\n      // try every 500ms until timeout is over\n      await retryInterval(timeout / 500, 500, async () => {\n        await this.setContext(viewName);\n      });\n    }\n  }\n\n  async initAUT () {\n    // populate appPackage, appActivity, appWaitPackage, appWaitActivity,\n    // and the device being used\n    // in the opts and caps (so it gets back to the user on session creation)\n    let launchInfo = await helpers.getLaunchInfo(this.adb, this.opts);\n    Object.assign(this.opts, launchInfo);\n    Object.assign(this.caps, launchInfo);\n\n    // Uninstall any uninstallOtherPackages which were specified in caps\n    if (this.opts.uninstallOtherPackages) {\n      helpers.validateDesiredCaps(this.opts);\n      // Only SETTINGS_HELPER_PKG_ID package is used by UIA1\n      await helpers.uninstallOtherPackages(\n        this.adb,\n        helpers.parseArray(this.opts.uninstallOtherPackages),\n        [SETTINGS_HELPER_PKG_ID]\n      );\n    }\n\n    // Install any \"otherApps\" that were specified in caps\n    if (this.opts.otherApps) {\n      let otherApps;\n      try {\n        otherApps = helpers.parseArray(this.opts.otherApps);\n      } catch (e) {\n        this.log.errorAndThrow(`Could not parse \"otherApps\" capability: ${e.message}`);\n      }\n      otherApps = await B.all(otherApps.map((app) => this.helpers.configureApp(app, APP_EXTENSION)));\n      await helpers.installOtherApks(otherApps, this.adb, this.opts);\n    }\n\n    // install app\n    if (!this.opts.app) {\n      if (this.opts.fullReset) {\n        this.log.errorAndThrow('Full reset requires an app capability, use fastReset if app is not provided');\n      }\n      this.log.debug('No app capability. Assuming it is already on the device');\n      if (this.opts.fastReset) {\n        await helpers.resetApp(this.adb, this.opts);\n      }\n      return;\n    }\n    if (!this.opts.skipUninstall) {\n      await this.adb.uninstallApk(this.opts.appPackage);\n    }\n    await helpers.installApk(this.adb, this.opts);\n    const apkStringsForLanguage = await helpers.pushStrings(this.opts.language, this.adb, this.opts);\n    if (this.opts.language) {\n      this.apkStrings[this.opts.language] = apkStringsForLanguage;\n    }\n\n    // This must run after installing the apk, otherwise it would cause the\n    // install to fail. And before running the app.\n    if (!_.isUndefined(this.opts.sharedPreferences)) {\n      await this.setSharedPreferences(this.opts);\n    }\n  }\n\n  async checkAppPresent () {\n    this.log.debug('Checking whether app is actually present');\n    if (!(await fs.exists(this.opts.app))) {\n      this.log.errorAndThrow(`Could not find app apk at ${this.opts.app}`);\n    }\n  }\n\n  async checkPackagePresent () {\n    this.log.debug('Checking whether package is present on the device');\n    if (!(await this.adb.shell(['pm', 'list', 'packages', this.opts.appPackage]))) {\n      this.log.errorAndThrow(`Could not find package ${this.opts.appPackage} on the device`);\n    }\n  }\n\n  // Set CompressedLayoutHierarchy on the device\n  async setCompressedLayoutHierarchy (compress) {\n    await this.bootstrap.sendAction('compressedLayoutHierarchy', {compressLayout: compress});\n  }\n\n  async deleteSession () {\n    this.log.debug('Shutting down Android driver');\n\n    try {\n      if (!_.isEmpty(this._screenRecordingProperties)) {\n        await this.stopRecordingScreen();\n      }\n    } catch (ign) {}\n\n    await helpers.removeAllSessionWebSocketHandlers(this.server, this.sessionId);\n\n    await this.mobileStopScreenStreaming();\n\n    await super.deleteSession();\n\n    if (this.bootstrap) {\n      // certain cleanup we only care to do if the bootstrap was ever run\n      await this.stopChromedriverProxies();\n      if (this.opts.unicodeKeyboard && this.opts.resetKeyboard && this.defaultIME) {\n        this.log.debug(`Resetting IME to ${this.defaultIME}`);\n        await this.adb?.setIME(this.defaultIME);\n      }\n      if (!this.isChromeSession && !this.opts.dontStopAppOnReset) {\n        await this.adb?.forceStop(this.opts.appPackage);\n      }\n      await this.adb?.goToHome();\n      if (this.opts.fullReset && !this.opts.skipUninstall && !this.appOnDevice) {\n        await this.adb?.uninstallApk(this.opts.appPackage);\n      }\n      await this.bootstrap.shutdown();\n      this.bootstrap = null;\n    } else {\n      this.log.debug(\"Called deleteSession but bootstrap wasn't active\");\n    }\n    // some cleanup we want to do regardless, in case we are shutting down\n    // mid-startup\n    await this.adb?.stopLogcat();\n    if (this.useUnlockHelperApp) {\n      await this.adb?.forceStop('io.appium.unlock');\n    }\n    if (this._wasWindowAnimationDisabled) {\n      this.log.info('Restoring window animation state');\n      await this.adb?.setAnimationState(true);\n\n      // This was necessary to change animation scale over Android P. We must reset the policy for the security.\n      if (await this.adb?.getApiLevel() >= 28) {\n        this.log.info('Restoring hidden api policy to the device default configuration');\n        await this.adb?.setDefaultHiddenApiPolicy(!!this.opts.ignoreHiddenApiPolicyError);\n      }\n    }\n\n    if (this.opts.reboot) {\n      let avdName = this.opts.avd.replace('@', '');\n      this.log.debug(`closing emulator '${avdName}'`);\n      await this.adb?.killEmulator(avdName);\n    }\n  }\n\n  async setSharedPreferences () {\n    let sharedPrefs = this.opts.sharedPreferences;\n    this.log.info('Trying to set shared preferences');\n    let name = sharedPrefs.name;\n    if (_.isUndefined(name)) {\n      this.log.warn(`Skipping setting Shared preferences, name is undefined: ${JSON.stringify(sharedPrefs)}`);\n      return false;\n    }\n    let remotePath = `/data/data/${this.opts.appPackage}/shared_prefs`;\n    let remoteFile = `${remotePath}/${name}.xml`;\n    let localPath = `/tmp/${name}.xml`;\n    let builder = this.getPrefsBuilder();\n    builder.build(sharedPrefs.prefs);\n    this.log.info(`Creating temporary shared preferences: ${localPath}`);\n    builder.toFile(localPath);\n    this.log.info(`Creating shared_prefs remote folder: ${remotePath}`);\n    await this.adb.shell(['mkdir', '-p', remotePath]);\n    this.log.info(`Pushing shared_prefs to ${remoteFile}`);\n    await this.adb.push(localPath, remoteFile);\n    try {\n      this.log.info(`Trying to remove shared preferences temporary file`);\n      if (await fs.exists(localPath)) {\n        await fs.unlink(localPath);\n      }\n    } catch (e) {\n      this.log.warn(`Error trying to remove temporary file ${localPath}`);\n    }\n    return true;\n  }\n\n  getPrefsBuilder () {\n    /* Add this method to create a new SharedPrefsBuilder instead of\n     * directly creating the object on setSharedPreferences for testing purposes\n    */\n    return new SharedPrefsBuilder();\n  }\n\n  validateDesiredCaps (caps) {\n    if (!super.validateDesiredCaps(caps)) {\n      return false;\n    }\n    if ((!caps.browserName || !helpers.isChromeBrowser(caps.browserName)) && !caps.app && !caps.appPackage) {\n      this.log.errorAndThrow('The desired capabilities must include either an app, appPackage or browserName');\n    }\n    return helpers.validateDesiredCaps(caps);\n  }\n\n  proxyActive (sessionId) {\n    super.proxyActive(sessionId);\n\n    return this.jwpProxyActive;\n  }\n\n  getProxyAvoidList (sessionId) {\n    super.getProxyAvoidList(sessionId);\n\n    return this.jwpProxyAvoid;\n  }\n\n  canProxy (sessionId) {\n    super.canProxy(sessionId);\n\n    // this will change depending on ChromeDriver status\n    return _.isFunction(this.proxyReqRes);\n  }\n}\n\nexport { AndroidDriver };\nexport default AndroidDriver;\n"],"mappings":"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"}
|
|
@@ -7,7 +7,23 @@ import { errors } from 'appium/driver';
|
|
|
7
7
|
const APP_EXTENSIONS = ['.apk', '.apks'];
|
|
8
8
|
const RESOLVER_ACTIVITY_NAME = 'android/com.android.internal.app.ResolverActivity';
|
|
9
9
|
|
|
10
|
-
|
|
10
|
+
const commands = {};
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Assert the presence of particular keys in the given object
|
|
14
|
+
*
|
|
15
|
+
* @param {string|Array<string>} argNames one or more key names
|
|
16
|
+
* @param {Object} opts the object to check
|
|
17
|
+
* @returns {Object} the same given object
|
|
18
|
+
*/
|
|
19
|
+
function requireArgs (argNames, opts = {}) {
|
|
20
|
+
for (const argName of (_.isArray(argNames) ? argNames : [argNames])) {
|
|
21
|
+
if (!_.has(opts, argName)) {
|
|
22
|
+
throw new errors.InvalidArgumentError(`'${argName}' argument must be provided`);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
return opts;
|
|
26
|
+
}
|
|
11
27
|
|
|
12
28
|
/**
|
|
13
29
|
* Verify whether an application is installed or not
|
|
@@ -19,6 +35,22 @@ commands.isAppInstalled = async function isAppInstalled (appId) {
|
|
|
19
35
|
return await this.adb.isAppInstalled(appId);
|
|
20
36
|
};
|
|
21
37
|
|
|
38
|
+
/**
|
|
39
|
+
* @typedef {Object} MobileAppInstalledOptions
|
|
40
|
+
* @property {string} appId - Application package identifier. Must be always provided.
|
|
41
|
+
*/
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Verify whether an application is installed or not
|
|
45
|
+
*
|
|
46
|
+
* @param {MobileAppInstalledOptions} opts
|
|
47
|
+
* @returns {boolean} Same as in `isAppInstalled`
|
|
48
|
+
*/
|
|
49
|
+
commands.mobileIsAppInstalled = async function mobileIsAppInstalled (opts = {}) {
|
|
50
|
+
const { appId } = requireArgs('appId', opts);
|
|
51
|
+
return await this.isAppInstalled(appId);
|
|
52
|
+
};
|
|
53
|
+
|
|
22
54
|
/**
|
|
23
55
|
* Queries the current state of the app.
|
|
24
56
|
*
|
|
@@ -47,12 +79,29 @@ commands.queryAppState = async function queryAppState (appId) {
|
|
|
47
79
|
return APP_STATE.RUNNING_IN_BACKGROUND;
|
|
48
80
|
};
|
|
49
81
|
|
|
82
|
+
/**
|
|
83
|
+
* @typedef {Object} MobileQueryAppStateOptions
|
|
84
|
+
* @property {string} appId - Application package identifier. Must be always provided.
|
|
85
|
+
*/
|
|
86
|
+
|
|
87
|
+
/**
|
|
88
|
+
* Queries the current state of the app.
|
|
89
|
+
*
|
|
90
|
+
* @param {MobileQueryAppStateOptions} opts
|
|
91
|
+
* @returns {number} Same as in `queryAppState`
|
|
92
|
+
*/
|
|
93
|
+
commands.mobileQueryAppState = async function mobileQueryAppState (opts = {}) {
|
|
94
|
+
const { appId } = requireArgs('appId', opts);
|
|
95
|
+
return await this.queryAppState(appId);
|
|
96
|
+
};
|
|
97
|
+
|
|
50
98
|
/**
|
|
51
99
|
* Activates the given application or launches it if necessary.
|
|
52
|
-
* The action
|
|
100
|
+
* The action literally simulates
|
|
53
101
|
* clicking the corresponding application icon on the dashboard.
|
|
54
102
|
*
|
|
55
103
|
* @param {string} appId - Application package identifier
|
|
104
|
+
* @throws {Error} If the app cannot be activated
|
|
56
105
|
*/
|
|
57
106
|
commands.activateApp = async function activateApp (appId) {
|
|
58
107
|
this.log.debug(`Activating '${appId}'`);
|
|
@@ -104,6 +153,24 @@ commands.activateApp = async function activateApp (appId) {
|
|
|
104
153
|
}
|
|
105
154
|
};
|
|
106
155
|
|
|
156
|
+
/**
|
|
157
|
+
* @typedef {Object} MobileActivateAppOptions
|
|
158
|
+
* @property {string} appId - Application package identifier. Must be always provided.
|
|
159
|
+
*/
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* Activates the given application or launches it if necessary.
|
|
163
|
+
* The action literally simulates
|
|
164
|
+
* clicking the corresponding application icon on the dashboard.
|
|
165
|
+
*
|
|
166
|
+
* @param {MobileActivateAppOptions} opts
|
|
167
|
+
* @throws {Error} If the app cannot be activated
|
|
168
|
+
*/
|
|
169
|
+
commands.mobileActivateApp = async function mobileActivateApp (opts = {}) {
|
|
170
|
+
const { appId } = requireArgs('appId', opts);
|
|
171
|
+
return await this.activateApp(appId);
|
|
172
|
+
};
|
|
173
|
+
|
|
107
174
|
/**
|
|
108
175
|
* @typedef {Object} UninstallOptions
|
|
109
176
|
* @property {number} timeout [20000] - The count of milliseconds to wait until the
|
|
@@ -125,6 +192,23 @@ commands.removeApp = async function removeApp (appId, options = {}) {
|
|
|
125
192
|
return await this.adb.uninstallApk(appId, options);
|
|
126
193
|
};
|
|
127
194
|
|
|
195
|
+
/**
|
|
196
|
+
* @typedef {Object} MobileRemoveAppOptions
|
|
197
|
+
* @property {string} appId - Application package identifier. Must be always provided.
|
|
198
|
+
*/
|
|
199
|
+
|
|
200
|
+
/**
|
|
201
|
+
* Remove the corresponding application if is installed.
|
|
202
|
+
* The call is ignored if the app is not installed.
|
|
203
|
+
*
|
|
204
|
+
* @param {MobileRemoveAppOptions} opts
|
|
205
|
+
* @returns {boolean} Same as in `removeApp`
|
|
206
|
+
*/
|
|
207
|
+
commands.mobileRemoveApp = async function mobileRemoveApp (opts = {}) {
|
|
208
|
+
const { appId } = requireArgs('appId', opts);
|
|
209
|
+
return await this.removeApp(appId, opts);
|
|
210
|
+
};
|
|
211
|
+
|
|
128
212
|
/**
|
|
129
213
|
* @typedef {Object} TerminateOptions
|
|
130
214
|
* @property {number|string} timeout [500] - The count of milliseconds to wait until the
|
|
@@ -157,6 +241,25 @@ commands.terminateApp = async function terminateApp (appId, options = {}) {
|
|
|
157
241
|
return true;
|
|
158
242
|
};
|
|
159
243
|
|
|
244
|
+
/**
|
|
245
|
+
* @typedef {Object} MobileTerminateAppOptions
|
|
246
|
+
* @property {string} appId - Application package identifier. Must be always provided.
|
|
247
|
+
* @property {number|string} timeout [500] - The count of milliseconds to wait until the
|
|
248
|
+
* app is terminated.
|
|
249
|
+
*/
|
|
250
|
+
|
|
251
|
+
/**
|
|
252
|
+
* Terminates the app if it is running.
|
|
253
|
+
*
|
|
254
|
+
* @param {MobileTerminateAppOptions} opts
|
|
255
|
+
* @returns {boolean} Same as in `terminateApp`
|
|
256
|
+
* @throws {Error} if the app has not been terminated within the given timeout.
|
|
257
|
+
*/
|
|
258
|
+
commands.mobileTerminateApp = async function mobileTerminateApp (opts = {}) {
|
|
259
|
+
const { appId } = requireArgs('appId', opts);
|
|
260
|
+
return await this.terminateApp(appId, opts);
|
|
261
|
+
};
|
|
262
|
+
|
|
160
263
|
/**
|
|
161
264
|
* @typedef {Object} InstallOptions
|
|
162
265
|
* @property {number} timeout [60000] - The count of milliseconds to wait until the
|
|
@@ -186,6 +289,35 @@ commands.installApp = async function installApp (appPath, options = {}) {
|
|
|
186
289
|
await this.adb.install(localPath, options);
|
|
187
290
|
};
|
|
188
291
|
|
|
292
|
+
/**
|
|
293
|
+
* @typedef {Object} MobileInstallAppOptions
|
|
294
|
+
* @property {string} appPath - The local apk path or a remote url. Must be always provided.
|
|
295
|
+
* @property {number} timeout [60000] - The count of milliseconds to wait until the
|
|
296
|
+
* app is installed.
|
|
297
|
+
* @property {boolean} allowTestPackages [false] - Set to true in order to allow test
|
|
298
|
+
* packages installation.
|
|
299
|
+
* @property {boolean} useSdcard [false] - Set to true to install the app on sdcard
|
|
300
|
+
* instead of the device memory.
|
|
301
|
+
* @property {boolean} grantPermissions [false] - Set to true in order to grant all the
|
|
302
|
+
* permissions requested in the application's manifest
|
|
303
|
+
* automatically after the installation is completed
|
|
304
|
+
* under Android 6+.
|
|
305
|
+
* @property {boolean} replace [true] - Set it to false if you don't want
|
|
306
|
+
* the application to be upgraded/reinstalled
|
|
307
|
+
* if it is already present on the device.
|
|
308
|
+
*/
|
|
309
|
+
|
|
310
|
+
/**
|
|
311
|
+
* Installs the given application to the device under test
|
|
312
|
+
*
|
|
313
|
+
* @param {MobileInstallAppOptions} opts
|
|
314
|
+
* @throws {Error} if the given apk does not exist or is not reachable
|
|
315
|
+
*/
|
|
316
|
+
commands.mobileInstallApp = async function mobileInstallApp (opts = {}) {
|
|
317
|
+
const { appPath } = requireArgs('appPath', opts);
|
|
318
|
+
return await this.installApp(appPath, opts);
|
|
319
|
+
};
|
|
320
|
+
|
|
189
321
|
/**
|
|
190
322
|
* @typedef {Object} ClearAppOptions
|
|
191
323
|
* @property {!string} appId The identifier of the application package to be cleared
|
package/lib/commands/execute.js
CHANGED
|
@@ -48,6 +48,12 @@ extensions.executeMobile = async function executeMobile (mobileCommand, opts = {
|
|
|
48
48
|
|
|
49
49
|
deleteFile: 'mobileDeleteFile',
|
|
50
50
|
|
|
51
|
+
isAppInstalled: 'mobileIsAppInstalled',
|
|
52
|
+
queryAppState: 'mobileQueryAppState',
|
|
53
|
+
activateApp: 'mobileActivateApp',
|
|
54
|
+
removeApp: 'mobileRemoveApp',
|
|
55
|
+
terminateApp: 'mobileTerminateApp',
|
|
56
|
+
installApp: 'mobileInstallApp',
|
|
51
57
|
clearApp: 'mobileClearApp',
|
|
52
58
|
|
|
53
59
|
startService: 'mobileStartService',
|
|
@@ -93,11 +93,14 @@ async function scheduleScreenRecord (adb, recordingProperties, log = null) {
|
|
|
93
93
|
: MAX_RECORDING_TIME_SEC;
|
|
94
94
|
log?.debug(`Starting the next ${chunkDuration}s-chunk ` +
|
|
95
95
|
`of screen recording in order to achieve ${timeLimitInt}s total duration`);
|
|
96
|
-
|
|
97
|
-
|
|
96
|
+
(async () => {
|
|
97
|
+
try {
|
|
98
|
+
await scheduleScreenRecord(adb, recordingProperties, log);
|
|
99
|
+
} catch (e) {
|
|
98
100
|
log?.error(e.stack);
|
|
99
101
|
recordingProperties.stopped = true;
|
|
100
|
-
}
|
|
102
|
+
}
|
|
103
|
+
})();
|
|
101
104
|
});
|
|
102
105
|
|
|
103
106
|
await recordingProc.start(0);
|
package/lib/driver.js
CHANGED
|
@@ -263,11 +263,16 @@ class AndroidDriver extends BaseDriver {
|
|
|
263
263
|
this.bootstrap = new helpers.bootstrap(this.adb, this.opts.bootstrapPort, this.opts.websocket);
|
|
264
264
|
await this.bootstrap.start(this.opts.appPackage, this.opts.disableAndroidWatchers, this.opts.acceptSslCerts);
|
|
265
265
|
// handling unexpected shutdown
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
await this.
|
|
266
|
+
(async () => {
|
|
267
|
+
try {
|
|
268
|
+
await this.bootstrap.onUnexpectedShutdown;
|
|
269
|
+
} catch (err) {
|
|
270
|
+
if (!this.bootstrap.ignoreUnexpectedShutdown) {
|
|
271
|
+
await this.startUnexpectedShutdown(err);
|
|
272
|
+
}
|
|
269
273
|
}
|
|
270
|
-
});
|
|
274
|
+
})();
|
|
275
|
+
|
|
271
276
|
|
|
272
277
|
if (!this.opts.skipUnlock) {
|
|
273
278
|
// Let's try to unlock the device
|