UNPKG

appium-adb-test

Version:

Android Debug Bridge interface

497 lines (403 loc) 34.1 kB
'use strict'; 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":"..\\.."}