appium-adb-test
Version:
Android Debug Bridge interface
497 lines (403 loc) • 34.1 kB
JavaScript
var _regeneratorRuntime = require('babel-runtime/regenerator')['default'];
var _getIterator = require('babel-runtime/core-js/get-iterator')['default'];
var _interopRequireDefault = require('babel-runtime/helpers/interop-require-default')['default'];
Object.defineProperty(exports, '__esModule', {
value: true
});
var _path = require('path');
var _path2 = _interopRequireDefault(_path);
var _appiumSupport = require('appium-support');
var _loggerJs = require('./logger.js');
var _loggerJs2 = _interopRequireDefault(_loggerJs);
var _admZip = require('adm-zip');
var _admZip2 = _interopRequireDefault(_admZip);
var _teen_process = require('teen_process');
var _lodash = require('lodash');
var _lodash2 = _interopRequireDefault(_lodash);
var rootDir = _path2['default'].resolve(__dirname, process.env.NO_PRECOMPILE ? '..' : '../..');
var androidPlatforms = ['android-4.2', 'android-17', 'android-4.3', 'android-18', 'android-4.4', 'android-19', 'android-L', 'android-20', 'android-5.0', 'android-21', 'android-22', 'android-MNC', 'android-23', 'android-6.0'];
function getDirectories(rootPath) {
var files, dirs, _iteratorNormalCompletion, _didIteratorError, _iteratorError, _iterator, _step, file, pathString;
return _regeneratorRuntime.async(function getDirectories$(context$1$0) {
while (1) switch (context$1$0.prev = context$1$0.next) {
case 0:
context$1$0.next = 2;
return _regeneratorRuntime.awrap(_appiumSupport.fs.readdir(rootPath));
case 2:
files = context$1$0.sent;
dirs = [];
_iteratorNormalCompletion = true;
_didIteratorError = false;
_iteratorError = undefined;
context$1$0.prev = 7;
_iterator = _getIterator(files);
case 9:
if (_iteratorNormalCompletion = (_step = _iterator.next()).done) {
context$1$0.next = 19;
break;
}
file = _step.value;
pathString = _path2['default'].resolve(rootPath, file);
context$1$0.next = 14;
return _regeneratorRuntime.awrap(_appiumSupport.fs.lstat(pathString));
case 14:
if (!context$1$0.sent.isDirectory()) {
context$1$0.next = 16;
break;
}
dirs.push(file);
case 16:
_iteratorNormalCompletion = true;
context$1$0.next = 9;
break;
case 19:
context$1$0.next = 25;
break;
case 21:
context$1$0.prev = 21;
context$1$0.t0 = context$1$0['catch'](7);
_didIteratorError = true;
_iteratorError = context$1$0.t0;
case 25:
context$1$0.prev = 25;
context$1$0.prev = 26;
if (!_iteratorNormalCompletion && _iterator['return']) {
_iterator['return']();
}
case 28:
context$1$0.prev = 28;
if (!_didIteratorError) {
context$1$0.next = 31;
break;
}
throw _iteratorError;
case 31:
return context$1$0.finish(28);
case 32:
return context$1$0.finish(25);
case 33:
return context$1$0.abrupt('return', dirs.sort());
case 34:
case 'end':
return context$1$0.stop();
}
}, null, this, [[7, 21, 25, 33], [26,, 28, 32]]);
}
function getAndroidPlatformAndPath() {
var androidHome, platforms, _iteratorNormalCompletion2, _didIteratorError2, _iteratorError2, _iterator2, _step2, platform, platformPath;
return _regeneratorRuntime.async(function getAndroidPlatformAndPath$(context$1$0) {
while (1) switch (context$1$0.prev = context$1$0.next) {
case 0:
androidHome = process.env.ANDROID_HOME;
if (_lodash2['default'].isString(androidHome)) {
context$1$0.next = 4;
break;
}
_loggerJs2['default'].error("ANDROID_HOME was not exported!");
return context$1$0.abrupt('return', null);
case 4:
platforms = _path2['default'].resolve(androidHome, 'platforms');
_iteratorNormalCompletion2 = true;
_didIteratorError2 = false;
_iteratorError2 = undefined;
context$1$0.prev = 8;
_iterator2 = _getIterator(_lodash2['default'].clone(androidPlatforms).reverse());
case 10:
if (_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done) {
context$1$0.next = 20;
break;
}
platform = _step2.value;
platformPath = _path2['default'].resolve(platforms, platform);
context$1$0.next = 15;
return _regeneratorRuntime.awrap(_appiumSupport.fs.exists(platformPath));
case 15:
if (!context$1$0.sent) {
context$1$0.next = 17;
break;
}
return context$1$0.abrupt('return', { platform: platform, platformPath: platformPath });
case 17:
_iteratorNormalCompletion2 = true;
context$1$0.next = 10;
break;
case 20:
context$1$0.next = 26;
break;
case 22:
context$1$0.prev = 22;
context$1$0.t0 = context$1$0['catch'](8);
_didIteratorError2 = true;
_iteratorError2 = context$1$0.t0;
case 26:
context$1$0.prev = 26;
context$1$0.prev = 27;
if (!_iteratorNormalCompletion2 && _iterator2['return']) {
_iterator2['return']();
}
case 29:
context$1$0.prev = 29;
if (!_didIteratorError2) {
context$1$0.next = 32;
break;
}
throw _iteratorError2;
case 32:
return context$1$0.finish(29);
case 33:
return context$1$0.finish(26);
case 34:
return context$1$0.abrupt('return', null);
case 35:
case 'end':
return context$1$0.stop();
}
}, null, this, [[8, 22, 26, 34], [27,, 29, 33]]);
}
function unzipFile(zipPath) {
var zip;
return _regeneratorRuntime.async(function unzipFile$(context$1$0) {
while (1) switch (context$1$0.prev = context$1$0.next) {
case 0:
_loggerJs2['default'].debug('Unzipping ' + zipPath);
context$1$0.prev = 1;
context$1$0.next = 4;
return _regeneratorRuntime.awrap(assertZipArchive(zipPath));
case 4:
if (!_appiumSupport.system.isWindows()) {
context$1$0.next = 10;
break;
}
zip = new _admZip2['default'](zipPath);
zip.extractAllTo(_path2['default'].dirname(zipPath), true);
_loggerJs2['default'].debug("Unzip successful");
context$1$0.next = 13;
break;
case 10:
context$1$0.next = 12;
return _regeneratorRuntime.awrap((0, _teen_process.exec)('unzip', ['-o', zipPath], { cwd: _path2['default'].dirname(zipPath) }));
case 12:
_loggerJs2['default'].debug("Unzip successful");
case 13:
context$1$0.next = 18;
break;
case 15:
context$1$0.prev = 15;
context$1$0.t0 = context$1$0['catch'](1);
throw new Error('Error occurred while unzipping. Original error: ' + context$1$0.t0.message);
case 18:
case 'end':
return context$1$0.stop();
}
}, null, this, [[1, 15]]);
}
function assertZipArchive(zipPath) {
var execOpts;
return _regeneratorRuntime.async(function assertZipArchive$(context$1$0) {
while (1) switch (context$1$0.prev = context$1$0.next) {
case 0:
_loggerJs2['default'].debug('Testing zip archive: ' + zipPath);
if (!_appiumSupport.system.isWindows()) {
context$1$0.next = 11;
break;
}
context$1$0.next = 4;
return _regeneratorRuntime.awrap(_appiumSupport.fs.exists(zipPath));
case 4:
if (!context$1$0.sent) {
context$1$0.next = 8;
break;
}
_loggerJs2['default'].debug("Zip archive tested clean");
context$1$0.next = 9;
break;
case 8:
throw new Error('Zip archive not present at ' + zipPath);
case 9:
context$1$0.next = 14;
break;
case 11:
execOpts = { cwd: _path2['default'].dirname(zipPath) };
context$1$0.next = 14;
return _regeneratorRuntime.awrap((0, _teen_process.exec)('unzip', ['-tq', zipPath], execOpts));
case 14:
case 'end':
return context$1$0.stop();
}
}, null, this);
}
function getIMEListFromOutput(stdout) {
var engines = [];
var _iteratorNormalCompletion3 = true;
var _didIteratorError3 = false;
var _iteratorError3 = undefined;
try {
for (var _iterator3 = _getIterator(stdout.split('\n')), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) {
var line = _step3.value;
if (line.length > 0 && line[0] !== ' ') {
// remove newline and trailing colon, and add to the list
engines.push(line.trim().replace(/:$/, ''));
}
}
} catch (err) {
_didIteratorError3 = true;
_iteratorError3 = err;
} finally {
try {
if (!_iteratorNormalCompletion3 && _iterator3['return']) {
_iterator3['return']();
}
} finally {
if (_didIteratorError3) {
throw _iteratorError3;
}
}
}
return engines;
}
function getJavaForOs() {
var sep = _path2['default'].sep;
var java = '' + getJavaHome() + sep + 'bin' + sep + 'java';
if (_appiumSupport.system.isWindows()) {
java = java + '.exe';
}
return java;
}
function getJavaHome() {
if (process.env.JAVA_HOME) {
return process.env.JAVA_HOME;
}
throw new Error("JAVA_HOME is not set currently. Please set JAVA_HOME.");
}
/*
* Checks mShowingLockscreen in dumpsys output to determine if lock screen is showing
*/
function isShowingLockscreen(dumpsys) {
var m = /mShowingLockscreen=\w+/gi.exec(dumpsys);
var ret = m && m.length && m[0].split('=')[1] === 'true' || false;
return ret;
}
/*
* Checks mCurrentFocus in dumpsys output to determine if Keyguard is activated
*/
function isCurrentFocusOnKeyguard(dumpsys) {
var m = /mCurrentFocus.+Keyguard/gi.exec(dumpsys);
return m && m.length && m[0] ? true : false;
}
/*
* Reads SurfaceOrientation in dumpsys output
*/
function getSurfaceOrientation(dumpsys) {
var m = /SurfaceOrientation: \d/gi.exec(dumpsys);
return m && parseInt(m[0].split(':')[1]);
}
/*
* Checks mScreenOnFully in dumpsys output to determine if screen is showing
* Default is true
*/
function isScreenOnFully(dumpsys) {
var m = /mScreenOnFully=\w+/gi.exec(dumpsys);
return !m || // if information is missing we assume screen is fully on
m && m.length > 0 && m[0].split('=')[1] === 'true' || false;
}
function buildStartCmd(startAppOptions, apiLevel) {
var cmd = ['am', 'start', '-W', '-n', startAppOptions.pkg + '/' + startAppOptions.activity];
if (startAppOptions.stopApp && apiLevel >= 15) {
cmd.push('-S');
}
if (startAppOptions.action) {
cmd.push('-a', startAppOptions.action);
}
if (startAppOptions.category) {
cmd.push('-c', startAppOptions.category);
}
if (startAppOptions.flags) {
cmd.push('-f', startAppOptions.flags);
}
if (startAppOptions.optionalIntentArguments) {
// expect optionalIntentArguments to be a single string of the form:
// "-flag key"
// "-flag key value"
// or a combination of these (e.g., "-flag1 key1 -flag2 key2 value2")
// take a string and parse out the part before any spaces, and anything after
// the first space
var parseKeyValue = function parseKeyValue(str) {
str = str.trim();
var space = str.indexOf(' ');
if (space === -1) {
return str.length ? [str] : [];
} else {
return [str.substring(0, space).trim(), str.substring(space + 1).trim()];
}
};
// cycle through the optionalIntentArguments and pull out the arguments
// add a space initially so flags can be distinguished from arguments that
// have internal hyphens
var optionalIntentArguments = ' ' + startAppOptions.optionalIntentArguments;
var re = / (-[^\s]+) (.+)/;
while (true) {
// eslint-disable-line no-constant-condition
var args = re.exec(optionalIntentArguments);
if (!args) {
if (optionalIntentArguments.length) {
// no more flags, so the remainder can be treated as 'key' or 'key value'
cmd.push.apply(cmd, parseKeyValue(optionalIntentArguments));
}
// we are done
break;
}
// take the flag and see if it is at the beginning of the string
// if it is not, then it means we have been through already, and
// what is before the flag is the argument for the previous flag
var flag = args[1];
var flagPos = optionalIntentArguments.indexOf(flag);
if (flagPos !== 0) {
var prevArgs = optionalIntentArguments.substring(0, flagPos);
cmd.push.apply(cmd, parseKeyValue(prevArgs));
}
// add the flag, as there are no more earlier arguments
cmd.push(flag);
// make optionalIntentArguments hold the remainder
optionalIntentArguments = args[2];
}
}
return cmd;
}
// turns pkg.activity.name to .activity.name
// also turns activity.name to .activity.name
function getPossibleActivityNames(pkgName, activityName) {
var names = [activityName];
// need to beware of namespaces with overlapping chars:
// com.foo.bar
// com.foo.barx
if (activityName.indexOf(pkgName + '.') === 0) {
names.push(activityName.substring(pkgName.length));
}
if (activityName[0] !== '.') {
names.push('.' + activityName);
}
return names;
}
exports.getDirectories = getDirectories;
exports.getAndroidPlatformAndPath = getAndroidPlatformAndPath;
exports.unzipFile = unzipFile;
exports.assertZipArchive = assertZipArchive;
exports.getIMEListFromOutput = getIMEListFromOutput;
exports.getJavaForOs = getJavaForOs;
exports.isShowingLockscreen = isShowingLockscreen;
exports.isCurrentFocusOnKeyguard = isCurrentFocusOnKeyguard;
exports.getSurfaceOrientation = getSurfaceOrientation;
exports.isScreenOnFully = isScreenOnFully;
exports.buildStartCmd = buildStartCmd;
exports.getPossibleActivityNames = getPossibleActivityNames;
exports.getJavaHome = getJavaHome;
exports.rootDir = rootDir;
exports.androidPlatforms = androidPlatforms;
// It is not a clean way to sort it, but in this case would work fine because
// we have numerics and alphanumeric
// will return some thing like this
// ["17.0.0", "18.0.1", "19.0.0", "19.0.1", "19.1.0", "20.0.0",
// "android-4.2.2", "android-4.3", "android-4.4"]
// get the latest platform and path
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/helpers.js"],"names":[],"mappings":"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","file":"lib/helpers.js","sourcesContent":["import path from 'path';\r\nimport { system, fs } from 'appium-support';\r\nimport log from './logger.js';\r\nimport AdmZip from 'adm-zip';\r\nimport { exec } from 'teen_process';\r\nimport _ from 'lodash';\r\n\r\n\r\nconst rootDir = path.resolve(__dirname, process.env.NO_PRECOMPILE ? '..' : '../..');\r\nconst androidPlatforms = ['android-4.2', 'android-17', 'android-4.3', 'android-18',\r\n                          'android-4.4', 'android-19', 'android-L', 'android-20',\r\n                          'android-5.0', 'android-21', 'android-22', 'android-MNC',\r\n                          'android-23', 'android-6.0'];\r\n\r\nasync function getDirectories (rootPath) {\r\n  let files = await fs.readdir(rootPath);\r\n  let dirs = [];\r\n  for (let file of files) {\r\n    let pathString = path.resolve(rootPath, file);\r\n    if ((await fs.lstat(pathString)).isDirectory()) {\r\n      dirs.push(file);\r\n    }\r\n  }\r\n  // It is not a clean way to sort it, but in this case would work fine because\r\n  // we have numerics and alphanumeric\r\n  // will return some thing like this\r\n  // [\"17.0.0\", \"18.0.1\", \"19.0.0\", \"19.0.1\", \"19.1.0\", \"20.0.0\",\r\n  //  \"android-4.2.2\", \"android-4.3\", \"android-4.4\"]\r\n  return dirs.sort();\r\n}\r\n\r\nasync function getAndroidPlatformAndPath () {\r\n  const androidHome = process.env.ANDROID_HOME;\r\n  if (!_.isString(androidHome)) {\r\n    log.error(\"ANDROID_HOME was not exported!\");\r\n    return null;\r\n  }\r\n\r\n  // get the latest platform and path\r\n  let platforms = path.resolve(androidHome, 'platforms');\r\n  for (let platform of _.clone(androidPlatforms).reverse()) {\r\n    let platformPath = path.resolve(platforms, platform);\r\n    if (await fs.exists(platformPath)) {\r\n      return {platform, platformPath};\r\n    }\r\n  }\r\n  return null;\r\n}\r\n\r\nasync function unzipFile (zipPath) {\r\n  log.debug(`Unzipping ${zipPath}`);\r\n  try {\r\n    await assertZipArchive(zipPath);\r\n    if (system.isWindows()) {\r\n      let zip = new AdmZip(zipPath);\r\n      zip.extractAllTo(path.dirname(zipPath), true);\r\n      log.debug(\"Unzip successful\");\r\n    } else {\r\n      await exec('unzip', ['-o', zipPath], {cwd: path.dirname(zipPath)});\r\n      log.debug(\"Unzip successful\");\r\n    }\r\n  } catch (e) {\r\n    throw new Error(`Error occurred while unzipping. Original error: ${e.message}`);\r\n  }\r\n}\r\n\r\nasync function assertZipArchive (zipPath) {\r\n  log.debug(`Testing zip archive: ${zipPath}`);\r\n  if (system.isWindows()) {\r\n    if (await fs.exists(zipPath)) {\r\n      log.debug(\"Zip archive tested clean\");\r\n    } else {\r\n      throw new Error(`Zip archive not present at ${zipPath}`);\r\n    }\r\n  } else {\r\n    let execOpts = {cwd: path.dirname(zipPath)};\r\n    await exec('unzip', ['-tq', zipPath], execOpts);\r\n  }\r\n}\r\n\r\nfunction getIMEListFromOutput (stdout) {\r\n  let engines = [];\r\n  for (let line of stdout.split('\\n')) {\r\n    if (line.length > 0 && line[0] !== ' ') {\r\n      // remove newline and trailing colon, and add to the list\r\n      engines.push(line.trim().replace(/:$/, ''));\r\n    }\r\n  }\r\n  return engines;\r\n}\r\n\r\nfunction getJavaForOs () {\r\n  const sep = path.sep;\r\n  let java = `${getJavaHome()}${sep}bin${sep}java`;\r\n  if (system.isWindows()) {\r\n    java = java + '.exe';\r\n  }\r\n  return java;\r\n}\r\n\r\nfunction getJavaHome () {\r\n  if (process.env.JAVA_HOME) {\r\n    return process.env.JAVA_HOME;\r\n  }\r\n  throw new Error(\"JAVA_HOME is not set currently. Please set JAVA_HOME.\");\r\n}\r\n\r\n/*\r\n * Checks mShowingLockscreen in dumpsys output to determine if lock screen is showing\r\n */\r\nfunction isShowingLockscreen (dumpsys) {\r\n  let m = /mShowingLockscreen=\\w+/gi.exec(dumpsys);\r\n  let ret = (m && m.length && m[0].split('=')[1] === 'true') || false;\r\n  return ret;\r\n}\r\n\r\n/*\r\n * Checks mCurrentFocus in dumpsys output to determine if Keyguard is activated\r\n */\r\nfunction isCurrentFocusOnKeyguard (dumpsys) {\r\n  let m = /mCurrentFocus.+Keyguard/gi.exec(dumpsys);\r\n  return (m && m.length && m[0]) ? true : false;\r\n}\r\n\r\n/*\r\n * Reads SurfaceOrientation in dumpsys output\r\n */\r\nfunction getSurfaceOrientation (dumpsys) {\r\n  let m = /SurfaceOrientation: \\d/gi.exec(dumpsys);\r\n  return m && parseInt(m[0].split(':')[1]);\r\n}\r\n\r\n/*\r\n * Checks mScreenOnFully in dumpsys output to determine if screen is showing\r\n * Default is true\r\n */\r\nfunction isScreenOnFully (dumpsys) {\r\n  let m = /mScreenOnFully=\\w+/gi.exec(dumpsys);\r\n  return !m || // if information is missing we assume screen is fully on\r\n         (m && m.length > 0 && m[0].split('=')[1] === 'true') || false;\r\n}\r\n\r\nfunction buildStartCmd (startAppOptions, apiLevel) {\r\n  let cmd = ['am', 'start', '-W', '-n', `${startAppOptions.pkg}/${startAppOptions.activity}`];\r\n  if (startAppOptions.stopApp && apiLevel >= 15) {\r\n    cmd.push('-S');\r\n  }\r\n  if (startAppOptions.action) {\r\n    cmd.push('-a', startAppOptions.action);\r\n  }\r\n  if (startAppOptions.category) {\r\n    cmd.push('-c', startAppOptions.category);\r\n  }\r\n  if (startAppOptions.flags) {\r\n    cmd.push('-f', startAppOptions.flags);\r\n  }\r\n  if (startAppOptions.optionalIntentArguments) {\r\n    // expect optionalIntentArguments to be a single string of the form:\r\n    //     \"-flag key\"\r\n    //     \"-flag key value\"\r\n    // or a combination of these (e.g., \"-flag1 key1 -flag2 key2 value2\")\r\n\r\n    // take a string and parse out the part before any spaces, and anything after\r\n    // the first space\r\n    let parseKeyValue = function (str) {\r\n      str = str.trim();\r\n      let space = str.indexOf(' ');\r\n      if (space === -1) {\r\n        return str.length ? [str] : [];\r\n      } else {\r\n        return [str.substring(0, space).trim(), str.substring(space + 1).trim()];\r\n      }\r\n    };\r\n\r\n    // cycle through the optionalIntentArguments and pull out the arguments\r\n    // add a space initially so flags can be distinguished from arguments that\r\n    // have internal hyphens\r\n    let optionalIntentArguments = ` ${startAppOptions.optionalIntentArguments}`;\r\n    let re = / (-[^\\s]+) (.+)/;\r\n    while (true) { // eslint-disable-line no-constant-condition\r\n      let args = re.exec(optionalIntentArguments);\r\n      if (!args) {\r\n        if (optionalIntentArguments.length) {\r\n          // no more flags, so the remainder can be treated as 'key' or 'key value'\r\n          cmd.push.apply(cmd, parseKeyValue(optionalIntentArguments));\r\n        }\r\n        // we are done\r\n        break;\r\n      }\r\n\r\n      // take the flag and see if it is at the beginning of the string\r\n      // if it is not, then it means we have been through already, and\r\n      // what is before the flag is the argument for the previous flag\r\n      let flag = args[1];\r\n      let flagPos = optionalIntentArguments.indexOf(flag);\r\n      if (flagPos !== 0) {\r\n        let prevArgs = optionalIntentArguments.substring(0, flagPos);\r\n        cmd.push.apply(cmd, parseKeyValue(prevArgs));\r\n      }\r\n\r\n      // add the flag, as there are no more earlier arguments\r\n      cmd.push(flag);\r\n\r\n      // make optionalIntentArguments hold the remainder\r\n      optionalIntentArguments = args[2];\r\n    }\r\n  }\r\n  return cmd;\r\n}\r\n\r\n// turns pkg.activity.name to .activity.name\r\n// also turns activity.name to .activity.name\r\nfunction getPossibleActivityNames (pkgName, activityName) {\r\n  let names = [activityName];\r\n  // need to beware of namespaces with overlapping chars:\r\n  //   com.foo.bar\r\n  //   com.foo.barx\r\n  if (activityName.indexOf(`${pkgName}.`) === 0) {\r\n    names.push(activityName.substring(pkgName.length));\r\n  }\r\n  if (activityName[0] !== '.') {\r\n    names.push(`.${activityName}`);\r\n  }\r\n  return names;\r\n}\r\n\r\nexport { getDirectories, getAndroidPlatformAndPath, unzipFile, assertZipArchive,\r\n         getIMEListFromOutput, getJavaForOs, isShowingLockscreen, isCurrentFocusOnKeyguard,\r\n         getSurfaceOrientation, isScreenOnFully, buildStartCmd, getPossibleActivityNames,\r\n         getJavaHome, rootDir, androidPlatforms };\r\n"],"sourceRoot":"..\\.."}
;