appium-android-driver 4.51.0 → 4.54.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/context.js +4 -1
- package/build/lib/commands/execute.js +3 -2
- package/build/lib/commands/file-actions.js +16 -9
- package/build/lib/commands/network.js +8 -1
- package/build/lib/commands/performance.js +106 -88
- package/lib/commands/context.js +3 -0
- package/lib/commands/execute.js +2 -0
- package/lib/commands/file-actions.js +27 -7
- package/lib/commands/network.js +21 -0
- package/lib/commands/performance.js +146 -115
- package/package.json +8 -8
|
@@ -192,12 +192,14 @@ helpers.startChromedriverProxy = async function startChromedriverProxy(context,
|
|
|
192
192
|
|
|
193
193
|
this.chromedriver = cd;
|
|
194
194
|
this.proxyReqRes = this.chromedriver.proxyReq.bind(this.chromedriver);
|
|
195
|
+
this.proxyCommand = this.chromedriver.jwproxy.command.bind(this.chromedriver.jwproxy);
|
|
195
196
|
this.jwpProxyActive = true;
|
|
196
197
|
};
|
|
197
198
|
|
|
198
199
|
helpers.suspendChromedriverProxy = function suspendChromedriverProxy() {
|
|
199
200
|
this.chromedriver = null;
|
|
200
201
|
this.proxyReqRes = null;
|
|
202
|
+
this.proxyCommand = null;
|
|
201
203
|
this.jwpProxyActive = false;
|
|
202
204
|
};
|
|
203
205
|
|
|
@@ -286,6 +288,7 @@ helpers.startChromeSession = async function startChromeSession() {
|
|
|
286
288
|
this.curContext = _webviewHelpers.CHROMIUM_WIN;
|
|
287
289
|
this.sessionChromedrivers[_webviewHelpers.CHROMIUM_WIN] = this.chromedriver;
|
|
288
290
|
this.proxyReqRes = this.chromedriver.proxyReq.bind(this.chromedriver);
|
|
291
|
+
this.proxyCommand = this.chromedriver.jwproxy.command.bind(this.chromedriver.jwproxy);
|
|
289
292
|
this.jwpProxyActive = true;
|
|
290
293
|
|
|
291
294
|
if (this.shouldDismissChromeWelcome()) {
|
|
@@ -420,4 +423,4 @@ var _default = extensions;
|
|
|
420
423
|
exports.default = _default;require('source-map-support').install();
|
|
421
424
|
|
|
422
425
|
|
|
423
|
-
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/context.js"],"names":["CHROMEDRIVER_AUTODOWNLOAD_FEATURE","commands","helpers","extensions","getCurrentContext","curContext","defaultContextName","getContexts","opts","Object","assign","isChromeSession","webviewsMapping","webviewHelpers","getWebViewsMapping","adb","assignContexts","setContext","name","util","hasValue","WEBVIEW_WIN","defaultWebviewName","contexts","_","includes","errors","NoSuchContextError","switchContext","mobileGetContexts","androidDeviceSocket","ensureWebviewsHavePages","webviewDevtoolsPort","enableWebviewDetailsCollection","webviews","parseWebviewNames","NATIVE_WIN","log","debug","JSON","stringify","isChromedriverContext","startChromedriverProxy","recreateChromeDriverSessions","stopChromedriverProxies","suspendChromedriverProxy","Error","WEBVIEW_BASE","appPackage","isWebContext","context","cd","sessionChromedrivers","setupExistingChromedriver","cloneDeep","chromeUseRunningApp","extractChromeAndroidPackageFromContextName","androidPackage","match","length","chromeAndroidPackage","has","map","wm","webviewName","knownPackage","KNOWN_CHROME_PACKAGE_NAMES","appState","queryAppState","APP_STATE","RUNNING_IN_BACKGROUND","RUNNING_IN_FOREGROUND","info","setupNewChromedriver","curDeviceId","on","Chromedriver","EVENT_CHANGED","msg","state","STATE_STOPPED","onChromedriverStop","chromedriver","proxyReqRes","proxyReq","bind","jwpProxyActive","warn","err","startUnexpectedShutdown","keys","removeAllListeners","stop","message","viewName","CHROMIUM_WIN","shouldDismissChromeWelcome","chromeOptions","isArray","args","dismissChromeWelcome","activity","getCurrentActivity","el","findElOrEls","click","ELEMENT","e","startChromeSession","knownPackages","chromeBundleId","chromeAndroidActivity","appActivity","hasWorkingWebview","restart","getChromedriverPort","portSpec","getPort","B","promisify","PortFinder","port","foundPort","potentialPort","stopPort","parseInt","isChromedriverAutodownloadEnabled","isFeatureEnabled","chromeDriverPort","chromedriverPort","chromedriverPorts","details","getWebviewDetails","undefined","isEmpty","executable","chromedriverExecutable","cmdArgs","chromedriverArgs","verbose","showChromedriverLog","executableDir","chromedriverExecutableDir","mappingPath","chromedriverChromeMappingFile","bundleId","useSystemExecutable","chromedriverUseSystemExecutable","disableBuildCheck","chromedriverDisableBuildCheck","isAutodownloadEnabled","opt","endsWith","merge","caps","createChromedriverCaps","start"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport log from '../logger';\nimport Chromedriver from 'appium-chromedriver';\nimport PortFinder from 'portfinder';\nimport B from 'bluebird';\nimport { util } from 'appium-support';\nimport { errors } from 'appium-base-driver';\nimport {\n  default as webviewHelpers,\n  NATIVE_WIN, WEBVIEW_BASE, WEBVIEW_WIN, CHROMIUM_WIN, KNOWN_CHROME_PACKAGE_NAMES\n} from '../webview-helpers';\nimport { APP_STATE } from '../android-helpers';\n\nconst CHROMEDRIVER_AUTODOWNLOAD_FEATURE = 'chromedriver_autodownload';\n\nlet commands = {}, helpers = {}, extensions = {};\n\n\n/* -------------------------------\n * Actual MJSONWP command handlers\n * ------------------------------- */\ncommands.getCurrentContext = async function getCurrentContext () { // eslint-disable-line require-await\n  // if the current context is `null`, indicating no context\n  // explicitly set, it is the default context\n  return this.curContext || this.defaultContextName();\n};\n\ncommands.getContexts = async function getContexts () {\n  const opts = Object.assign({isChromeSession: this.isChromeSession}, this.opts);\n  const webviewsMapping = await webviewHelpers.getWebViewsMapping(this.adb, opts);\n  return this.assignContexts(webviewsMapping);\n};\n\ncommands.setContext = async function setContext (name) {\n  if (!util.hasValue(name)) {\n    name = this.defaultContextName();\n  } else if (name === WEBVIEW_WIN) {\n    // handle setContext \"WEBVIEW\"\n    name = this.defaultWebviewName();\n  }\n  // if we're already in the context we want, do nothing\n  if (name === this.curContext) {\n    return;\n  }\n\n  const opts = Object.assign({isChromeSession: this.isChromeSession}, this.opts);\n  const webviewsMapping = await webviewHelpers.getWebViewsMapping(this.adb, opts);\n  const contexts = this.assignContexts(webviewsMapping);\n  // if the context we want doesn't exist, fail\n  if (!_.includes(contexts, name)) {\n    throw new errors.NoSuchContextError();\n  }\n\n  await this.switchContext(name, webviewsMapping);\n  this.curContext = name;\n};\n\n/**\n * @typedef {Object} WebviewsMapping\n * @property {string} proc The name of the Devtools Unix socket\n * @property {string} webview The web view alias. Looks like `WEBVIEW_`\n * prefix plus PID or package name\n * @property {?Object} info Webview information as it is retrieved by\n * /json/version CDP endpoint\n * @property {?Array<Object>} pages Webview pages list as it is retrieved by\n * /json/list CDP endpoint\n * @propery {?string} webviewName An actual webview name for switching context.\n * This value becomes null when failing to find a PID for a webview.\n *\n * The following json demonstrates the example of WebviewsMapping object.\n * Note that `description` in `page` can be an empty string most likely when it comes to Mobile Chrome)\n * {\n *   \"proc\": \"@webview_devtools_remote_22138\",\n *   \"webview\": \"WEBVIEW_22138\",\n *   \"info\": {\n *     \"Android-Package\": \"io.appium.settings\",\n *     \"Browser\": \"Chrome/74.0.3729.185\",\n *     \"Protocol-Version\": \"1.3\",\n *     \"User-Agent\": \"Mozilla/5.0 (Linux; Android 10; Android SDK built for x86 Build/QSR1.190920.001; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/74.0.3729.185 Mobile Safari/537.36\",\n *     \"V8-Version\": \"7.4.288.28\",\n *     \"WebKit-Version\": \"537.36 (@22955682f94ce09336197bfb8dffea991fa32f0d)\",\n *     \"webSocketDebuggerUrl\": \"ws://127.0.0.1:10900/devtools/browser\"\n *   },\n *   \"pages\": [\n *     {\n *       \"description\": \"{\\\"attached\\\":true,\\\"empty\\\":false,\\\"height\\\":1458,\\\"screenX\\\":0,\\\"screenY\\\":336,\\\"visible\\\":true,\\\"width\\\":1080}\",\n *       \"devtoolsFrontendUrl\": \"http://chrome-devtools-frontend.appspot.com/serve_rev/@22955682f94ce09336197bfb8dffea991fa32f0d/inspector.html?ws=127.0.0.1:10900/devtools/page/27325CC50B600D31B233F45E09487B1F\",\n *       \"id\": \"27325CC50B600D31B233F45E09487B1F\",\n *       \"title\": \"Releases · appium/appium · GitHub\",\n *       \"type\": \"page\",\n *       \"url\": \"https://github.com/appium/appium/releases\",\n *       \"webSocketDebuggerUrl\": \"ws://127.0.0.1:10900/devtools/page/27325CC50B600D31B233F45E09487B1F\"\n *     }\n *   ],\n *   \"webviewName\": \"WEBVIEW_com.io.appium.setting\"\n * }\n */\n\n/**\n * Returns a webviewsMapping based on CDP endpoints\n *\n * @return {Array<WebviewsMapping>} webviewsMapping\n */\ncommands.mobileGetContexts = async function mobileGetContexts () {\n  const opts = {\n    androidDeviceSocket: this.opts.androidDeviceSocket,\n    ensureWebviewsHavePages: true,\n    webviewDevtoolsPort: this.opts.webviewDevtoolsPort,\n    enableWebviewDetailsCollection: true,\n    isChromeSession: this.isChromeSession\n  };\n  return await webviewHelpers.getWebViewsMapping(this.adb, opts);\n};\n\nhelpers.assignContexts = function assignContexts (webviewsMapping) {\n  const opts = Object.assign({isChromeSession: this.isChromeSession}, this.opts);\n  const webviews = webviewHelpers.parseWebviewNames(webviewsMapping, opts);\n  this.contexts = [NATIVE_WIN, ...webviews];\n  log.debug(`Available contexts: ${JSON.stringify(this.contexts)}`);\n  return this.contexts;\n};\n\nhelpers.switchContext = async function switchContext (name, webviewsMapping) {\n  // We have some options when it comes to webviews. If we want a\n  // Chromedriver webview, we can only control one at a time.\n  if (this.isChromedriverContext(name)) {\n    // start proxying commands directly to chromedriver\n    await this.startChromedriverProxy(name, webviewsMapping);\n  } else if (this.isChromedriverContext(this.curContext)) {\n    // if we're moving to a non-chromedriver webview, and our current context\n    // _is_ a chromedriver webview, if caps recreateChromeDriverSessions is set\n    // to true then kill chromedriver session using stopChromedriverProxies or\n    // else simply suspend proxying to the latter\n    if (this.opts.recreateChromeDriverSessions) {\n      log.debug('recreateChromeDriverSessions set to true; killing existing chromedrivers');\n      await this.stopChromedriverProxies();\n    } else {\n      await this.suspendChromedriverProxy();\n    }\n  } else {\n    throw new Error(`Didn't know how to handle switching to context '${name}'`);\n  }\n};\n\n\n/* ---------------------------------\n * On-object context-related helpers\n * --------------------------------- */\n\n// The reason this is a function and not just a constant is that both android-\n// driver and selendroid-driver use this logic, and each one returns\n// a different default context name\nhelpers.defaultContextName = function defaultContextName () {\n  return NATIVE_WIN;\n};\n\nhelpers.defaultWebviewName = function defaultWebviewName () {\n  return WEBVIEW_BASE + this.opts.appPackage;\n};\n\nhelpers.isWebContext = function isWebContext () {\n  return this.curContext !== null && this.curContext !== NATIVE_WIN;\n};\n\n// Turn on proxying to an existing Chromedriver session or a new one\nhelpers.startChromedriverProxy = async function startChromedriverProxy (context, webviewsMapping) {\n  log.debug(`Connecting to chrome-backed webview context '${context}'`);\n\n  let cd;\n  if (this.sessionChromedrivers[context]) {\n    // in the case where we've already set up a chromedriver for a context,\n    // we want to reconnect to it, not create a whole new one\n    log.debug(`Found existing Chromedriver for context '${context}'. Using it.`);\n    cd = this.sessionChromedrivers[context];\n    await setupExistingChromedriver(cd);\n  } else {\n    let opts = _.cloneDeep(this.opts);\n    opts.chromeUseRunningApp = true;\n\n    // if requested, tell chromedriver to attach to the android package we have\n    // associated with the context name, rather than the package of the AUT.\n    // And turn this on by default for chrome--if chrome pops up with a webview\n    // and someone wants to switch to it, we should let chromedriver connect to\n    // chrome rather than staying stuck on the AUT\n    if (opts.extractChromeAndroidPackageFromContextName || context === `${WEBVIEW_BASE}chrome`) {\n      let androidPackage = context.match(`${WEBVIEW_BASE}(.+)`);\n      if (androidPackage && androidPackage.length > 0) {\n        opts.chromeAndroidPackage = androidPackage[1];\n      }\n      if (!opts.extractChromeAndroidPackageFromContextName) {\n        if (_.has(this.opts, 'enableWebviewDetailsCollection') && !this.opts.enableWebviewDetailsCollection) {\n          // When enableWebviewDetailsCollection capability is explicitly disabled, try to identify\n          // chromeAndroidPackage based on contexts, known chrome variant packages and queryAppState result\n          // since webviewsMapping does not have info object\n          const contexts = webviewsMapping.map((wm) => wm.webviewName);\n          for (const knownPackage of KNOWN_CHROME_PACKAGE_NAMES) {\n            if (_.includes(contexts, `${WEBVIEW_BASE}${knownPackage}`)) {\n              continue;\n            }\n            const appState = await this.queryAppState(knownPackage);\n            if (_.includes([APP_STATE.RUNNING_IN_BACKGROUND, APP_STATE.RUNNING_IN_FOREGROUND], appState)) {\n              opts.chromeAndroidPackage = knownPackage;\n              log.debug(`Identified chromeAndroidPackage as '${opts.chromeAndroidPackage}' ` +\n                `for context '${context}' by querying states of Chrome app packages`);\n              break;\n            }\n          }\n        } else {\n          for (const wm of webviewsMapping) {\n            if (wm.webviewName === context && _.has(wm?.info, 'Android-Package')) {\n              opts.chromeAndroidPackage = wm.info['Android-Package'];\n              log.debug(`Identified chromeAndroidPackage as '${opts.chromeAndroidPackage}' ` +\n                `for context '${context}' by CDP`);\n              break;\n            }\n          }\n        }\n      }\n    }\n\n    cd = await this.setupNewChromedriver(opts, this.adb.curDeviceId, this.adb, context);\n    // bind our stop/exit handler, passing in context so we know which\n    // one stopped unexpectedly\n    cd.on(Chromedriver.EVENT_CHANGED, (msg) => {\n      if (msg.state === Chromedriver.STATE_STOPPED) {\n        this.onChromedriverStop(context);\n      }\n    });\n    // save the chromedriver object under the context\n    this.sessionChromedrivers[context] = cd;\n  }\n  // hook up the local variables so we can proxy this biz\n  this.chromedriver = cd;\n  this.proxyReqRes = this.chromedriver.proxyReq.bind(this.chromedriver);\n  this.jwpProxyActive = true;\n};\n\n// Stop proxying to any Chromedriver\nhelpers.suspendChromedriverProxy = function suspendChromedriverProxy () {\n  this.chromedriver = null;\n  this.proxyReqRes = null;\n  this.jwpProxyActive = false;\n};\n\n// Handle an out-of-band Chromedriver stop event\nhelpers.onChromedriverStop = async function onChromedriverStop (context) {\n  log.warn(`Chromedriver for context ${context} stopped unexpectedly`);\n  if (context === this.curContext) {\n    // we exited unexpectedly while automating the current context and so want\n    // to shut down the session and respond with an error\n    let err = new Error('Chromedriver quit unexpectedly during session');\n    await this.startUnexpectedShutdown(err);\n  } else {\n    // if a Chromedriver in the non-active context barfs, we don't really\n    // care, we'll just make a new one next time we need the context.\n    log.warn(\"Chromedriver quit unexpectedly, but it wasn't the active \" +\n                'context, ignoring');\n    delete this.sessionChromedrivers[context];\n  }\n};\n\n// Intentionally stop all the chromedrivers currently active, and ignore\n// their exit events\nhelpers.stopChromedriverProxies = async function stopChromedriverProxies () {\n  this.suspendChromedriverProxy(); // make sure we turn off the proxy flag\n  for (let context of _.keys(this.sessionChromedrivers)) {\n    let cd = this.sessionChromedrivers[context];\n    log.debug(`Stopping chromedriver for context ${context}`);\n    // stop listening for the stopped state event\n    cd.removeAllListeners(Chromedriver.EVENT_CHANGED);\n    try {\n      await cd.stop();\n    } catch (err) {\n      log.warn(`Error stopping Chromedriver: ${err.message}`);\n    }\n    delete this.sessionChromedrivers[context];\n  }\n};\n\nhelpers.isChromedriverContext = function isChromedriverContext (viewName) {\n  return _.includes(viewName, WEBVIEW_WIN) || viewName === CHROMIUM_WIN;\n};\n\nhelpers.shouldDismissChromeWelcome = function shouldDismissChromeWelcome () {\n  return !!this.opts.chromeOptions &&\n         _.isArray(this.opts.chromeOptions.args) &&\n         this.opts.chromeOptions.args.includes('--no-first-run');\n};\n\nhelpers.dismissChromeWelcome = async function dismissChromeWelcome () {\n  log.info('Trying to dismiss Chrome welcome');\n  let activity = await this.getCurrentActivity();\n  if (activity !== 'org.chromium.chrome.browser.firstrun.FirstRunActivity') {\n    log.info('Chrome welcome dialog never showed up! Continuing');\n    return;\n  }\n  let el = await this.findElOrEls('id', 'com.android.chrome:id/terms_accept', false);\n  await this.click(el.ELEMENT);\n  try {\n    let el = await this.findElOrEls('id', 'com.android.chrome:id/negative_button', false);\n    await this.click(el.ELEMENT);\n  } catch (e) {\n    // DO NOTHING, THIS DEVICE DIDNT LAUNCH THE SIGNIN DIALOG\n    // IT MUST BE A NON GMS DEVICE\n    log.warn(`This device did not show Chrome SignIn dialog, ${e.message}`);\n  }\n};\n\nhelpers.startChromeSession = async function startChromeSession () {\n  log.info('Starting a chrome-based browser session');\n  let opts = _.cloneDeep(this.opts);\n\n  const knownPackages = [\n    'org.chromium.chrome.shell',\n    'com.android.chrome',\n    'com.chrome.beta',\n    'org.chromium.chrome',\n    'org.chromium.webview_shell',\n  ];\n\n  if (_.includes(knownPackages, this.opts.appPackage)) {\n    opts.chromeBundleId = this.opts.appPackage;\n  } else {\n    opts.chromeAndroidActivity = this.opts.appActivity;\n  }\n  this.chromedriver = await this.setupNewChromedriver(opts, this.adb.curDeviceId, this.adb);\n  this.chromedriver.on(Chromedriver.EVENT_CHANGED, (msg) => {\n    if (msg.state === Chromedriver.STATE_STOPPED) {\n      this.onChromedriverStop(CHROMIUM_WIN);\n    }\n  });\n\n  // Now that we have a Chrome session, we ensure that the context is\n  // appropriately set and that this chromedriver is added to the list\n  // of session chromedrivers so we can switch back and forth\n  this.curContext = CHROMIUM_WIN;\n  this.sessionChromedrivers[CHROMIUM_WIN] = this.chromedriver;\n  this.proxyReqRes = this.chromedriver.proxyReq.bind(this.chromedriver);\n  this.jwpProxyActive = true;\n\n  if (this.shouldDismissChromeWelcome()) {\n    // dismiss Chrome welcome dialog\n    await this.dismissChromeWelcome();\n  }\n};\n\n\n/* --------------------------\n * Internal library functions\n * -------------------------- */\n\nasync function setupExistingChromedriver (chromedriver) {\n  // check the status by sending a simple window-based command to ChromeDriver\n  // if there is an error, we want to recreate the ChromeDriver session\n  if (!await chromedriver.hasWorkingWebview()) {\n    log.debug('ChromeDriver is not associated with a window. ' +\n                 'Re-initializing the session.');\n    await chromedriver.restart();\n  }\n  return chromedriver;\n}\n\n/**\n * Find a free port to have Chromedriver listen on.\n *\n * @param {array} [portSpec] - Array which is a list of ports. A list item may\n * also itself be an array of length 2 specifying a start and end port of\n * a range. Some valid port specs:\n *    - [8000, 8001, 8002]\n *    - [[8000, 8005]]\n *    - [8000, [9000, 9100]]\n *\n * @return {number} A free port\n */\nasync function getChromedriverPort (portSpec) {\n  const getPort = B.promisify(PortFinder.getPort, {context: PortFinder});\n\n  // if the user didn't give us any specific information about chromedriver\n  // port ranges, just find any free port\n  if (!portSpec) {\n    const port = await getPort();\n    log.debug(`A port was not given, using random free port: ${port}`);\n    return port;\n  }\n\n  // otherwise find the free port based on a list or range provided by the user\n  log.debug(`Finding a free port for chromedriver using spec ${JSON.stringify(portSpec)}`);\n  let foundPort = null;\n  for (const potentialPort of portSpec) {\n    let port, stopPort;\n    if (_.isArray(potentialPort)) {\n      ([port, stopPort] = potentialPort);\n    } else {\n      port = parseInt(potentialPort, 10); // ensure we have a number and not a string\n      stopPort = port;\n    }\n    try {\n      log.debug(`Checking port range ${port}:${stopPort}`);\n      foundPort = await getPort({port, stopPort});\n      break;\n    } catch (e) {\n      log.debug(`Nothing in port range ${port}:${stopPort} was available`);\n    }\n  }\n\n  if (foundPort === null) {\n    throw new Error(`Could not find a free port for chromedriver using ` +\n                    `chromedriverPorts spec ${JSON.stringify(portSpec)}`);\n  }\n\n  log.debug(`Using free port ${foundPort} for chromedriver`);\n  return foundPort;\n}\n\nhelpers.isChromedriverAutodownloadEnabled = function isChromedriverAutodownloadEnabled () {\n  if (this.isFeatureEnabled(CHROMEDRIVER_AUTODOWNLOAD_FEATURE)) {\n    return true;\n  }\n  log.debug(`Automated Chromedriver download is disabled. ` +\n    `Use '${CHROMEDRIVER_AUTODOWNLOAD_FEATURE}' server feature to enable it`);\n  return false;\n};\n\nhelpers.setupNewChromedriver = async function setupNewChromedriver (opts, curDeviceId, adb, context = null) {\n  if (opts.chromeDriverPort) {\n    log.warn(`The 'chromeDriverPort' capability is deprecated. Please use 'chromedriverPort' instead`);\n    opts.chromedriverPort = opts.chromeDriverPort;\n  }\n\n  if (opts.chromedriverPort) {\n    log.debug(`Using user-specified port ${opts.chromedriverPort} for chromedriver`);\n  } else {\n    // if a single port wasn't given, we'll look for a free one\n    opts.chromedriverPort = await getChromedriverPort(opts.chromedriverPorts);\n  }\n\n  const details = context ? webviewHelpers.getWebviewDetails(adb, context) : undefined;\n  if (!_.isEmpty(details)) {\n    log.debug('Passing web view details to the Chromedriver constructor: ' +\n      JSON.stringify(details, null, 2));\n  }\n\n  const chromedriver = new Chromedriver({\n    port: opts.chromedriverPort,\n    executable: opts.chromedriverExecutable,\n    adb,\n    cmdArgs: opts.chromedriverArgs,\n    verbose: !!opts.showChromedriverLog,\n    executableDir: opts.chromedriverExecutableDir,\n    mappingPath: opts.chromedriverChromeMappingFile,\n    bundleId: opts.chromeBundleId,\n    useSystemExecutable: opts.chromedriverUseSystemExecutable,\n    disableBuildCheck: opts.chromedriverDisableBuildCheck,\n    details,\n    isAutodownloadEnabled: (this || {}).isChromedriverAutodownloadEnabled\n      ? this.isChromedriverAutodownloadEnabled() : undefined,\n  });\n\n  // make sure there are chromeOptions\n  opts.chromeOptions = opts.chromeOptions || {};\n  // try out any prefixed chromeOptions,\n  // and strip the prefix\n  for (const opt of _.keys(opts)) {\n    if (opt.endsWith(':chromeOptions')) {\n      log.warn(`Merging '${opt}' into 'chromeOptions'. This may cause unexpected behavior`);\n      _.merge(opts.chromeOptions, opts[opt]);\n    }\n  }\n\n  const caps = webviewHelpers.createChromedriverCaps(opts, curDeviceId, details);\n  log.debug(`Before starting chromedriver, androidPackage is '${caps.chromeOptions.androidPackage}'`);\n  await chromedriver.start(caps);\n  return chromedriver;\n};\nconst setupNewChromedriver = helpers.setupNewChromedriver;\n\n\nObject.assign(extensions, commands, helpers);\nexport { commands, helpers, setupNewChromedriver };\nexport default extensions;\n"],"file":"lib/commands/context.js","sourceRoot":"../../.."}
|
|
426
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/context.js"],"names":["CHROMEDRIVER_AUTODOWNLOAD_FEATURE","commands","helpers","extensions","getCurrentContext","curContext","defaultContextName","getContexts","opts","Object","assign","isChromeSession","webviewsMapping","webviewHelpers","getWebViewsMapping","adb","assignContexts","setContext","name","util","hasValue","WEBVIEW_WIN","defaultWebviewName","contexts","_","includes","errors","NoSuchContextError","switchContext","mobileGetContexts","androidDeviceSocket","ensureWebviewsHavePages","webviewDevtoolsPort","enableWebviewDetailsCollection","webviews","parseWebviewNames","NATIVE_WIN","log","debug","JSON","stringify","isChromedriverContext","startChromedriverProxy","recreateChromeDriverSessions","stopChromedriverProxies","suspendChromedriverProxy","Error","WEBVIEW_BASE","appPackage","isWebContext","context","cd","sessionChromedrivers","setupExistingChromedriver","cloneDeep","chromeUseRunningApp","extractChromeAndroidPackageFromContextName","androidPackage","match","length","chromeAndroidPackage","has","map","wm","webviewName","knownPackage","KNOWN_CHROME_PACKAGE_NAMES","appState","queryAppState","APP_STATE","RUNNING_IN_BACKGROUND","RUNNING_IN_FOREGROUND","info","setupNewChromedriver","curDeviceId","on","Chromedriver","EVENT_CHANGED","msg","state","STATE_STOPPED","onChromedriverStop","chromedriver","proxyReqRes","proxyReq","bind","proxyCommand","jwproxy","command","jwpProxyActive","warn","err","startUnexpectedShutdown","keys","removeAllListeners","stop","message","viewName","CHROMIUM_WIN","shouldDismissChromeWelcome","chromeOptions","isArray","args","dismissChromeWelcome","activity","getCurrentActivity","el","findElOrEls","click","ELEMENT","e","startChromeSession","knownPackages","chromeBundleId","chromeAndroidActivity","appActivity","hasWorkingWebview","restart","getChromedriverPort","portSpec","getPort","B","promisify","PortFinder","port","foundPort","potentialPort","stopPort","parseInt","isChromedriverAutodownloadEnabled","isFeatureEnabled","chromeDriverPort","chromedriverPort","chromedriverPorts","details","getWebviewDetails","undefined","isEmpty","executable","chromedriverExecutable","cmdArgs","chromedriverArgs","verbose","showChromedriverLog","executableDir","chromedriverExecutableDir","mappingPath","chromedriverChromeMappingFile","bundleId","useSystemExecutable","chromedriverUseSystemExecutable","disableBuildCheck","chromedriverDisableBuildCheck","isAutodownloadEnabled","opt","endsWith","merge","caps","createChromedriverCaps","start"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport log from '../logger';\nimport Chromedriver from 'appium-chromedriver';\nimport PortFinder from 'portfinder';\nimport B from 'bluebird';\nimport { util } from 'appium-support';\nimport { errors } from 'appium-base-driver';\nimport {\n  default as webviewHelpers,\n  NATIVE_WIN, WEBVIEW_BASE, WEBVIEW_WIN, CHROMIUM_WIN, KNOWN_CHROME_PACKAGE_NAMES\n} from '../webview-helpers';\nimport { APP_STATE } from '../android-helpers';\n\nconst CHROMEDRIVER_AUTODOWNLOAD_FEATURE = 'chromedriver_autodownload';\n\nlet commands = {}, helpers = {}, extensions = {};\n\n\n/* -------------------------------\n * Actual MJSONWP command handlers\n * ------------------------------- */\ncommands.getCurrentContext = async function getCurrentContext () { // eslint-disable-line require-await\n  // if the current context is `null`, indicating no context\n  // explicitly set, it is the default context\n  return this.curContext || this.defaultContextName();\n};\n\ncommands.getContexts = async function getContexts () {\n  const opts = Object.assign({isChromeSession: this.isChromeSession}, this.opts);\n  const webviewsMapping = await webviewHelpers.getWebViewsMapping(this.adb, opts);\n  return this.assignContexts(webviewsMapping);\n};\n\ncommands.setContext = async function setContext (name) {\n  if (!util.hasValue(name)) {\n    name = this.defaultContextName();\n  } else if (name === WEBVIEW_WIN) {\n    // handle setContext \"WEBVIEW\"\n    name = this.defaultWebviewName();\n  }\n  // if we're already in the context we want, do nothing\n  if (name === this.curContext) {\n    return;\n  }\n\n  const opts = Object.assign({isChromeSession: this.isChromeSession}, this.opts);\n  const webviewsMapping = await webviewHelpers.getWebViewsMapping(this.adb, opts);\n  const contexts = this.assignContexts(webviewsMapping);\n  // if the context we want doesn't exist, fail\n  if (!_.includes(contexts, name)) {\n    throw new errors.NoSuchContextError();\n  }\n\n  await this.switchContext(name, webviewsMapping);\n  this.curContext = name;\n};\n\n/**\n * @typedef {Object} WebviewsMapping\n * @property {string} proc The name of the Devtools Unix socket\n * @property {string} webview The web view alias. Looks like `WEBVIEW_`\n * prefix plus PID or package name\n * @property {?Object} info Webview information as it is retrieved by\n * /json/version CDP endpoint\n * @property {?Array<Object>} pages Webview pages list as it is retrieved by\n * /json/list CDP endpoint\n * @propery {?string} webviewName An actual webview name for switching context.\n * This value becomes null when failing to find a PID for a webview.\n *\n * The following json demonstrates the example of WebviewsMapping object.\n * Note that `description` in `page` can be an empty string most likely when it comes to Mobile Chrome)\n * {\n *   \"proc\": \"@webview_devtools_remote_22138\",\n *   \"webview\": \"WEBVIEW_22138\",\n *   \"info\": {\n *     \"Android-Package\": \"io.appium.settings\",\n *     \"Browser\": \"Chrome/74.0.3729.185\",\n *     \"Protocol-Version\": \"1.3\",\n *     \"User-Agent\": \"Mozilla/5.0 (Linux; Android 10; Android SDK built for x86 Build/QSR1.190920.001; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/74.0.3729.185 Mobile Safari/537.36\",\n *     \"V8-Version\": \"7.4.288.28\",\n *     \"WebKit-Version\": \"537.36 (@22955682f94ce09336197bfb8dffea991fa32f0d)\",\n *     \"webSocketDebuggerUrl\": \"ws://127.0.0.1:10900/devtools/browser\"\n *   },\n *   \"pages\": [\n *     {\n *       \"description\": \"{\\\"attached\\\":true,\\\"empty\\\":false,\\\"height\\\":1458,\\\"screenX\\\":0,\\\"screenY\\\":336,\\\"visible\\\":true,\\\"width\\\":1080}\",\n *       \"devtoolsFrontendUrl\": \"http://chrome-devtools-frontend.appspot.com/serve_rev/@22955682f94ce09336197bfb8dffea991fa32f0d/inspector.html?ws=127.0.0.1:10900/devtools/page/27325CC50B600D31B233F45E09487B1F\",\n *       \"id\": \"27325CC50B600D31B233F45E09487B1F\",\n *       \"title\": \"Releases · appium/appium · GitHub\",\n *       \"type\": \"page\",\n *       \"url\": \"https://github.com/appium/appium/releases\",\n *       \"webSocketDebuggerUrl\": \"ws://127.0.0.1:10900/devtools/page/27325CC50B600D31B233F45E09487B1F\"\n *     }\n *   ],\n *   \"webviewName\": \"WEBVIEW_com.io.appium.setting\"\n * }\n */\n\n/**\n * Returns a webviewsMapping based on CDP endpoints\n *\n * @return {Array<WebviewsMapping>} webviewsMapping\n */\ncommands.mobileGetContexts = async function mobileGetContexts () {\n  const opts = {\n    androidDeviceSocket: this.opts.androidDeviceSocket,\n    ensureWebviewsHavePages: true,\n    webviewDevtoolsPort: this.opts.webviewDevtoolsPort,\n    enableWebviewDetailsCollection: true,\n    isChromeSession: this.isChromeSession\n  };\n  return await webviewHelpers.getWebViewsMapping(this.adb, opts);\n};\n\nhelpers.assignContexts = function assignContexts (webviewsMapping) {\n  const opts = Object.assign({isChromeSession: this.isChromeSession}, this.opts);\n  const webviews = webviewHelpers.parseWebviewNames(webviewsMapping, opts);\n  this.contexts = [NATIVE_WIN, ...webviews];\n  log.debug(`Available contexts: ${JSON.stringify(this.contexts)}`);\n  return this.contexts;\n};\n\nhelpers.switchContext = async function switchContext (name, webviewsMapping) {\n  // We have some options when it comes to webviews. If we want a\n  // Chromedriver webview, we can only control one at a time.\n  if (this.isChromedriverContext(name)) {\n    // start proxying commands directly to chromedriver\n    await this.startChromedriverProxy(name, webviewsMapping);\n  } else if (this.isChromedriverContext(this.curContext)) {\n    // if we're moving to a non-chromedriver webview, and our current context\n    // _is_ a chromedriver webview, if caps recreateChromeDriverSessions is set\n    // to true then kill chromedriver session using stopChromedriverProxies or\n    // else simply suspend proxying to the latter\n    if (this.opts.recreateChromeDriverSessions) {\n      log.debug('recreateChromeDriverSessions set to true; killing existing chromedrivers');\n      await this.stopChromedriverProxies();\n    } else {\n      await this.suspendChromedriverProxy();\n    }\n  } else {\n    throw new Error(`Didn't know how to handle switching to context '${name}'`);\n  }\n};\n\n\n/* ---------------------------------\n * On-object context-related helpers\n * --------------------------------- */\n\n// The reason this is a function and not just a constant is that both android-\n// driver and selendroid-driver use this logic, and each one returns\n// a different default context name\nhelpers.defaultContextName = function defaultContextName () {\n  return NATIVE_WIN;\n};\n\nhelpers.defaultWebviewName = function defaultWebviewName () {\n  return WEBVIEW_BASE + this.opts.appPackage;\n};\n\nhelpers.isWebContext = function isWebContext () {\n  return this.curContext !== null && this.curContext !== NATIVE_WIN;\n};\n\n// Turn on proxying to an existing Chromedriver session or a new one\nhelpers.startChromedriverProxy = async function startChromedriverProxy (context, webviewsMapping) {\n  log.debug(`Connecting to chrome-backed webview context '${context}'`);\n\n  let cd;\n  if (this.sessionChromedrivers[context]) {\n    // in the case where we've already set up a chromedriver for a context,\n    // we want to reconnect to it, not create a whole new one\n    log.debug(`Found existing Chromedriver for context '${context}'. Using it.`);\n    cd = this.sessionChromedrivers[context];\n    await setupExistingChromedriver(cd);\n  } else {\n    let opts = _.cloneDeep(this.opts);\n    opts.chromeUseRunningApp = true;\n\n    // if requested, tell chromedriver to attach to the android package we have\n    // associated with the context name, rather than the package of the AUT.\n    // And turn this on by default for chrome--if chrome pops up with a webview\n    // and someone wants to switch to it, we should let chromedriver connect to\n    // chrome rather than staying stuck on the AUT\n    if (opts.extractChromeAndroidPackageFromContextName || context === `${WEBVIEW_BASE}chrome`) {\n      let androidPackage = context.match(`${WEBVIEW_BASE}(.+)`);\n      if (androidPackage && androidPackage.length > 0) {\n        opts.chromeAndroidPackage = androidPackage[1];\n      }\n      if (!opts.extractChromeAndroidPackageFromContextName) {\n        if (_.has(this.opts, 'enableWebviewDetailsCollection') && !this.opts.enableWebviewDetailsCollection) {\n          // When enableWebviewDetailsCollection capability is explicitly disabled, try to identify\n          // chromeAndroidPackage based on contexts, known chrome variant packages and queryAppState result\n          // since webviewsMapping does not have info object\n          const contexts = webviewsMapping.map((wm) => wm.webviewName);\n          for (const knownPackage of KNOWN_CHROME_PACKAGE_NAMES) {\n            if (_.includes(contexts, `${WEBVIEW_BASE}${knownPackage}`)) {\n              continue;\n            }\n            const appState = await this.queryAppState(knownPackage);\n            if (_.includes([APP_STATE.RUNNING_IN_BACKGROUND, APP_STATE.RUNNING_IN_FOREGROUND], appState)) {\n              opts.chromeAndroidPackage = knownPackage;\n              log.debug(`Identified chromeAndroidPackage as '${opts.chromeAndroidPackage}' ` +\n                `for context '${context}' by querying states of Chrome app packages`);\n              break;\n            }\n          }\n        } else {\n          for (const wm of webviewsMapping) {\n            if (wm.webviewName === context && _.has(wm?.info, 'Android-Package')) {\n              opts.chromeAndroidPackage = wm.info['Android-Package'];\n              log.debug(`Identified chromeAndroidPackage as '${opts.chromeAndroidPackage}' ` +\n                `for context '${context}' by CDP`);\n              break;\n            }\n          }\n        }\n      }\n    }\n\n    cd = await this.setupNewChromedriver(opts, this.adb.curDeviceId, this.adb, context);\n    // bind our stop/exit handler, passing in context so we know which\n    // one stopped unexpectedly\n    cd.on(Chromedriver.EVENT_CHANGED, (msg) => {\n      if (msg.state === Chromedriver.STATE_STOPPED) {\n        this.onChromedriverStop(context);\n      }\n    });\n    // save the chromedriver object under the context\n    this.sessionChromedrivers[context] = cd;\n  }\n  // hook up the local variables so we can proxy this biz\n  this.chromedriver = cd;\n  this.proxyReqRes = this.chromedriver.proxyReq.bind(this.chromedriver);\n  this.proxyCommand = this.chromedriver.jwproxy.command.bind(this.chromedriver.jwproxy);\n  this.jwpProxyActive = true;\n};\n\n// Stop proxying to any Chromedriver\nhelpers.suspendChromedriverProxy = function suspendChromedriverProxy () {\n  this.chromedriver = null;\n  this.proxyReqRes = null;\n  this.proxyCommand = null;\n  this.jwpProxyActive = false;\n};\n\n// Handle an out-of-band Chromedriver stop event\nhelpers.onChromedriverStop = async function onChromedriverStop (context) {\n  log.warn(`Chromedriver for context ${context} stopped unexpectedly`);\n  if (context === this.curContext) {\n    // we exited unexpectedly while automating the current context and so want\n    // to shut down the session and respond with an error\n    let err = new Error('Chromedriver quit unexpectedly during session');\n    await this.startUnexpectedShutdown(err);\n  } else {\n    // if a Chromedriver in the non-active context barfs, we don't really\n    // care, we'll just make a new one next time we need the context.\n    log.warn(\"Chromedriver quit unexpectedly, but it wasn't the active \" +\n                'context, ignoring');\n    delete this.sessionChromedrivers[context];\n  }\n};\n\n// Intentionally stop all the chromedrivers currently active, and ignore\n// their exit events\nhelpers.stopChromedriverProxies = async function stopChromedriverProxies () {\n  this.suspendChromedriverProxy(); // make sure we turn off the proxy flag\n  for (let context of _.keys(this.sessionChromedrivers)) {\n    let cd = this.sessionChromedrivers[context];\n    log.debug(`Stopping chromedriver for context ${context}`);\n    // stop listening for the stopped state event\n    cd.removeAllListeners(Chromedriver.EVENT_CHANGED);\n    try {\n      await cd.stop();\n    } catch (err) {\n      log.warn(`Error stopping Chromedriver: ${err.message}`);\n    }\n    delete this.sessionChromedrivers[context];\n  }\n};\n\nhelpers.isChromedriverContext = function isChromedriverContext (viewName) {\n  return _.includes(viewName, WEBVIEW_WIN) || viewName === CHROMIUM_WIN;\n};\n\nhelpers.shouldDismissChromeWelcome = function shouldDismissChromeWelcome () {\n  return !!this.opts.chromeOptions &&\n         _.isArray(this.opts.chromeOptions.args) &&\n         this.opts.chromeOptions.args.includes('--no-first-run');\n};\n\nhelpers.dismissChromeWelcome = async function dismissChromeWelcome () {\n  log.info('Trying to dismiss Chrome welcome');\n  let activity = await this.getCurrentActivity();\n  if (activity !== 'org.chromium.chrome.browser.firstrun.FirstRunActivity') {\n    log.info('Chrome welcome dialog never showed up! Continuing');\n    return;\n  }\n  let el = await this.findElOrEls('id', 'com.android.chrome:id/terms_accept', false);\n  await this.click(el.ELEMENT);\n  try {\n    let el = await this.findElOrEls('id', 'com.android.chrome:id/negative_button', false);\n    await this.click(el.ELEMENT);\n  } catch (e) {\n    // DO NOTHING, THIS DEVICE DIDNT LAUNCH THE SIGNIN DIALOG\n    // IT MUST BE A NON GMS DEVICE\n    log.warn(`This device did not show Chrome SignIn dialog, ${e.message}`);\n  }\n};\n\nhelpers.startChromeSession = async function startChromeSession () {\n  log.info('Starting a chrome-based browser session');\n  let opts = _.cloneDeep(this.opts);\n\n  const knownPackages = [\n    'org.chromium.chrome.shell',\n    'com.android.chrome',\n    'com.chrome.beta',\n    'org.chromium.chrome',\n    'org.chromium.webview_shell',\n  ];\n\n  if (_.includes(knownPackages, this.opts.appPackage)) {\n    opts.chromeBundleId = this.opts.appPackage;\n  } else {\n    opts.chromeAndroidActivity = this.opts.appActivity;\n  }\n  this.chromedriver = await this.setupNewChromedriver(opts, this.adb.curDeviceId, this.adb);\n  this.chromedriver.on(Chromedriver.EVENT_CHANGED, (msg) => {\n    if (msg.state === Chromedriver.STATE_STOPPED) {\n      this.onChromedriverStop(CHROMIUM_WIN);\n    }\n  });\n\n  // Now that we have a Chrome session, we ensure that the context is\n  // appropriately set and that this chromedriver is added to the list\n  // of session chromedrivers so we can switch back and forth\n  this.curContext = CHROMIUM_WIN;\n  this.sessionChromedrivers[CHROMIUM_WIN] = this.chromedriver;\n  this.proxyReqRes = this.chromedriver.proxyReq.bind(this.chromedriver);\n  this.proxyCommand = this.chromedriver.jwproxy.command.bind(this.chromedriver.jwproxy);\n  this.jwpProxyActive = true;\n\n  if (this.shouldDismissChromeWelcome()) {\n    // dismiss Chrome welcome dialog\n    await this.dismissChromeWelcome();\n  }\n};\n\n\n/* --------------------------\n * Internal library functions\n * -------------------------- */\n\nasync function setupExistingChromedriver (chromedriver) {\n  // check the status by sending a simple window-based command to ChromeDriver\n  // if there is an error, we want to recreate the ChromeDriver session\n  if (!await chromedriver.hasWorkingWebview()) {\n    log.debug('ChromeDriver is not associated with a window. ' +\n                 'Re-initializing the session.');\n    await chromedriver.restart();\n  }\n  return chromedriver;\n}\n\n/**\n * Find a free port to have Chromedriver listen on.\n *\n * @param {array} [portSpec] - Array which is a list of ports. A list item may\n * also itself be an array of length 2 specifying a start and end port of\n * a range. Some valid port specs:\n *    - [8000, 8001, 8002]\n *    - [[8000, 8005]]\n *    - [8000, [9000, 9100]]\n *\n * @return {number} A free port\n */\nasync function getChromedriverPort (portSpec) {\n  const getPort = B.promisify(PortFinder.getPort, {context: PortFinder});\n\n  // if the user didn't give us any specific information about chromedriver\n  // port ranges, just find any free port\n  if (!portSpec) {\n    const port = await getPort();\n    log.debug(`A port was not given, using random free port: ${port}`);\n    return port;\n  }\n\n  // otherwise find the free port based on a list or range provided by the user\n  log.debug(`Finding a free port for chromedriver using spec ${JSON.stringify(portSpec)}`);\n  let foundPort = null;\n  for (const potentialPort of portSpec) {\n    let port, stopPort;\n    if (_.isArray(potentialPort)) {\n      ([port, stopPort] = potentialPort);\n    } else {\n      port = parseInt(potentialPort, 10); // ensure we have a number and not a string\n      stopPort = port;\n    }\n    try {\n      log.debug(`Checking port range ${port}:${stopPort}`);\n      foundPort = await getPort({port, stopPort});\n      break;\n    } catch (e) {\n      log.debug(`Nothing in port range ${port}:${stopPort} was available`);\n    }\n  }\n\n  if (foundPort === null) {\n    throw new Error(`Could not find a free port for chromedriver using ` +\n                    `chromedriverPorts spec ${JSON.stringify(portSpec)}`);\n  }\n\n  log.debug(`Using free port ${foundPort} for chromedriver`);\n  return foundPort;\n}\n\nhelpers.isChromedriverAutodownloadEnabled = function isChromedriverAutodownloadEnabled () {\n  if (this.isFeatureEnabled(CHROMEDRIVER_AUTODOWNLOAD_FEATURE)) {\n    return true;\n  }\n  log.debug(`Automated Chromedriver download is disabled. ` +\n    `Use '${CHROMEDRIVER_AUTODOWNLOAD_FEATURE}' server feature to enable it`);\n  return false;\n};\n\nhelpers.setupNewChromedriver = async function setupNewChromedriver (opts, curDeviceId, adb, context = null) {\n  if (opts.chromeDriverPort) {\n    log.warn(`The 'chromeDriverPort' capability is deprecated. Please use 'chromedriverPort' instead`);\n    opts.chromedriverPort = opts.chromeDriverPort;\n  }\n\n  if (opts.chromedriverPort) {\n    log.debug(`Using user-specified port ${opts.chromedriverPort} for chromedriver`);\n  } else {\n    // if a single port wasn't given, we'll look for a free one\n    opts.chromedriverPort = await getChromedriverPort(opts.chromedriverPorts);\n  }\n\n  const details = context ? webviewHelpers.getWebviewDetails(adb, context) : undefined;\n  if (!_.isEmpty(details)) {\n    log.debug('Passing web view details to the Chromedriver constructor: ' +\n      JSON.stringify(details, null, 2));\n  }\n\n  const chromedriver = new Chromedriver({\n    port: opts.chromedriverPort,\n    executable: opts.chromedriverExecutable,\n    adb,\n    cmdArgs: opts.chromedriverArgs,\n    verbose: !!opts.showChromedriverLog,\n    executableDir: opts.chromedriverExecutableDir,\n    mappingPath: opts.chromedriverChromeMappingFile,\n    bundleId: opts.chromeBundleId,\n    useSystemExecutable: opts.chromedriverUseSystemExecutable,\n    disableBuildCheck: opts.chromedriverDisableBuildCheck,\n    details,\n    isAutodownloadEnabled: (this || {}).isChromedriverAutodownloadEnabled\n      ? this.isChromedriverAutodownloadEnabled() : undefined,\n  });\n\n  // make sure there are chromeOptions\n  opts.chromeOptions = opts.chromeOptions || {};\n  // try out any prefixed chromeOptions,\n  // and strip the prefix\n  for (const opt of _.keys(opts)) {\n    if (opt.endsWith(':chromeOptions')) {\n      log.warn(`Merging '${opt}' into 'chromeOptions'. This may cause unexpected behavior`);\n      _.merge(opts.chromeOptions, opts[opt]);\n    }\n  }\n\n  const caps = webviewHelpers.createChromedriverCaps(opts, curDeviceId, details);\n  log.debug(`Before starting chromedriver, androidPackage is '${caps.chromeOptions.androidPackage}'`);\n  await chromedriver.start(caps);\n  return chromedriver;\n};\nconst setupNewChromedriver = helpers.setupNewChromedriver;\n\n\nObject.assign(extensions, commands, helpers);\nexport { commands, helpers, setupNewChromedriver };\nexport default extensions;\n"],"file":"lib/commands/context.js","sourceRoot":"../../.."}
|
|
@@ -58,7 +58,8 @@ extensions.executeMobile = async function executeMobile(mobileCommand, opts = {}
|
|
|
58
58
|
startActivity: 'mobileStartActivity',
|
|
59
59
|
broadcast: 'mobileBroadcast',
|
|
60
60
|
getContexts: 'mobileGetContexts',
|
|
61
|
-
unlock: 'mobileUnlock'
|
|
61
|
+
unlock: 'mobileUnlock',
|
|
62
|
+
refreshGpsCache: 'mobileRefreshGpsCache'
|
|
62
63
|
};
|
|
63
64
|
|
|
64
65
|
if (!_lodash.default.has(mobileCommandsMapping, mobileCommand)) {
|
|
@@ -72,4 +73,4 @@ var _default = extensions;
|
|
|
72
73
|
exports.default = _default;require('source-map-support').install();
|
|
73
74
|
|
|
74
75
|
|
|
75
|
-
//# sourceMappingURL=data:application/json;charset=utf8;base64,
|
|
76
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9jb21tYW5kcy9leGVjdXRlLmpzIl0sIm5hbWVzIjpbImV4dGVuc2lvbnMiLCJleGVjdXRlIiwic2NyaXB0IiwiYXJncyIsIm1hdGNoIiwibG9nZ2VyIiwiaW5mbyIsInJlcGxhY2UiLCJ0cmltIiwiZXhlY3V0ZU1vYmlsZSIsIl8iLCJpc0FycmF5IiwiaXNXZWJDb250ZXh0IiwiZXJyb3JzIiwiTm90SW1wbGVtZW50ZWRFcnJvciIsImVuZHBvaW50IiwiY2hyb21lZHJpdmVyIiwiandwcm94eSIsImRvd25zdHJlYW1Qcm90b2NvbCIsIlBST1RPQ09MUyIsIk1KU09OV1AiLCJjb21tYW5kIiwibW9iaWxlQ29tbWFuZCIsIm9wdHMiLCJtb2JpbGVDb21tYW5kc01hcHBpbmciLCJzaGVsbCIsImV4ZWNFbXVDb25zb2xlQ29tbWFuZCIsInN0YXJ0TG9nc0Jyb2FkY2FzdCIsInN0b3BMb2dzQnJvYWRjYXN0IiwiY2hhbmdlUGVybWlzc2lvbnMiLCJnZXRQZXJtaXNzaW9ucyIsInBlcmZvcm1FZGl0b3JBY3Rpb24iLCJzZW5zb3JTZXQiLCJnZXREZXZpY2VUaW1lIiwic3RhcnRTY3JlZW5TdHJlYW1pbmciLCJzdG9wU2NyZWVuU3RyZWFtaW5nIiwiZ2V0Tm90aWZpY2F0aW9ucyIsImxpc3RTbXMiLCJkZWxldGVGaWxlIiwiY2xlYXJBcHAiLCJzdGFydFNlcnZpY2UiLCJzdG9wU2VydmljZSIsInN0YXJ0QWN0aXZpdHkiLCJicm9hZGNhc3QiLCJnZXRDb250ZXh0cyIsInVubG9jayIsInJlZnJlc2hHcHNDYWNoZSIsImhhcyIsIlVua25vd25Db21tYW5kRXJyb3IiLCJrZXlzIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7OztBQUFBOztBQUNBOztBQUNBOztBQUVBLElBQUlBLFVBQVUsR0FBRyxFQUFqQjs7QUFFQUEsVUFBVSxDQUFDQyxPQUFYLEdBQXFCLGVBQWVBLE9BQWYsQ0FBd0JDLE1BQXhCLEVBQWdDQyxJQUFoQyxFQUFzQztBQUN6RCxNQUFJRCxNQUFNLENBQUNFLEtBQVAsQ0FBYSxVQUFiLENBQUosRUFBOEI7QUFDNUJDLG9CQUFPQyxJQUFQLENBQWEsNkJBQTRCSixNQUFPLEdBQWhEOztBQUNBQSxJQUFBQSxNQUFNLEdBQUdBLE1BQU0sQ0FBQ0ssT0FBUCxDQUFlLFVBQWYsRUFBMkIsRUFBM0IsRUFBK0JDLElBQS9CLEVBQVQ7QUFDQSxXQUFPLE1BQU0sS0FBS0MsYUFBTCxDQUFtQlAsTUFBbkIsRUFBMkJRLGdCQUFFQyxPQUFGLENBQVVSLElBQVYsSUFBa0JBLElBQUksQ0FBQyxDQUFELENBQXRCLEdBQTRCQSxJQUF2RCxDQUFiO0FBQ0Q7O0FBQ0QsTUFBSSxDQUFDLEtBQUtTLFlBQUwsRUFBTCxFQUEwQjtBQUN4QixVQUFNLElBQUlDLHlCQUFPQyxtQkFBWCxFQUFOO0FBQ0Q7O0FBQ0QsUUFBTUMsUUFBUSxHQUFHLEtBQUtDLFlBQUwsQ0FBa0JDLE9BQWxCLENBQTBCQyxrQkFBMUIsS0FBaURDLDRCQUFVQyxPQUEzRCxHQUNiLFVBRGEsR0FFYixlQUZKO0FBR0EsU0FBTyxNQUFNLEtBQUtKLFlBQUwsQ0FBa0JDLE9BQWxCLENBQTBCSSxPQUExQixDQUFrQ04sUUFBbEMsRUFBNEMsTUFBNUMsRUFBb0Q7QUFDL0RiLElBQUFBLE1BRCtEO0FBRS9EQyxJQUFBQTtBQUYrRCxHQUFwRCxDQUFiO0FBSUQsQ0FoQkQ7O0FBa0JBSCxVQUFVLENBQUNTLGFBQVgsR0FBMkIsZUFBZUEsYUFBZixDQUE4QmEsYUFBOUIsRUFBNkNDLElBQUksR0FBRyxFQUFwRCxFQUF3RDtBQUNqRixRQUFNQyxxQkFBcUIsR0FBRztBQUM1QkMsSUFBQUEsS0FBSyxFQUFFLGFBRHFCO0FBRzVCQyxJQUFBQSxxQkFBcUIsRUFBRSw2QkFISztBQUs1QkMsSUFBQUEsa0JBQWtCLEVBQUUsMEJBTFE7QUFNNUJDLElBQUFBLGlCQUFpQixFQUFFLHlCQU5TO0FBUTVCQyxJQUFBQSxpQkFBaUIsRUFBRSx5QkFSUztBQVM1QkMsSUFBQUEsY0FBYyxFQUFFLHNCQVRZO0FBVzVCQyxJQUFBQSxtQkFBbUIsRUFBRSwyQkFYTztBQWE1QkMsSUFBQUEsU0FBUyxFQUFFLFdBYmlCO0FBZTVCQyxJQUFBQSxhQUFhLEVBQUUscUJBZmE7QUFpQjVCQyxJQUFBQSxvQkFBb0IsRUFBRSw0QkFqQk07QUFrQjVCQyxJQUFBQSxtQkFBbUIsRUFBRSwyQkFsQk87QUFvQjVCQyxJQUFBQSxnQkFBZ0IsRUFBRSx3QkFwQlU7QUFzQjVCQyxJQUFBQSxPQUFPLEVBQUUsZUF0Qm1CO0FBd0I1QkMsSUFBQUEsVUFBVSxFQUFFLGtCQXhCZ0I7QUEwQjVCQyxJQUFBQSxRQUFRLEVBQUUsZ0JBMUJrQjtBQTRCNUJDLElBQUFBLFlBQVksRUFBRSxvQkE1QmM7QUE2QjVCQyxJQUFBQSxXQUFXLEVBQUUsbUJBN0JlO0FBOEI1QkMsSUFBQUEsYUFBYSxFQUFFLHFCQTlCYTtBQStCNUJDLElBQUFBLFNBQVMsRUFBRSxpQkEvQmlCO0FBaUM1QkMsSUFBQUEsV0FBVyxFQUFFLG1CQWpDZTtBQW1DNUJDLElBQUFBLE1BQU0sRUFBRSxjQW5Db0I7QUFxQzVCQyxJQUFBQSxlQUFlLEVBQUU7QUFyQ1csR0FBOUI7O0FBd0NBLE1BQUksQ0FBQ3BDLGdCQUFFcUMsR0FBRixDQUFNdkIscUJBQU4sRUFBNkJGLGFBQTdCLENBQUwsRUFBa0Q7QUFDaEQsVUFBTSxJQUFJVCx5QkFBT21DLG1CQUFYLENBQWdDLDJCQUEwQjFCLGFBQWMsS0FBekMsR0FDbEMsUUFBT1osZ0JBQUV1QyxJQUFGLENBQU96QixxQkFBUCxDQUE4QiwwQkFEbEMsQ0FBTjtBQUVEOztBQUNELFNBQU8sTUFBTSxLQUFLQSxxQkFBcUIsQ0FBQ0YsYUFBRCxDQUExQixFQUEyQ0MsSUFBM0MsQ0FBYjtBQUNELENBOUNEOztlQWdEZXZCLFUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgXyBmcm9tICdsb2Rhc2gnO1xuaW1wb3J0IHsgZXJyb3JzLCBQUk9UT0NPTFMgfSBmcm9tICdhcHBpdW0tYmFzZS1kcml2ZXInO1xuaW1wb3J0IGxvZ2dlciBmcm9tICcuLi9sb2dnZXInO1xuXG5sZXQgZXh0ZW5zaW9ucyA9IHt9O1xuXG5leHRlbnNpb25zLmV4ZWN1dGUgPSBhc3luYyBmdW5jdGlvbiBleGVjdXRlIChzY3JpcHQsIGFyZ3MpIHtcbiAgaWYgKHNjcmlwdC5tYXRjaCgvXm1vYmlsZTovKSkge1xuICAgIGxvZ2dlci5pbmZvKGBFeGVjdXRpbmcgbmF0aXZlIGNvbW1hbmQgJyR7c2NyaXB0fSdgKTtcbiAgICBzY3JpcHQgPSBzY3JpcHQucmVwbGFjZSgvXm1vYmlsZTovLCAnJykudHJpbSgpO1xuICAgIHJldHVybiBhd2FpdCB0aGlzLmV4ZWN1dGVNb2JpbGUoc2NyaXB0LCBfLmlzQXJyYXkoYXJncykgPyBhcmdzWzBdIDogYXJncyk7XG4gIH1cbiAgaWYgKCF0aGlzLmlzV2ViQ29udGV4dCgpKSB7XG4gICAgdGhyb3cgbmV3IGVycm9ycy5Ob3RJbXBsZW1lbnRlZEVycm9yKCk7XG4gIH1cbiAgY29uc3QgZW5kcG9pbnQgPSB0aGlzLmNocm9tZWRyaXZlci5qd3Byb3h5LmRvd25zdHJlYW1Qcm90b2NvbCA9PT0gUFJPVE9DT0xTLk1KU09OV1BcbiAgICA/ICcvZXhlY3V0ZSdcbiAgICA6ICcvZXhlY3V0ZS9zeW5jJztcbiAgcmV0dXJuIGF3YWl0IHRoaXMuY2hyb21lZHJpdmVyLmp3cHJveHkuY29tbWFuZChlbmRwb2ludCwgJ1BPU1QnLCB7XG4gICAgc2NyaXB0LFxuICAgIGFyZ3MsXG4gIH0pO1xufTtcblxuZXh0ZW5zaW9ucy5leGVjdXRlTW9iaWxlID0gYXN5bmMgZnVuY3Rpb24gZXhlY3V0ZU1vYmlsZSAobW9iaWxlQ29tbWFuZCwgb3B0cyA9IHt9KSB7XG4gIGNvbnN0IG1vYmlsZUNvbW1hbmRzTWFwcGluZyA9IHtcbiAgICBzaGVsbDogJ21vYmlsZVNoZWxsJyxcblxuICAgIGV4ZWNFbXVDb25zb2xlQ29tbWFuZDogJ21vYmlsZUV4ZWNFbXVDb25zb2xlQ29tbWFuZCcsXG5cbiAgICBzdGFydExvZ3NCcm9hZGNhc3Q6ICdtb2JpbGVTdGFydExvZ3NCcm9hZGNhc3QnLFxuICAgIHN0b3BMb2dzQnJvYWRjYXN0OiAnbW9iaWxlU3RvcExvZ3NCcm9hZGNhc3QnLFxuXG4gICAgY2hhbmdlUGVybWlzc2lvbnM6ICdtb2JpbGVDaGFuZ2VQZXJtaXNzaW9ucycsXG4gICAgZ2V0UGVybWlzc2lvbnM6ICdtb2JpbGVHZXRQZXJtaXNzaW9ucycsXG5cbiAgICBwZXJmb3JtRWRpdG9yQWN0aW9uOiAnbW9iaWxlUGVyZm9ybUVkaXRvckFjdGlvbicsXG5cbiAgICBzZW5zb3JTZXQ6ICdzZW5zb3JTZXQnLFxuXG4gICAgZ2V0RGV2aWNlVGltZTogJ21vYmlsZUdldERldmljZVRpbWUnLFxuXG4gICAgc3RhcnRTY3JlZW5TdHJlYW1pbmc6ICdtb2JpbGVTdGFydFNjcmVlblN0cmVhbWluZycsXG4gICAgc3RvcFNjcmVlblN0cmVhbWluZzogJ21vYmlsZVN0b3BTY3JlZW5TdHJlYW1pbmcnLFxuXG4gICAgZ2V0Tm90aWZpY2F0aW9uczogJ21vYmlsZUdldE5vdGlmaWNhdGlvbnMnLFxuXG4gICAgbGlzdFNtczogJ21vYmlsZUxpc3RTbXMnLFxuXG4gICAgZGVsZXRlRmlsZTogJ21vYmlsZURlbGV0ZUZpbGUnLFxuXG4gICAgY2xlYXJBcHA6ICdtb2JpbGVDbGVhckFwcCcsXG5cbiAgICBzdGFydFNlcnZpY2U6ICdtb2JpbGVTdGFydFNlcnZpY2UnLFxuICAgIHN0b3BTZXJ2aWNlOiAnbW9iaWxlU3RvcFNlcnZpY2UnLFxuICAgIHN0YXJ0QWN0aXZpdHk6ICdtb2JpbGVTdGFydEFjdGl2aXR5JyxcbiAgICBicm9hZGNhc3Q6ICdtb2JpbGVCcm9hZGNhc3QnLFxuXG4gICAgZ2V0Q29udGV4dHM6ICdtb2JpbGVHZXRDb250ZXh0cycsXG5cbiAgICB1bmxvY2s6ICdtb2JpbGVVbmxvY2snLFxuXG4gICAgcmVmcmVzaEdwc0NhY2hlOiAnbW9iaWxlUmVmcmVzaEdwc0NhY2hlJyxcbiAgfTtcblxuICBpZiAoIV8uaGFzKG1vYmlsZUNvbW1hbmRzTWFwcGluZywgbW9iaWxlQ29tbWFuZCkpIHtcbiAgICB0aHJvdyBuZXcgZXJyb3JzLlVua25vd25Db21tYW5kRXJyb3IoYFVua25vd24gbW9iaWxlIGNvbW1hbmQgXCIke21vYmlsZUNvbW1hbmR9XCIuIGAgK1xuICAgICAgYE9ubHkgJHtfLmtleXMobW9iaWxlQ29tbWFuZHNNYXBwaW5nKX0gY29tbWFuZHMgYXJlIHN1cHBvcnRlZC5gKTtcbiAgfVxuICByZXR1cm4gYXdhaXQgdGhpc1ttb2JpbGVDb21tYW5kc01hcHBpbmdbbW9iaWxlQ29tbWFuZF1dKG9wdHMpO1xufTtcblxuZXhwb3J0IGRlZmF1bHQgZXh0ZW5zaW9ucztcbiJdLCJmaWxlIjoibGliL2NvbW1hbmRzL2V4ZWN1dGUuanMiLCJzb3VyY2VSb290IjoiLi4vLi4vLi4ifQ==
|
|
@@ -33,6 +33,20 @@ function parseContainerPath(remotePath) {
|
|
|
33
33
|
return [match[1], _path.default.posix.resolve(`/data/data/${match[1]}`, match[2])];
|
|
34
34
|
}
|
|
35
35
|
|
|
36
|
+
async function scanMedia(adb, remotePath) {
|
|
37
|
+
_logger.default.debug(`Performing media scan of '${remotePath}'`);
|
|
38
|
+
|
|
39
|
+
try {
|
|
40
|
+
if ((await adb.getApiLevel()) >= 29) {
|
|
41
|
+
await adb.scanMedia(remotePath);
|
|
42
|
+
} else {
|
|
43
|
+
await adb.shell(['am', 'broadcast', '-a', ANDROID_MEDIA_RESCAN_INTENT, '-d', `file://${remotePath}`]);
|
|
44
|
+
}
|
|
45
|
+
} catch (e) {
|
|
46
|
+
_logger.default.warn(`Ignoring an unexpected error upon media scanning of '${remotePath}': ${e.stderr || e.message}`);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
36
50
|
function escapePath(p) {
|
|
37
51
|
return p.replace(/'/g, `\\'`);
|
|
38
52
|
}
|
|
@@ -116,14 +130,7 @@ commands.pushFile = async function pushFile(remotePath, base64Data) {
|
|
|
116
130
|
}
|
|
117
131
|
} else {
|
|
118
132
|
await this.adb.push(localFile, remotePath);
|
|
119
|
-
|
|
120
|
-
_logger.default.info('After pushing media file, broadcasting media scan intent');
|
|
121
|
-
|
|
122
|
-
try {
|
|
123
|
-
await this.adb.shell(['am', 'broadcast', '-a', ANDROID_MEDIA_RESCAN_INTENT, '-d', `file://${remotePath}`]);
|
|
124
|
-
} catch (e) {
|
|
125
|
-
_logger.default.warn(`Got error broadcasting media scan intent: ${e.message}; ignoring`);
|
|
126
|
-
}
|
|
133
|
+
await scanMedia(this.adb, remotePath);
|
|
127
134
|
}
|
|
128
135
|
} finally {
|
|
129
136
|
if (await _appiumSupport.fs.exists(localFile)) {
|
|
@@ -232,4 +239,4 @@ var _default = commands;
|
|
|
232
239
|
exports.default = _default;require('source-map-support').install();
|
|
233
240
|
|
|
234
241
|
|
|
235
|
-
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/file-actions.js"],"names":["CONTAINER_PATH_MARKER","CONTAINER_PATH_PATTERN","RegExp","ANDROID_MEDIA_RESCAN_INTENT","commands","parseContainerPath","remotePath","match","exec","log","errorAndThrow","path","posix","resolve","escapePath","p","replace","pullFile","endsWith","tmpDestination","startsWith","packageId","pathInContainer","debug","basename","adb","shell","e","message","localFile","tempDir","prefix","suffix","pull","util","toInMemoryBase64","toString","fs","exists","unlink","pushFile","base64Data","_","isArray","Buffer","from","content","writeFile","dirname","push","info","warn","pullFolder","localFolder","zip","toInMemoryZip","encodeToBase64","deleteFileOrFolder","performRemoteFsCheck","op","runAs","passFlag","checkCmd","fullCmd","includes","ign","isFile","isDir","isPresent","dstPath","pkgId","expectsFile","mobileDeleteFile","opts"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport { fs, util, zip, tempDir} from 'appium-support';\nimport log from '../logger';\nimport path from 'path';\n\n\nconst CONTAINER_PATH_MARKER = '@';\n// https://regex101.com/r/PLdB0G/2\nconst CONTAINER_PATH_PATTERN = new RegExp(`^${CONTAINER_PATH_MARKER}([^/]+)/(.+)`);\nconst ANDROID_MEDIA_RESCAN_INTENT = 'android.intent.action.MEDIA_SCANNER_SCAN_FILE';\n\n\nconst commands = {};\n\n/**\n * Parses the actual destination path from the given value\n *\n * @param {string} remotePath The preformatted remote path, which looks like\n * `@my.app.id/my/path`\n * @returns {Array<string>} An array, where the first item is the parsed package\n * identifier and the second one is the actual destination path inside the package.\n * @throws {Error} If the given string cannot be parsed\n */\nfunction parseContainerPath (remotePath) {\n  const match = CONTAINER_PATH_PATTERN.exec(remotePath);\n  if (!match) {\n    log.errorAndThrow(`It is expected that package identifier is separated from the relative path with a single slash. ` +\n                      `'${remotePath}' is given instead`);\n  }\n  return [match[1], path.posix.resolve(`/data/data/${match[1]}`, match[2])];\n}\n\n/**\n * A small helper, which escapes single quotes in paths,\n * so they are safe to be passed as arguments of shell commands\n *\n * @param {string} p The initial remote path\n * @returns {string} The escaped path value\n */\nfunction escapePath (p) {\n  return p.replace(/'/g, `\\\\'`);\n}\n\n/**\n * Pulls a remote file from the device.\n * It is required, that a package has debugging flag enabled\n * in order to access its files.\n *\n * @param {string} remotePath The full path to the remote file\n * or a specially formatted path, which points to an item inside app bundle\n * @returns {string} Base64 encoded content of the pulled file\n * @throws {Error} If the pull operation failed\n */\ncommands.pullFile = async function pullFile (remotePath) {\n  if (remotePath.endsWith('/')) {\n    log.errorAndThrow(`It is expected that remote path points to a file and not to a folder. ` +\n                      `'${remotePath}' is given instead`);\n  }\n  let tmpDestination = null;\n  if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n    const [packageId, pathInContainer] = parseContainerPath(remotePath);\n    log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'. Will get the data from '${pathInContainer}'`);\n    tmpDestination = `/data/local/tmp/${path.posix.basename(pathInContainer)}`;\n    try {\n      await this.adb.shell(['run-as', packageId, `chmod 777 '${escapePath(pathInContainer)}'`]);\n      await this.adb.shell([\n        'run-as', packageId,\n        `cp -f '${escapePath(pathInContainer)}' '${escapePath(tmpDestination)}'`\n      ]);\n    } catch (e) {\n      log.errorAndThrow(`Cannot access the container of '${packageId}' application. ` +\n                        `Is the application installed and has 'debuggable' build option set to true? ` +\n                        `Original error: ${e.message}`);\n    }\n  }\n  const localFile = await tempDir.path({prefix: 'appium', suffix: '.tmp'});\n  try {\n    await this.adb.pull(tmpDestination || remotePath, localFile);\n    return (await util.toInMemoryBase64(localFile)).toString();\n  } finally {\n    if (await fs.exists(localFile)) {\n      await fs.unlink(localFile);\n    }\n    if (tmpDestination) {\n      await this.adb.shell(['rm', '-f', tmpDestination]);\n    }\n  }\n};\n\n/**\n * Pushed the given data to a file on the remote device\n * It is required, that a package has debugging flag enabled\n * in order to access its files.\n *\n * @param {string} remotePath The full path to the remote file or\n * a file inside a package bundle\n * @param {string} base64Data Base64 encoded data to be written to the\n * remote file. The remote file will be silently overridden if it already exists.\n * @throws {Error} If there was an error while pushing the data\n */\ncommands.pushFile = async function pushFile (remotePath, base64Data) {\n  if (remotePath.endsWith('/')) {\n    log.errorAndThrow(`It is expected that remote path points to a file and not to a folder. ` +\n                      `'${remotePath}' is given instead`);\n  }\n  const localFile = await tempDir.path({prefix: 'appium', suffix: '.tmp'});\n  if (_.isArray(base64Data)) {\n    // some clients (ahem) java, send a byte array encoding utf8 characters\n    // instead of a string, which would be infinitely better!\n    base64Data = Buffer.from(base64Data).toString('utf8');\n  }\n  const content = Buffer.from(base64Data, 'base64');\n  let tmpDestination = null;\n  try {\n    await fs.writeFile(localFile, content.toString('binary'), 'binary');\n    if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n      const [packageId, pathInContainer] = parseContainerPath(remotePath);\n      log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'. Will put the data into '${pathInContainer}'`);\n      tmpDestination = `/data/local/tmp/${path.posix.basename(pathInContainer)}`;\n      try {\n        await this.adb.shell(\n          ['run-as', packageId, `mkdir -p '${escapePath(path.posix.dirname(pathInContainer))}'`]\n        );\n        await this.adb.shell(['run-as', packageId, `touch '${escapePath(pathInContainer)}'`]);\n        await this.adb.shell(['run-as', packageId, `chmod 777 '${escapePath(pathInContainer)}'`]);\n        await this.adb.push(localFile, tmpDestination);\n        await this.adb.shell([\n          'run-as', packageId,\n          `cp -f '${escapePath(tmpDestination)}' '${escapePath(pathInContainer)}'`\n        ]);\n      } catch (e) {\n        log.errorAndThrow(`Cannot access the container of '${packageId}' application. ` +\n                          `Is the application installed and has 'debuggable' build option set to true? ` +\n                          `Original error: ${e.message}`);\n      }\n    } else {\n      // adb push creates folders and overwrites existing files.\n      await this.adb.push(localFile, remotePath);\n\n      // if we have pushed a file, it might be a media file, so ensure that\n      // apps know about it\n      log.info('After pushing media file, broadcasting media scan intent');\n      try {\n        await this.adb.shell(['am', 'broadcast', '-a',\n          ANDROID_MEDIA_RESCAN_INTENT, '-d', `file://${remotePath}`]);\n      } catch (e) {\n        log.warn(`Got error broadcasting media scan intent: ${e.message}; ignoring`);\n      }\n    }\n  } finally {\n    if (await fs.exists(localFile)) {\n      await fs.unlink(localFile);\n    }\n    if (tmpDestination) {\n      await this.adb.shell(['rm', '-f', tmpDestination]);\n    }\n  }\n};\n\n/**\n * Pulls the whole folder from the remote device\n *\n * @param {string} remotePath The full path to a folder on the\n * remote device or a folder inside an application bundle\n * @returns {string} Base64-encoded and zipped content of the folder\n * @throws {Error} If there was a failure while getting the folder content\n */\ncommands.pullFolder = async function pullFolder (remotePath) {\n  let localFolder = await tempDir.path({prefix: 'appium'});\n  await this.adb.pull(remotePath, localFolder);\n  return (await zip.toInMemoryZip(localFolder, {\n    encodeToBase64: true,\n  })).toString();\n};\n\n/**\n * Deletes the given folder or file from the remote device\n *\n * @param {ADB} adb\n * @param {string} remotePath The full path to the remote folder\n * or file (folder names must end with a single slash)\n * @throws {Error} If the provided remote path is invalid or\n * the package content cannot be accessed\n * @returns {boolean} `true` if the remote item has been successfully deleted.\n * If the remote path is valid, but the remote path does not exist\n * this function return `false`.\n */\nasync function deleteFileOrFolder (adb, remotePath) {\n  const performRemoteFsCheck = async (p, op, runAs = null) => {\n    const passFlag = '__PASS__';\n    const checkCmd = `[ -${op} '${escapePath(p)}' ] && echo ${passFlag}`;\n    const fullCmd = runAs ? `run-as ${runAs} ${checkCmd}` : checkCmd;\n    try {\n      return _.includes(await adb.shell([fullCmd]), passFlag);\n    } catch (ign) {\n      return false;\n    }\n  };\n  const isFile = async (p, runAs = null) => await performRemoteFsCheck(p, 'f', runAs);\n  const isDir = async (p, runAs = null) => await performRemoteFsCheck(p, 'd', runAs);\n  const isPresent = async (p, runAs = null) => await performRemoteFsCheck(p, 'e', runAs);\n\n  let dstPath = remotePath;\n  let pkgId = null;\n  if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n    const [packageId, pathInContainer] = parseContainerPath(remotePath);\n    log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'`);\n    dstPath = pathInContainer;\n    pkgId = packageId;\n  }\n\n  if (pkgId) {\n    try {\n      await adb.shell(['run-as', pkgId, 'ls']);\n    } catch (e) {\n      log.errorAndThrow(`Cannot access the container of '${pkgId}' application. ` +\n        `Is the application installed and has 'debuggable' build option set to true? ` +\n        `Original error: ${e.message}`);\n    }\n  }\n\n  if (!await isPresent(dstPath, pkgId)) {\n    log.info(`The item at '${dstPath}' does not exist. Perhaps, already deleted?`);\n    return false;\n  }\n\n  const expectsFile = !remotePath.endsWith('/');\n  if (expectsFile && !await isFile(dstPath, pkgId)) {\n    log.errorAndThrow(`The item at '${dstPath}' is not a file`);\n  } else if (!expectsFile && !await isDir(dstPath, pkgId)) {\n    log.errorAndThrow(`The item at '${dstPath}' is not a folder`);\n  }\n\n  if (pkgId) {\n    await adb.shell(\n      ['run-as', pkgId, `rm -f${expectsFile ? '' : 'r'} '${escapePath(dstPath)}'`]);\n  } else {\n    await adb.shell(['rm', `-f${expectsFile ? '' : 'r'}`, dstPath]);\n  }\n  if (await isPresent(dstPath, pkgId)) {\n    log.errorAndThrow(`The item at '${dstPath}' still exists after being deleted. ` +\n      `Is it writable?`);\n  }\n  return true;\n}\n\n/**\n * @typedef {Object} DeleteFileOpts\n * @property {!string} remotePath The full path to the remote file\n * or a file inside an application bundle (for example `@my.app.id/path/in/bundle`)\n */\n\n/**\n * Deletes a file on the remote device\n *\n * @param {DeleteFileOpts} opts\n * @returns {boolean} `true` if the remote file has been successfully deleted.\n * If the path to a remote file is valid, but the file itself does not exist\n * then `false` is returned.\n * @throws {Error} If the argument is invalid or there was an error while\n * deleting the file\n */\ncommands.mobileDeleteFile = async function mobileDeleteFile (opts = {}) {\n  const {remotePath} = opts;\n  if (!remotePath) {\n    log.errorAndThrow(`The 'remotePath' argument is mandatory`);\n  }\n  if (remotePath.endsWith('/')) {\n    log.errorAndThrow(`It is expected that remote path points to a folder and not to a file. ` +\n                      `'${remotePath}' is given instead`);\n  }\n  return await deleteFileOrFolder(this.adb, remotePath);\n};\n\nexport { commands };\nexport default commands;\n"],"file":"lib/commands/file-actions.js","sourceRoot":"../../.."}
|
|
242
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/file-actions.js"],"names":["CONTAINER_PATH_MARKER","CONTAINER_PATH_PATTERN","RegExp","ANDROID_MEDIA_RESCAN_INTENT","commands","parseContainerPath","remotePath","match","exec","log","errorAndThrow","path","posix","resolve","scanMedia","adb","debug","getApiLevel","shell","e","warn","stderr","message","escapePath","p","replace","pullFile","endsWith","tmpDestination","startsWith","packageId","pathInContainer","basename","localFile","tempDir","prefix","suffix","pull","util","toInMemoryBase64","toString","fs","exists","unlink","pushFile","base64Data","_","isArray","Buffer","from","content","writeFile","dirname","push","pullFolder","localFolder","zip","toInMemoryZip","encodeToBase64","deleteFileOrFolder","performRemoteFsCheck","op","runAs","passFlag","checkCmd","fullCmd","includes","ign","isFile","isDir","isPresent","dstPath","pkgId","info","expectsFile","mobileDeleteFile","opts"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport { fs, util, zip, tempDir} from 'appium-support';\nimport log from '../logger';\nimport path from 'path';\n\n\nconst CONTAINER_PATH_MARKER = '@';\n// https://regex101.com/r/PLdB0G/2\nconst CONTAINER_PATH_PATTERN = new RegExp(`^${CONTAINER_PATH_MARKER}([^/]+)/(.+)`);\nconst ANDROID_MEDIA_RESCAN_INTENT = 'android.intent.action.MEDIA_SCANNER_SCAN_FILE';\n\n\nconst commands = {};\n\n/**\n * Parses the actual destination path from the given value\n *\n * @param {string} remotePath The preformatted remote path, which looks like\n * `@my.app.id/my/path`\n * @returns {Array<string>} An array, where the first item is the parsed package\n * identifier and the second one is the actual destination path inside the package.\n * @throws {Error} If the given string cannot be parsed\n */\nfunction parseContainerPath (remotePath) {\n  const match = CONTAINER_PATH_PATTERN.exec(remotePath);\n  if (!match) {\n    log.errorAndThrow(`It is expected that package identifier is separated from the relative path with a single slash. ` +\n                      `'${remotePath}' is given instead`);\n  }\n  return [match[1], path.posix.resolve(`/data/data/${match[1]}`, match[2])];\n}\n\n/**\n * Scans the given file/folder on the remote device\n * and adds matching items to the device's media library.\n * Exceptions are ignored and written into the log.\n *\n * @param {ADB} adb ADB instance\n * @param {string} remotePath The file/folder path on the remote device\n */\nasync function scanMedia (adb, remotePath) {\n  log.debug(`Performing media scan of '${remotePath}'`);\n  try {\n    // https://github.com/appium/appium/issues/16184\n    if (await adb.getApiLevel() >= 29) {\n      await adb.scanMedia(remotePath);\n    } else {\n      await adb.shell([\n        'am', 'broadcast',\n        '-a', ANDROID_MEDIA_RESCAN_INTENT,\n        '-d', `file://${remotePath}`\n      ]);\n    }\n  } catch (e) {\n    log.warn(`Ignoring an unexpected error upon media scanning of '${remotePath}': ${e.stderr || e.message}`);\n  }\n}\n\n/**\n * A small helper, which escapes single quotes in paths,\n * so they are safe to be passed as arguments of shell commands\n *\n * @param {string} p The initial remote path\n * @returns {string} The escaped path value\n */\nfunction escapePath (p) {\n  return p.replace(/'/g, `\\\\'`);\n}\n\n/**\n * Pulls a remote file from the device.\n * It is required, that a package has debugging flag enabled\n * in order to access its files.\n *\n * @param {string} remotePath The full path to the remote file\n * or a specially formatted path, which points to an item inside app bundle\n * @returns {string} Base64 encoded content of the pulled file\n * @throws {Error} If the pull operation failed\n */\ncommands.pullFile = async function pullFile (remotePath) {\n  if (remotePath.endsWith('/')) {\n    log.errorAndThrow(`It is expected that remote path points to a file and not to a folder. ` +\n                      `'${remotePath}' is given instead`);\n  }\n  let tmpDestination = null;\n  if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n    const [packageId, pathInContainer] = parseContainerPath(remotePath);\n    log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'. Will get the data from '${pathInContainer}'`);\n    tmpDestination = `/data/local/tmp/${path.posix.basename(pathInContainer)}`;\n    try {\n      await this.adb.shell(['run-as', packageId, `chmod 777 '${escapePath(pathInContainer)}'`]);\n      await this.adb.shell([\n        'run-as', packageId,\n        `cp -f '${escapePath(pathInContainer)}' '${escapePath(tmpDestination)}'`\n      ]);\n    } catch (e) {\n      log.errorAndThrow(`Cannot access the container of '${packageId}' application. ` +\n                        `Is the application installed and has 'debuggable' build option set to true? ` +\n                        `Original error: ${e.message}`);\n    }\n  }\n  const localFile = await tempDir.path({prefix: 'appium', suffix: '.tmp'});\n  try {\n    await this.adb.pull(tmpDestination || remotePath, localFile);\n    return (await util.toInMemoryBase64(localFile)).toString();\n  } finally {\n    if (await fs.exists(localFile)) {\n      await fs.unlink(localFile);\n    }\n    if (tmpDestination) {\n      await this.adb.shell(['rm', '-f', tmpDestination]);\n    }\n  }\n};\n\n/**\n * Pushed the given data to a file on the remote device\n * It is required, that a package has debugging flag enabled\n * in order to access its files.\n *\n * @param {string} remotePath The full path to the remote file or\n * a file inside a package bundle\n * @param {string} base64Data Base64 encoded data to be written to the\n * remote file. The remote file will be silently overridden if it already exists.\n * @throws {Error} If there was an error while pushing the data\n */\ncommands.pushFile = async function pushFile (remotePath, base64Data) {\n  if (remotePath.endsWith('/')) {\n    log.errorAndThrow(`It is expected that remote path points to a file and not to a folder. ` +\n                      `'${remotePath}' is given instead`);\n  }\n  const localFile = await tempDir.path({prefix: 'appium', suffix: '.tmp'});\n  if (_.isArray(base64Data)) {\n    // some clients (ahem) java, send a byte array encoding utf8 characters\n    // instead of a string, which would be infinitely better!\n    base64Data = Buffer.from(base64Data).toString('utf8');\n  }\n  const content = Buffer.from(base64Data, 'base64');\n  let tmpDestination = null;\n  try {\n    await fs.writeFile(localFile, content.toString('binary'), 'binary');\n    if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n      const [packageId, pathInContainer] = parseContainerPath(remotePath);\n      log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'. Will put the data into '${pathInContainer}'`);\n      tmpDestination = `/data/local/tmp/${path.posix.basename(pathInContainer)}`;\n      try {\n        await this.adb.shell(\n          ['run-as', packageId, `mkdir -p '${escapePath(path.posix.dirname(pathInContainer))}'`]\n        );\n        await this.adb.shell(['run-as', packageId, `touch '${escapePath(pathInContainer)}'`]);\n        await this.adb.shell(['run-as', packageId, `chmod 777 '${escapePath(pathInContainer)}'`]);\n        await this.adb.push(localFile, tmpDestination);\n        await this.adb.shell([\n          'run-as', packageId,\n          `cp -f '${escapePath(tmpDestination)}' '${escapePath(pathInContainer)}'`\n        ]);\n      } catch (e) {\n        log.errorAndThrow(`Cannot access the container of '${packageId}' application. ` +\n                          `Is the application installed and has 'debuggable' build option set to true? ` +\n                          `Original error: ${e.message}`);\n      }\n    } else {\n      // adb push creates folders and overwrites existing files.\n      await this.adb.push(localFile, remotePath);\n\n      // if we have pushed a file, it might be a media file, so ensure that\n      // apps know about it\n      await scanMedia(this.adb, remotePath);\n    }\n  } finally {\n    if (await fs.exists(localFile)) {\n      await fs.unlink(localFile);\n    }\n    if (tmpDestination) {\n      await this.adb.shell(['rm', '-f', tmpDestination]);\n    }\n  }\n};\n\n/**\n * Pulls the whole folder from the remote device\n *\n * @param {string} remotePath The full path to a folder on the\n * remote device or a folder inside an application bundle\n * @returns {string} Base64-encoded and zipped content of the folder\n * @throws {Error} If there was a failure while getting the folder content\n */\ncommands.pullFolder = async function pullFolder (remotePath) {\n  let localFolder = await tempDir.path({prefix: 'appium'});\n  await this.adb.pull(remotePath, localFolder);\n  return (await zip.toInMemoryZip(localFolder, {\n    encodeToBase64: true,\n  })).toString();\n};\n\n/**\n * Deletes the given folder or file from the remote device\n *\n * @param {ADB} adb\n * @param {string} remotePath The full path to the remote folder\n * or file (folder names must end with a single slash)\n * @throws {Error} If the provided remote path is invalid or\n * the package content cannot be accessed\n * @returns {boolean} `true` if the remote item has been successfully deleted.\n * If the remote path is valid, but the remote path does not exist\n * this function return `false`.\n */\nasync function deleteFileOrFolder (adb, remotePath) {\n  const performRemoteFsCheck = async (p, op, runAs = null) => {\n    const passFlag = '__PASS__';\n    const checkCmd = `[ -${op} '${escapePath(p)}' ] && echo ${passFlag}`;\n    const fullCmd = runAs ? `run-as ${runAs} ${checkCmd}` : checkCmd;\n    try {\n      return _.includes(await adb.shell([fullCmd]), passFlag);\n    } catch (ign) {\n      return false;\n    }\n  };\n  const isFile = async (p, runAs = null) => await performRemoteFsCheck(p, 'f', runAs);\n  const isDir = async (p, runAs = null) => await performRemoteFsCheck(p, 'd', runAs);\n  const isPresent = async (p, runAs = null) => await performRemoteFsCheck(p, 'e', runAs);\n\n  let dstPath = remotePath;\n  let pkgId = null;\n  if (remotePath.startsWith(CONTAINER_PATH_MARKER)) {\n    const [packageId, pathInContainer] = parseContainerPath(remotePath);\n    log.debug(`Parsed package identifier '${packageId}' from '${remotePath}'`);\n    dstPath = pathInContainer;\n    pkgId = packageId;\n  }\n\n  if (pkgId) {\n    try {\n      await adb.shell(['run-as', pkgId, 'ls']);\n    } catch (e) {\n      log.errorAndThrow(`Cannot access the container of '${pkgId}' application. ` +\n        `Is the application installed and has 'debuggable' build option set to true? ` +\n        `Original error: ${e.message}`);\n    }\n  }\n\n  if (!await isPresent(dstPath, pkgId)) {\n    log.info(`The item at '${dstPath}' does not exist. Perhaps, already deleted?`);\n    return false;\n  }\n\n  const expectsFile = !remotePath.endsWith('/');\n  if (expectsFile && !await isFile(dstPath, pkgId)) {\n    log.errorAndThrow(`The item at '${dstPath}' is not a file`);\n  } else if (!expectsFile && !await isDir(dstPath, pkgId)) {\n    log.errorAndThrow(`The item at '${dstPath}' is not a folder`);\n  }\n\n  if (pkgId) {\n    await adb.shell(\n      ['run-as', pkgId, `rm -f${expectsFile ? '' : 'r'} '${escapePath(dstPath)}'`]);\n  } else {\n    await adb.shell(['rm', `-f${expectsFile ? '' : 'r'}`, dstPath]);\n  }\n  if (await isPresent(dstPath, pkgId)) {\n    log.errorAndThrow(`The item at '${dstPath}' still exists after being deleted. ` +\n      `Is it writable?`);\n  }\n  return true;\n}\n\n/**\n * @typedef {Object} DeleteFileOpts\n * @property {!string} remotePath The full path to the remote file\n * or a file inside an application bundle (for example `@my.app.id/path/in/bundle`)\n */\n\n/**\n * Deletes a file on the remote device\n *\n * @param {DeleteFileOpts} opts\n * @returns {boolean} `true` if the remote file has been successfully deleted.\n * If the path to a remote file is valid, but the file itself does not exist\n * then `false` is returned.\n * @throws {Error} If the argument is invalid or there was an error while\n * deleting the file\n */\ncommands.mobileDeleteFile = async function mobileDeleteFile (opts = {}) {\n  const {remotePath} = opts;\n  if (!remotePath) {\n    log.errorAndThrow(`The 'remotePath' argument is mandatory`);\n  }\n  if (remotePath.endsWith('/')) {\n    log.errorAndThrow(`It is expected that remote path points to a folder and not to a file. ` +\n                      `'${remotePath}' is given instead`);\n  }\n  return await deleteFileOrFolder(this.adb, remotePath);\n};\n\nexport { commands };\nexport default commands;\n"],"file":"lib/commands/file-actions.js","sourceRoot":"../../.."}
|
|
@@ -156,6 +156,13 @@ commands.setGeoLocation = async function setGeoLocation(location) {
|
|
|
156
156
|
}
|
|
157
157
|
};
|
|
158
158
|
|
|
159
|
+
commands.mobileRefreshGpsCache = async function mobileRefreshGpsCache(opts = {}) {
|
|
160
|
+
const {
|
|
161
|
+
timeoutMs
|
|
162
|
+
} = opts;
|
|
163
|
+
await this.adb.refreshGeoLocationCache(timeoutMs);
|
|
164
|
+
};
|
|
165
|
+
|
|
159
166
|
commands.getGeoLocation = async function getGeoLocation() {
|
|
160
167
|
const {
|
|
161
168
|
latitude,
|
|
@@ -262,4 +269,4 @@ var _default = extensions;
|
|
|
262
269
|
exports.default = _default;require('source-map-support').install();
|
|
263
270
|
|
|
264
271
|
|
|
265
|
-
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/network.js"],"names":["commands","helpers","extensions","AIRPLANE_MODE_MASK","WIFI_MASK","DATA_MASK","GEO_EPSILON","Number","MIN_VALUE","getNetworkConnection","log","info","airplaneModeOn","adb","isAirplaneModeOn","connection","wifiOn","isWifiOn","dataOn","isDataOn","setNetworkConnection","type","shouldEnableAirplaneMode","shouldEnableWifi","shouldEnableDataConnection","currentState","isAirplaneModeEnabled","isWiFiEnabled","isDataEnabled","wrapBootstrapDisconnect","setAirplaneMode","broadcastAirplaneMode","setWifiState","setDataState","isEmulator","wifi","toggleData","data","setWifiAndData","toggleWiFi","toggleFlightMode","flightMode","setGeoLocation","location","getGeoLocation","e","warn","message","latitude","longitude","altitude","parseFloat","KeyCode","UP","DOWN","RIGHT","CENTER","toggleLocationServices","api","getApiLevel","providers","getLocationProviders","isGpsEnabled","indexOf","toggleGPSLocationProvider","seq","push","keyevent","toggleSetting","errors","NotYetImplementedError","setting","preKeySeq","_","isNull","openSettingsActivity","key","doKey","appPackage","appActivity","getFocusedPackageAndActivity","waitForNotActivity","ign","back","B","delay","wrapped","bootstrap","ignoreUnexpectedShutdown","restart","start","opts","disableAndroidWatchers","acceptSslCerts","Object","assign"],"mappings":"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","sourcesContent":["import log from '../logger';\nimport _ from 'lodash';\nimport { errors } from 'appium-base-driver';\nimport B from 'bluebird';\n\nlet commands = {}, helpers = {}, extensions = {};\n\nconst AIRPLANE_MODE_MASK = 0b001;\nconst WIFI_MASK = 0b010;\nconst DATA_MASK = 0b100;\n// The value close to zero, but not zero, is needed\n// to trick JSON generation and send a float value instead of an integer,\n// This allows strictly-typed clients, like Java, to properly\n// parse it. Otherwise float 0.0 is always represented as integer 0 in JS.\n// The value must not be greater than DBL_EPSILON (https://opensource.apple.com/source/Libc/Libc-498/include/float.h)\nconst GEO_EPSILON = Number.MIN_VALUE;\n\ncommands.getNetworkConnection = async function getNetworkConnection () {\n  log.info('Getting network connection');\n  let airplaneModeOn = await this.adb.isAirplaneModeOn();\n  let connection = airplaneModeOn ? AIRPLANE_MODE_MASK : 0;\n\n  // no need to check anything else if we are in airplane mode\n  if (!airplaneModeOn) {\n    let wifiOn = await this.isWifiOn();\n    connection |= (wifiOn ? WIFI_MASK : 0);\n    let dataOn = await this.adb.isDataOn();\n    connection |= (dataOn ? DATA_MASK : 0);\n  }\n\n  return connection;\n};\n\n/**\n * decoupling to override the behaviour in other drivers like UiAutomator2.\n */\ncommands.isWifiOn = async function isWifiOn () {\n  return await this.adb.isWifiOn();\n};\n\ncommands.setNetworkConnection = async function setNetworkConnection (type) {\n  log.info('Setting network connection');\n  // decode the input\n  const shouldEnableAirplaneMode = (type & AIRPLANE_MODE_MASK) !== 0;\n  const shouldEnableWifi = (type & WIFI_MASK) !== 0;\n  const shouldEnableDataConnection = (type & DATA_MASK) !== 0;\n\n  const currentState = await this.getNetworkConnection();\n  const isAirplaneModeEnabled = (currentState & AIRPLANE_MODE_MASK) !== 0;\n  const isWiFiEnabled = (currentState & WIFI_MASK) !== 0;\n  const isDataEnabled = (currentState & DATA_MASK) !== 0;\n\n  if (shouldEnableAirplaneMode !== isAirplaneModeEnabled) {\n    await this.wrapBootstrapDisconnect(async () => {\n      await this.adb.setAirplaneMode(shouldEnableAirplaneMode);\n    });\n    await this.wrapBootstrapDisconnect(async () => {\n      await this.adb.broadcastAirplaneMode(shouldEnableAirplaneMode);\n    });\n  } else {\n    log.info(`Not changing airplane mode, since it is already ` +\n             `${shouldEnableAirplaneMode ? 'enabled' : 'disabled'}`);\n  }\n\n  if (shouldEnableWifi === isWiFiEnabled && shouldEnableDataConnection === isDataEnabled) {\n    log.info('Not changing data connection/Wi-Fi states, since they are already set to expected values');\n    if (await this.adb.isAirplaneModeOn()) {\n      return AIRPLANE_MODE_MASK | currentState;\n    }\n    return ~AIRPLANE_MODE_MASK & currentState;\n  }\n\n  await this.wrapBootstrapDisconnect(async () => {\n    if (shouldEnableWifi !== isWiFiEnabled) {\n      await this.setWifiState(shouldEnableWifi);\n    } else {\n      log.info(`Not changing Wi-Fi state, since it is already ` +\n               `${shouldEnableWifi ? 'enabled' : 'disabled'}`);\n    }\n\n    if (shouldEnableAirplaneMode) {\n      log.info('Not changing data connection state, because airplane mode is enabled');\n    } else if (shouldEnableDataConnection === isDataEnabled) {\n      log.info(`Not changing data connection state, since it is already ` +\n               `${shouldEnableDataConnection ? 'enabled' : 'disabled'}`);\n    } else {\n      await this.adb.setDataState(shouldEnableDataConnection, this.isEmulator());\n    }\n  });\n\n  return await this.getNetworkConnection();\n};\n\n/**\n * decoupling to override behaviour in other drivers like UiAutomator2.\n */\ncommands.setWifiState = async function setWifiState (wifi) {\n  await this.adb.setWifiState(wifi, this.isEmulator());\n};\n\ncommands.toggleData = async function toggleData () {\n  let data = !(await this.adb.isDataOn());\n  log.info(`Turning network data ${data ? 'on' : 'off'}`);\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.adb.setWifiAndData({data}, this.isEmulator());\n  });\n};\n\ncommands.toggleWiFi = async function toggleWiFi () {\n  let wifi = !(await this.adb.isWifiOn());\n  log.info(`Turning WiFi ${wifi ? 'on' : 'off'}`);\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.adb.setWifiAndData({wifi}, this.isEmulator());\n  });\n};\n\ncommands.toggleFlightMode = async function toggleFlightMode () {\n  /*\n   * TODO: Implement isRealDevice(). This method fails on\n   * real devices, it should throw a NotYetImplementedError\n   */\n  let flightMode = !(await this.adb.isAirplaneModeOn());\n  log.info(`Turning flight mode ${flightMode ? 'on' : 'off'}`);\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.adb.setAirplaneMode(flightMode);\n  });\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.adb.broadcastAirplaneMode(flightMode);\n  });\n};\n\ncommands.setGeoLocation = async function setGeoLocation (location) {\n  await this.adb.setGeoLocation(location, this.isEmulator());\n  try {\n    return await this.getGeoLocation();\n  } catch (e) {\n    log.warn(`Could not get the current geolocation info: ${e.message}`);\n    log.warn(`Returning the default zero'ed values`);\n    return {\n      latitude: GEO_EPSILON,\n      longitude: GEO_EPSILON,\n      altitude: GEO_EPSILON,\n    };\n  }\n};\n\ncommands.getGeoLocation = async function getGeoLocation () {\n  const {latitude, longitude, altitude} = await this.adb.getGeoLocation();\n  return {\n    latitude: parseFloat(latitude) || GEO_EPSILON,\n    longitude: parseFloat(longitude) || GEO_EPSILON,\n    altitude: parseFloat(altitude) || GEO_EPSILON,\n  };\n};\n// https://developer.android.com/reference/android/view/KeyEvent.html#KEYCODE_DPAD_CENTER\n// in the android docs, this is how the keycodes are defined\nconst KeyCode = {\n  UP: 19,\n  DOWN: 20,\n  RIGHT: 22,\n  CENTER: 23\n};\ncommands.toggleLocationServices = async function toggleLocationServices () {\n  log.info('Toggling location services');\n  let api = await this.adb.getApiLevel();\n  if (this.isEmulator()) {\n    let providers = await this.adb.getLocationProviders();\n    let isGpsEnabled = providers.indexOf('gps') !== -1;\n    await this.adb.toggleGPSLocationProvider(!isGpsEnabled);\n    return;\n  }\n\n  if (api > 15) {\n    let seq = [KeyCode.UP, KeyCode.UP];\n    if (api === 16) {\n      // This version of Android has a \"parent\" button in its action bar\n      seq.push(KeyCode.DOWN);\n    } else if (api < 28) {\n      // Newer versions of Android have the toggle in the Action bar\n      seq = [KeyCode.RIGHT, KeyCode.RIGHT, KeyCode.UP];\n      /*\n       * Once the Location services switch is OFF, it won't receive focus\n       * when going back to the Location Services settings screen unless we\n       * send a dummy keyevent (UP) *before* opening the settings screen\n       */\n      await this.adb.keyevent(KeyCode.UP);\n    } else if (api >= 28) {\n      // Even newer versions of android have the toggle in a bar below the action bar\n      // this means a single right click will cause it to be selected.\n      seq = [KeyCode.RIGHT];\n      await this.adb.keyevent(KeyCode.UP);\n    }\n    await this.toggleSetting('LOCATION_SOURCE_SETTINGS', seq);\n  } else {\n    // There's no global location services toggle on older Android versions\n    throw new errors.NotYetImplementedError();\n  }\n};\n\nhelpers.toggleSetting = async function toggleSetting (setting, preKeySeq) {\n  /*\n   * preKeySeq is the keyevent sequence to send over ADB in order\n   * to position the cursor on the right option.\n   * By default it's [up, up, down] because we usually target the 1st item in\n   * the screen, and sometimes when opening settings activities the cursor is\n   * already positionned on the 1st item, but we can't know for sure\n   */\n  if (_.isNull(preKeySeq)) {\n    preKeySeq = [KeyCode.UP, KeyCode.UP, KeyCode.DOWN];\n  }\n\n  await this.openSettingsActivity(setting);\n\n  for (let key of preKeySeq) {\n    await this.doKey(key);\n  }\n\n  let {appPackage, appActivity} = await this.adb.getFocusedPackageAndActivity();\n\n  /*\n   * Click and handle potential ADB disconnect that occurs on official\n   * emulator when the network connection is disabled\n   */\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.doKey(KeyCode.CENTER);\n  });\n\n  /*\n   * In one particular case (enable Location Services), a pop-up is\n   * displayed on some platforms so the user accepts or refuses that Google\n   * collects location data. So we wait for that pop-up to open, if it\n   * doesn't then proceed\n   */\n  try {\n    await this.adb.waitForNotActivity(appPackage, appActivity, 5000);\n    await this.doKey(KeyCode.RIGHT);\n    await this.doKey(KeyCode.CENTER);\n    await this.adb.waitForNotActivity(appPackage, appActivity, 5000);\n  } catch (ign) {}\n\n  await this.adb.back();\n};\n\nhelpers.doKey = async function doKey (key) {\n  // TODO: Confirm we need this delay. Seems to work without it.\n  await B.delay(2000);\n  await this.adb.keyevent(key);\n};\n\nhelpers.wrapBootstrapDisconnect = async function wrapBootstrapDisconnect (wrapped) {\n  if (!this.bootstrap) {\n    return await wrapped();\n  }\n\n  this.bootstrap.ignoreUnexpectedShutdown = true;\n  try {\n    await wrapped();\n    await this.adb.restart();\n    await this.bootstrap.start(this.opts.appPackage, this.opts.disableAndroidWatchers, this.opts.acceptSslCerts);\n  } finally {\n    this.bootstrap.ignoreUnexpectedShutdown = false;\n  }\n};\n\nObject.assign(extensions, commands, helpers);\nexport { commands, helpers };\nexport default extensions;\n"],"file":"lib/commands/network.js","sourceRoot":"../../.."}
|
|
272
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/network.js"],"names":["commands","helpers","extensions","AIRPLANE_MODE_MASK","WIFI_MASK","DATA_MASK","GEO_EPSILON","Number","MIN_VALUE","getNetworkConnection","log","info","airplaneModeOn","adb","isAirplaneModeOn","connection","wifiOn","isWifiOn","dataOn","isDataOn","setNetworkConnection","type","shouldEnableAirplaneMode","shouldEnableWifi","shouldEnableDataConnection","currentState","isAirplaneModeEnabled","isWiFiEnabled","isDataEnabled","wrapBootstrapDisconnect","setAirplaneMode","broadcastAirplaneMode","setWifiState","setDataState","isEmulator","wifi","toggleData","data","setWifiAndData","toggleWiFi","toggleFlightMode","flightMode","setGeoLocation","location","getGeoLocation","e","warn","message","latitude","longitude","altitude","mobileRefreshGpsCache","opts","timeoutMs","refreshGeoLocationCache","parseFloat","KeyCode","UP","DOWN","RIGHT","CENTER","toggleLocationServices","api","getApiLevel","providers","getLocationProviders","isGpsEnabled","indexOf","toggleGPSLocationProvider","seq","push","keyevent","toggleSetting","errors","NotYetImplementedError","setting","preKeySeq","_","isNull","openSettingsActivity","key","doKey","appPackage","appActivity","getFocusedPackageAndActivity","waitForNotActivity","ign","back","B","delay","wrapped","bootstrap","ignoreUnexpectedShutdown","restart","start","disableAndroidWatchers","acceptSslCerts","Object","assign"],"mappings":"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","sourcesContent":["import log from '../logger';\nimport _ from 'lodash';\nimport { errors } from 'appium-base-driver';\nimport B from 'bluebird';\n\nlet commands = {}, helpers = {}, extensions = {};\n\nconst AIRPLANE_MODE_MASK = 0b001;\nconst WIFI_MASK = 0b010;\nconst DATA_MASK = 0b100;\n// The value close to zero, but not zero, is needed\n// to trick JSON generation and send a float value instead of an integer,\n// This allows strictly-typed clients, like Java, to properly\n// parse it. Otherwise float 0.0 is always represented as integer 0 in JS.\n// The value must not be greater than DBL_EPSILON (https://opensource.apple.com/source/Libc/Libc-498/include/float.h)\nconst GEO_EPSILON = Number.MIN_VALUE;\n\ncommands.getNetworkConnection = async function getNetworkConnection () {\n  log.info('Getting network connection');\n  let airplaneModeOn = await this.adb.isAirplaneModeOn();\n  let connection = airplaneModeOn ? AIRPLANE_MODE_MASK : 0;\n\n  // no need to check anything else if we are in airplane mode\n  if (!airplaneModeOn) {\n    let wifiOn = await this.isWifiOn();\n    connection |= (wifiOn ? WIFI_MASK : 0);\n    let dataOn = await this.adb.isDataOn();\n    connection |= (dataOn ? DATA_MASK : 0);\n  }\n\n  return connection;\n};\n\n/**\n * decoupling to override the behaviour in other drivers like UiAutomator2.\n */\ncommands.isWifiOn = async function isWifiOn () {\n  return await this.adb.isWifiOn();\n};\n\ncommands.setNetworkConnection = async function setNetworkConnection (type) {\n  log.info('Setting network connection');\n  // decode the input\n  const shouldEnableAirplaneMode = (type & AIRPLANE_MODE_MASK) !== 0;\n  const shouldEnableWifi = (type & WIFI_MASK) !== 0;\n  const shouldEnableDataConnection = (type & DATA_MASK) !== 0;\n\n  const currentState = await this.getNetworkConnection();\n  const isAirplaneModeEnabled = (currentState & AIRPLANE_MODE_MASK) !== 0;\n  const isWiFiEnabled = (currentState & WIFI_MASK) !== 0;\n  const isDataEnabled = (currentState & DATA_MASK) !== 0;\n\n  if (shouldEnableAirplaneMode !== isAirplaneModeEnabled) {\n    await this.wrapBootstrapDisconnect(async () => {\n      await this.adb.setAirplaneMode(shouldEnableAirplaneMode);\n    });\n    await this.wrapBootstrapDisconnect(async () => {\n      await this.adb.broadcastAirplaneMode(shouldEnableAirplaneMode);\n    });\n  } else {\n    log.info(`Not changing airplane mode, since it is already ` +\n             `${shouldEnableAirplaneMode ? 'enabled' : 'disabled'}`);\n  }\n\n  if (shouldEnableWifi === isWiFiEnabled && shouldEnableDataConnection === isDataEnabled) {\n    log.info('Not changing data connection/Wi-Fi states, since they are already set to expected values');\n    if (await this.adb.isAirplaneModeOn()) {\n      return AIRPLANE_MODE_MASK | currentState;\n    }\n    return ~AIRPLANE_MODE_MASK & currentState;\n  }\n\n  await this.wrapBootstrapDisconnect(async () => {\n    if (shouldEnableWifi !== isWiFiEnabled) {\n      await this.setWifiState(shouldEnableWifi);\n    } else {\n      log.info(`Not changing Wi-Fi state, since it is already ` +\n               `${shouldEnableWifi ? 'enabled' : 'disabled'}`);\n    }\n\n    if (shouldEnableAirplaneMode) {\n      log.info('Not changing data connection state, because airplane mode is enabled');\n    } else if (shouldEnableDataConnection === isDataEnabled) {\n      log.info(`Not changing data connection state, since it is already ` +\n               `${shouldEnableDataConnection ? 'enabled' : 'disabled'}`);\n    } else {\n      await this.adb.setDataState(shouldEnableDataConnection, this.isEmulator());\n    }\n  });\n\n  return await this.getNetworkConnection();\n};\n\n/**\n * decoupling to override behaviour in other drivers like UiAutomator2.\n */\ncommands.setWifiState = async function setWifiState (wifi) {\n  await this.adb.setWifiState(wifi, this.isEmulator());\n};\n\ncommands.toggleData = async function toggleData () {\n  let data = !(await this.adb.isDataOn());\n  log.info(`Turning network data ${data ? 'on' : 'off'}`);\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.adb.setWifiAndData({data}, this.isEmulator());\n  });\n};\n\ncommands.toggleWiFi = async function toggleWiFi () {\n  let wifi = !(await this.adb.isWifiOn());\n  log.info(`Turning WiFi ${wifi ? 'on' : 'off'}`);\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.adb.setWifiAndData({wifi}, this.isEmulator());\n  });\n};\n\ncommands.toggleFlightMode = async function toggleFlightMode () {\n  /*\n   * TODO: Implement isRealDevice(). This method fails on\n   * real devices, it should throw a NotYetImplementedError\n   */\n  let flightMode = !(await this.adb.isAirplaneModeOn());\n  log.info(`Turning flight mode ${flightMode ? 'on' : 'off'}`);\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.adb.setAirplaneMode(flightMode);\n  });\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.adb.broadcastAirplaneMode(flightMode);\n  });\n};\n\ncommands.setGeoLocation = async function setGeoLocation (location) {\n  await this.adb.setGeoLocation(location, this.isEmulator());\n  try {\n    return await this.getGeoLocation();\n  } catch (e) {\n    log.warn(`Could not get the current geolocation info: ${e.message}`);\n    log.warn(`Returning the default zero'ed values`);\n    return {\n      latitude: GEO_EPSILON,\n      longitude: GEO_EPSILON,\n      altitude: GEO_EPSILON,\n    };\n  }\n};\n\n/**\n * @typedef {Object} GpsCacheRefreshOptions\n * @property {number} timeoutMs [20000] The maximum number of milliseconds\n * to block until GPS cache is refreshed. Providing zero or a negative\n * value to it skips waiting completely.\n */\n\n/**\n * Sends an async request to refresh the GPS cache.\n * This feature only works if the device under test has\n * Google Play Services installed. In case the vanilla\n * LocationManager is used the device API level must be at\n * version 30 (Android R) or higher.\n *\n * @param {GpsCacheRefreshOptions} opts\n */\ncommands.mobileRefreshGpsCache = async function mobileRefreshGpsCache (opts = {}) {\n  const { timeoutMs } = opts;\n  await this.adb.refreshGeoLocationCache(timeoutMs);\n};\n\ncommands.getGeoLocation = async function getGeoLocation () {\n  const {latitude, longitude, altitude} = await this.adb.getGeoLocation();\n  return {\n    latitude: parseFloat(latitude) || GEO_EPSILON,\n    longitude: parseFloat(longitude) || GEO_EPSILON,\n    altitude: parseFloat(altitude) || GEO_EPSILON,\n  };\n};\n// https://developer.android.com/reference/android/view/KeyEvent.html#KEYCODE_DPAD_CENTER\n// in the android docs, this is how the keycodes are defined\nconst KeyCode = {\n  UP: 19,\n  DOWN: 20,\n  RIGHT: 22,\n  CENTER: 23\n};\ncommands.toggleLocationServices = async function toggleLocationServices () {\n  log.info('Toggling location services');\n  let api = await this.adb.getApiLevel();\n  if (this.isEmulator()) {\n    let providers = await this.adb.getLocationProviders();\n    let isGpsEnabled = providers.indexOf('gps') !== -1;\n    await this.adb.toggleGPSLocationProvider(!isGpsEnabled);\n    return;\n  }\n\n  if (api > 15) {\n    let seq = [KeyCode.UP, KeyCode.UP];\n    if (api === 16) {\n      // This version of Android has a \"parent\" button in its action bar\n      seq.push(KeyCode.DOWN);\n    } else if (api < 28) {\n      // Newer versions of Android have the toggle in the Action bar\n      seq = [KeyCode.RIGHT, KeyCode.RIGHT, KeyCode.UP];\n      /*\n       * Once the Location services switch is OFF, it won't receive focus\n       * when going back to the Location Services settings screen unless we\n       * send a dummy keyevent (UP) *before* opening the settings screen\n       */\n      await this.adb.keyevent(KeyCode.UP);\n    } else if (api >= 28) {\n      // Even newer versions of android have the toggle in a bar below the action bar\n      // this means a single right click will cause it to be selected.\n      seq = [KeyCode.RIGHT];\n      await this.adb.keyevent(KeyCode.UP);\n    }\n    await this.toggleSetting('LOCATION_SOURCE_SETTINGS', seq);\n  } else {\n    // There's no global location services toggle on older Android versions\n    throw new errors.NotYetImplementedError();\n  }\n};\n\nhelpers.toggleSetting = async function toggleSetting (setting, preKeySeq) {\n  /*\n   * preKeySeq is the keyevent sequence to send over ADB in order\n   * to position the cursor on the right option.\n   * By default it's [up, up, down] because we usually target the 1st item in\n   * the screen, and sometimes when opening settings activities the cursor is\n   * already positionned on the 1st item, but we can't know for sure\n   */\n  if (_.isNull(preKeySeq)) {\n    preKeySeq = [KeyCode.UP, KeyCode.UP, KeyCode.DOWN];\n  }\n\n  await this.openSettingsActivity(setting);\n\n  for (let key of preKeySeq) {\n    await this.doKey(key);\n  }\n\n  let {appPackage, appActivity} = await this.adb.getFocusedPackageAndActivity();\n\n  /*\n   * Click and handle potential ADB disconnect that occurs on official\n   * emulator when the network connection is disabled\n   */\n  await this.wrapBootstrapDisconnect(async () => {\n    await this.doKey(KeyCode.CENTER);\n  });\n\n  /*\n   * In one particular case (enable Location Services), a pop-up is\n   * displayed on some platforms so the user accepts or refuses that Google\n   * collects location data. So we wait for that pop-up to open, if it\n   * doesn't then proceed\n   */\n  try {\n    await this.adb.waitForNotActivity(appPackage, appActivity, 5000);\n    await this.doKey(KeyCode.RIGHT);\n    await this.doKey(KeyCode.CENTER);\n    await this.adb.waitForNotActivity(appPackage, appActivity, 5000);\n  } catch (ign) {}\n\n  await this.adb.back();\n};\n\nhelpers.doKey = async function doKey (key) {\n  // TODO: Confirm we need this delay. Seems to work without it.\n  await B.delay(2000);\n  await this.adb.keyevent(key);\n};\n\nhelpers.wrapBootstrapDisconnect = async function wrapBootstrapDisconnect (wrapped) {\n  if (!this.bootstrap) {\n    return await wrapped();\n  }\n\n  this.bootstrap.ignoreUnexpectedShutdown = true;\n  try {\n    await wrapped();\n    await this.adb.restart();\n    await this.bootstrap.start(this.opts.appPackage, this.opts.disableAndroidWatchers, this.opts.acceptSslCerts);\n  } finally {\n    this.bootstrap.ignoreUnexpectedShutdown = false;\n  }\n};\n\nObject.assign(extensions, commands, helpers);\nexport { commands, helpers };\nexport default extensions;\n"],"file":"lib/commands/network.js","sourceRoot":"../../.."}
|