appium 2.0.0-beta.22 → 2.0.0-beta.23

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.
@@ -179,6 +179,9 @@ class AppiumDriver extends _baseDriver.BaseDriver {
179
179
 
180
180
  this.assignCliArgsToExtension('driver', driverName, driverInstance);
181
181
  driverInstance.server = this.server;
182
+ driverInstance.serverHost = this.args.address;
183
+ driverInstance.serverPort = this.args.port;
184
+ driverInstance.serverPath = this.args.basePath;
182
185
 
183
186
  try {
184
187
  runningDriversData = await this.curSessionDataForDriver(InnerDriver);
@@ -560,4 +563,4 @@ class NoDriverProxyCommandError extends Error {
560
563
  exports.NoDriverProxyCommandError = NoDriverProxyCommandError;require('source-map-support').install();
561
564
 
562
565
 
563
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/appium.js"],"names":["desiredCapabilityConstraints","automationName","presence","isString","platformName","sessionsListGuard","AsyncLock","pendingDriversGuard","AppiumDriver","BaseDriver","constructor","args","tmpDir","process","env","APPIUM_TMP_DIR","desiredCapConstraints","newCommandTimeoutMs","sessions","pendingDrivers","pluginClasses","sessionPlugins","sessionlessPlugins","isCommandsQueueEnabled","sessionExists","sessionId","dstSession","driverForSession","getStatus","build","_","clone","getSessions","acquire","name","toPairs","map","id","driver","capabilities","caps","printNewSessionAnnouncement","driverName","driverVersion","driverBaseVersion","log","info","APPIUM_VER","baseVersion","_findMatchingDriver","assignCliArgsToExtension","extType","extName","extInstance","cliArgs","isEmpty","createSession","jsonwpCaps","reqCaps","w3cCapabilities","defaultCapabilities","cloneDeep","defaultSettings","jwpSettings","Object","assign","w3cSettings","alwaysMatch","firstMatchEntry","firstMatch","protocol","innerSessionId","dCaps","parsedCaps","desiredCaps","processedJsonwpCapabilities","processedW3CCapabilities","error","InnerDriver","version","driverConfig","sessionOverride","deleteAllSessions","runningDriversData","otherPendingDriversData","driverInstance","relaxedSecurityEnabled","denyInsecure","a","allowInsecure","server","curSessionDataForDriver","e","errors","SessionNotCreatedError","message","drv","driverData","push","pull","attachUnexpectedShutdownHandler","startNewCommandTimeout","isW3CProtocol","JSON","stringify","updateSettings","isMjsonwpProtocol","value","onShutdown","cause","Error","warn","plugin","isFunction","onUnexpectedShutdown","debug","data","values","filter","s","datum","deleteSession","otherSessionsData","curConstructorName","key","opts","sessionsCount","size","force","reason","util","pluralize","cleanupPromises","startUnexpectedShutdown","keys","cleanupPromise","pluginsForSession","createPluginInstances","pluginsToHandleCmd","cmd","p","handle","PluginClass","pluginName","executeCommand","isGetStatus","isUmbrellaCmd","isAppiumDriverCommand","isSessionCmd","reqForProxy","last","pop","plugins","cmdHandledBy","default","defaultBehavior","length","proxyCommand","NoDriverProxyCommandError","originalUrl","method","body","wrappedCmd","wrapCommandWithPlugins","next","res","executeWrappedCommand","logPluginHandlerReport","CREATE_SESSION_COMMAND","first","_next","didHandle","k","didntHandle","cmdRes","cmdErr","isPlainObject","has","proxyActive","getProxyAvoidList","canProxy"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport log from './logger';\nimport { getBuildInfo, updateBuildInfo, APPIUM_VER } from './config';\nimport { findMatchingDriver } from './drivers';\nimport { BaseDriver, errors, isSessionCommand,\n         CREATE_SESSION_COMMAND } from '@appium/base-driver';\nimport AsyncLock from 'async-lock';\nimport { parseCapsForInnerDriver, pullSettings } from './utils';\nimport { util } from '@appium/support';\n\nconst desiredCapabilityConstraints = {\n  automationName: {\n    presence: true,\n    isString: true,\n  },\n  platformName: {\n    presence: true,\n    isString: true,\n  },\n};\n\nconst sessionsListGuard = new AsyncLock();\nconst pendingDriversGuard = new AsyncLock();\n\nclass AppiumDriver extends BaseDriver {\n  constructor (args) {\n    // It is necessary to set `--tmp` here since it should be set to\n    // process.env.APPIUM_TMP_DIR once at an initial point in the Appium lifecycle.\n    // The process argument will be referenced by BaseDriver.\n    // Please call @appium/support.tempDir module to apply this benefit.\n    if (args.tmpDir) {\n      process.env.APPIUM_TMP_DIR = args.tmpDir;\n    }\n\n    super(args);\n\n    this.desiredCapConstraints = desiredCapabilityConstraints;\n\n    // the main Appium Driver has no new command timeout\n    this.newCommandTimeoutMs = 0;\n\n    this.args = {...args};\n\n    // Access to sessions list must be guarded with a Semaphore, because\n    // it might be changed by other async calls at any time\n    // It is not recommended to access this property directly from the outside\n    this.sessions = {};\n\n    // Access to pending drivers list must be guarded with a Semaphore, because\n    // it might be changed by other async calls at any time\n    // It is not recommended to access this property directly from the outside\n    this.pendingDrivers = {};\n\n    this.pluginClasses = []; // list of which plugins are active\n    this.sessionPlugins = {}; // map of sessions to actual plugin instances per session\n    this.sessionlessPlugins = []; // some commands are sessionless, so we need a set of plugins for them\n\n    // allow this to happen in the background, so no `await`\n    updateBuildInfo();\n  }\n\n  /** @type {DriverConfig|undefined} */\n  driverConfig;\n\n  /**\n   * Cancel commands queueing for the umbrella Appium driver\n   */\n  get isCommandsQueueEnabled () {\n    return false;\n  }\n\n  sessionExists (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && dstSession.sessionId !== null;\n  }\n\n  driverForSession (sessionId) {\n    return this.sessions[sessionId];\n  }\n\n  async getStatus () { // eslint-disable-line require-await\n    return {\n      build: _.clone(getBuildInfo()),\n    };\n  }\n\n  async getSessions () {\n    const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);\n    return _.toPairs(sessions)\n      .map(([id, driver]) => ({id, capabilities: driver.caps}));\n  }\n\n  printNewSessionAnnouncement (driverName, driverVersion, driverBaseVersion) {\n    log.info(driverVersion\n      ? `Appium v${APPIUM_VER} creating new ${driverName} (v${driverVersion}) session`\n      : `Appium v${APPIUM_VER} creating new ${driverName} session`\n    );\n    log.info(`Checking BaseDriver versions for Appium and ${driverName}`);\n    log.info(AppiumDriver.baseVersion\n      ? `Appium's BaseDriver version is ${AppiumDriver.baseVersion}`\n      : `Could not determine Appium's BaseDriver version`\n    );\n    log.info(driverBaseVersion\n      ? `${driverName}'s BaseDriver version is ${driverBaseVersion}`\n      : `Could not determine ${driverName}'s BaseDriver version`\n    );\n  }\n\n  /**\n   * This is just an alias for driver.js's method, which is necessary for\n   * mocking in the test suite\n   */\n  _findMatchingDriver (...args) {\n    return findMatchingDriver(...args);\n  }\n\n  /**\n   * Validate and assign CLI args for a driver or plugin\n   *\n   * If the extension has provided a schema, validation has already happened.\n   * @param {import('./ext-config-io').ExtensionType} extType 'driver' or 'plugin'\n   * @param {string} extName the name of the extension\n   * @param {Object} extInstance the driver or plugin instance\n   */\n  assignCliArgsToExtension (extType, extName, extInstance) {\n    const cliArgs = this.args[extType]?.[extName];\n    if (!_.isEmpty(cliArgs)) {\n      extInstance.cliArgs = cliArgs;\n    }\n  }\n\n  /**\n   * Create a new session\n   * @param {Object} jsonwpCaps JSONWP formatted desired capabilities\n   * @param {Object} reqCaps Required capabilities (JSONWP standard)\n   * @param {Object} w3cCapabilities W3C capabilities\n   * @return {Array} Unique session ID and capabilities\n   */\n  async createSession (jsonwpCaps, reqCaps, w3cCapabilities) {\n    const defaultCapabilities = _.cloneDeep(this.args.defaultCapabilities);\n    const defaultSettings = pullSettings(defaultCapabilities);\n    jsonwpCaps = _.cloneDeep(jsonwpCaps);\n    const jwpSettings = Object.assign({}, defaultSettings, pullSettings(jsonwpCaps));\n    w3cCapabilities = _.cloneDeep(w3cCapabilities);\n    // It is possible that the client only provides caps using JSONWP standard,\n    // although firstMatch/alwaysMatch properties are still present.\n    // In such case we assume the client understands W3C protocol and merge the given\n    // JSONWP caps to W3C caps\n    const w3cSettings = Object.assign({}, jwpSettings);\n    Object.assign(w3cSettings, pullSettings((w3cCapabilities || {}).alwaysMatch || {}));\n    for (const firstMatchEntry of ((w3cCapabilities || {}).firstMatch || [])) {\n      Object.assign(w3cSettings, pullSettings(firstMatchEntry));\n    }\n\n    let protocol;\n    let innerSessionId, dCaps;\n    try {\n      // Parse the caps into a format that the InnerDriver will accept\n      const parsedCaps = parseCapsForInnerDriver(\n        jsonwpCaps,\n        w3cCapabilities,\n        this.desiredCapConstraints,\n        defaultCapabilities\n      );\n\n      const {desiredCaps, processedJsonwpCapabilities, processedW3CCapabilities, error} = parsedCaps;\n      protocol = parsedCaps.protocol;\n\n      // If the parsing of the caps produced an error, throw it in here\n      if (error) {\n        throw error;\n      }\n\n      const {\n        driver: InnerDriver,\n        version: driverVersion,\n        driverName\n      } = this._findMatchingDriver(this.driverConfig, desiredCaps);\n      this.printNewSessionAnnouncement(InnerDriver.name, driverVersion, InnerDriver.baseVersion);\n\n      if (this.args.sessionOverride) {\n        await this.deleteAllSessions();\n      }\n\n      let runningDriversData, otherPendingDriversData;\n\n      const driverInstance = new InnerDriver(this.args, true);\n\n      // We want to assign security values directly on the driver. The driver\n      // should not read security values from `this.opts` because those values\n      // could have been set by a malicious user via capabilities, whereas we\n      // want a guarantee the values were set by the appium server admin\n      if (this.args.relaxedSecurityEnabled) {\n        log.info(`Applying relaxed security to '${InnerDriver.name}' as per ` +\n                 `server command line argument. All insecure features will be ` +\n                 `enabled unless explicitly disabled by --deny-insecure`);\n        driverInstance.relaxedSecurityEnabled = true;\n      }\n\n      if (!_.isEmpty(this.args.denyInsecure)) {\n        log.info('Explicitly preventing use of insecure features:');\n        this.args.denyInsecure.map((a) => log.info(`    ${a}`));\n        driverInstance.denyInsecure = this.args.denyInsecure;\n      }\n\n      if (!_.isEmpty(this.args.allowInsecure)) {\n        log.info('Explicitly enabling use of insecure features:');\n        this.args.allowInsecure.map((a) => log.info(`    ${a}`));\n        driverInstance.allowInsecure = this.args.allowInsecure;\n      }\n\n      // Likewise, any driver-specific CLI args that were passed in should be assigned directly to\n      // the driver so that they cannot be mimicked by a malicious user sending in capabilities\n      this.assignCliArgsToExtension('driver', driverName, driverInstance);\n\n\n      // This assignment is required for correct web sockets functionality inside the driver\n      driverInstance.server = this.server;\n      try {\n        runningDriversData = await this.curSessionDataForDriver(InnerDriver);\n      } catch (e) {\n        throw new errors.SessionNotCreatedError(e.message);\n      }\n      await pendingDriversGuard.acquire(AppiumDriver.name, () => {\n        this.pendingDrivers[InnerDriver.name] = this.pendingDrivers[InnerDriver.name] || [];\n        otherPendingDriversData = this.pendingDrivers[InnerDriver.name].map((drv) => drv.driverData);\n        this.pendingDrivers[InnerDriver.name].push(driverInstance);\n      });\n\n      try {\n        [innerSessionId, dCaps] = await driverInstance.createSession(\n          processedJsonwpCapabilities,\n          reqCaps,\n          processedW3CCapabilities,\n          [...runningDriversData, ...otherPendingDriversData]\n        );\n        protocol = driverInstance.protocol;\n        await sessionsListGuard.acquire(AppiumDriver.name, () => {\n          this.sessions[innerSessionId] = driverInstance;\n        });\n      } finally {\n        await pendingDriversGuard.acquire(AppiumDriver.name, () => {\n          _.pull(this.pendingDrivers[InnerDriver.name], driverInstance);\n        });\n      }\n\n      this.attachUnexpectedShutdownHandler(driverInstance, innerSessionId);\n\n      log.info(`New ${InnerDriver.name} session created successfully, session ` +\n              `${innerSessionId} added to master session list`);\n\n      // set the New Command Timeout for the inner driver\n      driverInstance.startNewCommandTimeout();\n\n      // apply initial values to Appium settings (if provided)\n      if (driverInstance.isW3CProtocol() && !_.isEmpty(w3cSettings)) {\n        log.info(`Applying the initial values to Appium settings parsed from W3C caps: ` +\n          JSON.stringify(w3cSettings));\n        await driverInstance.updateSettings(w3cSettings);\n      } else if (driverInstance.isMjsonwpProtocol() && !_.isEmpty(jwpSettings)) {\n        log.info(`Applying the initial values to Appium settings parsed from MJSONWP caps: ` +\n          JSON.stringify(jwpSettings));\n        await driverInstance.updateSettings(jwpSettings);\n      }\n    } catch (error) {\n      return {\n        protocol,\n        error,\n      };\n    }\n\n    return {\n      protocol,\n      value: [innerSessionId, dCaps, protocol]\n    };\n  }\n\n  attachUnexpectedShutdownHandler (driver, innerSessionId) {\n    const onShutdown = (cause = new Error('Unknown error')) => {\n      log.warn(`Ending session, cause was '${cause.message}'`);\n\n      if (this.sessionPlugins[innerSessionId]) {\n        for (const plugin of this.sessionPlugins[innerSessionId]) {\n          if (_.isFunction(plugin.onUnexpectedShutdown)) {\n            log.debug(`Plugin ${plugin.name} defines an unexpected shutdown handler; calling it now`);\n            try {\n              plugin.onUnexpectedShutdown(driver, cause);\n            } catch (e) {\n              log.warn(`Got an error when running plugin ${plugin.name} shutdown handler: ${e}`);\n            }\n          } else {\n            log.debug(`Plugin ${plugin.name} does not define an unexpected shutdown handler`);\n          }\n        }\n      }\n\n      log.info(`Removing session '${innerSessionId}' from our master session list`);\n      delete this.sessions[innerSessionId];\n      delete this.sessionPlugins[innerSessionId];\n    };\n\n    if (_.isFunction(driver.onUnexpectedShutdown)) {\n      driver.onUnexpectedShutdown(onShutdown);\n    } else {\n      log.warn(`Failed to attach the unexpected shutdown listener. ` +\n        `Is 'onUnexpectedShutdown' method available for '${driver.constructor.name}'?`);\n    }\n  }\n\n  async curSessionDataForDriver (InnerDriver) {\n    const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);\n    const data = _.values(sessions)\n                   .filter((s) => s.constructor.name === InnerDriver.name)\n                   .map((s) => s.driverData);\n    for (let datum of data) {\n      if (!datum) {\n        throw new Error(`Problem getting session data for driver type ` +\n                        `${InnerDriver.name}; does it implement 'get ` +\n                        `driverData'?`);\n      }\n    }\n    return data;\n  }\n\n  async deleteSession (sessionId) {\n    let protocol;\n    try {\n      let otherSessionsData = null;\n      let dstSession = null;\n      await sessionsListGuard.acquire(AppiumDriver.name, () => {\n        if (!this.sessions[sessionId]) {\n          return;\n        }\n        const curConstructorName = this.sessions[sessionId].constructor.name;\n        otherSessionsData = _.toPairs(this.sessions)\n              .filter(([key, value]) => value.constructor.name === curConstructorName && key !== sessionId)\n              .map(([, value]) => value.driverData);\n        dstSession = this.sessions[sessionId];\n        protocol = dstSession.protocol;\n        log.info(`Removing session ${sessionId} from our master session list`);\n        // regardless of whether the deleteSession completes successfully or not\n        // make the session unavailable, because who knows what state it might\n        // be in otherwise\n        delete this.sessions[sessionId];\n        delete this.sessionPlugins[sessionId];\n      });\n      return {\n        protocol,\n        value: await dstSession.deleteSession(sessionId, otherSessionsData),\n      };\n    } catch (e) {\n      log.error(`Had trouble ending session ${sessionId}: ${e.message}`);\n      return {\n        protocol,\n        error: e,\n      };\n    }\n  }\n\n  async deleteAllSessions (opts = {}) {\n    const sessionsCount = _.size(this.sessions);\n    if (0 === sessionsCount) {\n      log.debug('There are no active sessions for cleanup');\n      return;\n    }\n\n    const {\n      force = false,\n      reason,\n    } = opts;\n    log.debug(`Cleaning up ${util.pluralize('active session', sessionsCount, true)}`);\n    const cleanupPromises = force\n      ? _.values(this.sessions).map((drv) => drv.startUnexpectedShutdown(reason && new Error(reason)))\n      : _.keys(this.sessions).map((id) => this.deleteSession(id));\n    for (const cleanupPromise of cleanupPromises) {\n      try {\n        await cleanupPromise;\n      } catch (e) {\n        log.debug(e);\n      }\n    }\n  }\n\n  /**\n   * Get the appropriate plugins for a session (or sessionless plugins)\n   *\n   * @param {?string} sessionId - the sessionId (or null) to use to find plugins\n   * @returns {Array} - array of plugin instances\n   */\n  pluginsForSession (sessionId = null) {\n    if (sessionId) {\n      if (!this.sessionPlugins[sessionId]) {\n        this.sessionPlugins[sessionId] = this.createPluginInstances();\n      }\n      return this.sessionPlugins[sessionId];\n    }\n\n    if (_.isEmpty(this.sessionlessPlugins)) {\n      this.sessionlessPlugins = this.createPluginInstances();\n    }\n    return this.sessionlessPlugins;\n  }\n\n  /**\n   * To get plugins for a command, we either get the plugin instances associated with the\n   * particular command's session, or in the case of sessionless plugins, pull from the set of\n   * plugin instances reserved for sessionless commands (and we lazily create plugin instances on\n   * first use)\n   *\n   * @param {string} cmd - the name of the command to find a plugin to handle\n   * @param {?string} sessionId - the particular session for which to find a plugin, or null if\n   * sessionless\n   */\n  pluginsToHandleCmd (cmd, sessionId = null) {\n    // to handle a given command, a plugin should either implement that command as a plugin\n    // instance method or it should implement a generic 'handle' method\n    return this.pluginsForSession(sessionId)\n      .filter((p) => _.isFunction(p[cmd]) || _.isFunction(p.handle));\n  }\n\n  createPluginInstances () {\n    return this.pluginClasses.map((PluginClass) => {\n      const name = PluginClass.pluginName;\n      const plugin = new PluginClass(name);\n      this.assignCliArgsToExtension('plugin', name, plugin);\n      return plugin;\n    });\n  }\n\n  async executeCommand (cmd, ...args) {\n    // We have basically three cases for how to handle commands:\n    // 1. handle getStatus (we do this as a special out of band case so it doesn't get added to an\n    //    execution queue, and can be called while e.g. createSession is in progress)\n    // 2. handle commands that this umbrella driver should handle, rather than the actual session\n    //    driver (for example, deleteSession, or other non-session commands)\n    // 3. handle session driver commands.\n    // The tricky part is that because we support command plugins, we need to wrap any of these\n    // cases with plugin handling.\n\n    const isGetStatus = cmd === 'getStatus';\n    const isUmbrellaCmd = !isGetStatus && isAppiumDriverCommand(cmd);\n    const isSessionCmd = !isGetStatus && !isUmbrellaCmd;\n\n    // if a plugin override proxying for this command and that is why we are here instead of just\n    // letting the protocol proxy the command entirely, determine that, get the request object for\n    // use later on, then clean up the args\n    const reqForProxy = _.last(args)?.reqForProxy;\n    if (reqForProxy) {\n      args.pop();\n    }\n\n\n    // first do some error checking. If we're requesting a session command execution, then make\n    // sure that session actually exists on the session driver, and set the session driver itself\n    let sessionId = null;\n    let dstSession = null;\n    let protocol = null;\n    let driver = this;\n    if (isSessionCmd) {\n      sessionId = _.last(args);\n      dstSession = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions[sessionId]);\n      if (!dstSession) {\n        throw new Error(`The session with id '${sessionId}' does not exist`);\n      }\n      // now save the response protocol given that the session driver's protocol might differ\n      protocol = dstSession.protocol;\n      driver = dstSession;\n    }\n\n    // get any plugins which are registered as handling this command\n    const plugins = this.pluginsToHandleCmd(cmd, sessionId);\n\n    // now we define a 'cmdHandledBy' object which will keep track of which plugins have handled this\n    // command. we care about this because (a) multiple plugins can handle the same command, and\n    // (b) there's no guarantee that a plugin will actually call the next() method which runs the\n    // original command execution. This results in a situation where the command might be handled\n    // by some but not all plugins, or by plugin(s) but not by the default behavior. So start out\n    // this object declaring that the default handler has not been executed.\n    const cmdHandledBy = {default: false};\n\n    // now we define an async function which will be passed to plugins, and successively wrapped\n    // if there is more than one plugin that can handle the command. To start off with, the async\n    // function is defined as calling the default behavior, i.e., whichever of the 3 cases above is\n    // the appropriate one\n    const defaultBehavior = async () => {\n      // if we're running with plugins, make sure we log that the default behavior is actually\n      // happening so we can tell when the plugin call chain is unwrapping to the default behavior\n      // if that's what happens\n      plugins.length && log.info(`Executing default handling behavior for command '${cmd}'`);\n\n      // if we make it here, we know that the default behavior is handled\n      cmdHandledBy.default = true;\n\n      if (reqForProxy) {\n        // we would have proxied this command had a plugin not handled it, so the default behavior\n        // is to do the proxy and retrieve the result internally so it can be passed to the plugin\n        // in case it calls 'await next()'. This requires that the driver have defined\n        // 'proxyCommand' and not just 'proxyReqRes'.\n        if (!dstSession.proxyCommand) {\n          throw new NoDriverProxyCommandError();\n        }\n        return await dstSession.proxyCommand(reqForProxy.originalUrl, reqForProxy.method,\n          reqForProxy.body);\n      }\n\n      if (isGetStatus) {\n        return await this.getStatus();\n      }\n\n      if (isUmbrellaCmd) {\n        // some commands, like deleteSession, we want to make sure to handle on *this* driver,\n        // not the platform driver\n        return await super.executeCommand(cmd, ...args);\n      }\n\n      // here we know that we are executing a session command, and have a valid session driver\n      return await dstSession.executeCommand(cmd, ...args);\n    };\n\n    // now take our default behavior, wrap it with any number of plugin behaviors, and run it\n    const wrappedCmd = this.wrapCommandWithPlugins({\n      driver, cmd, args, plugins, cmdHandledBy, next: defaultBehavior\n    });\n    const res = await this.executeWrappedCommand({wrappedCmd, protocol});\n\n    // if we had plugins, make sure to log out the helpful report about which plugins ended up\n    // handling the command and which didn't\n    this.logPluginHandlerReport(plugins, {cmd, cmdHandledBy});\n\n    // And finally, if the command was createSession, we want to migrate any plugins which were\n    // previously sessionless to use the new sessionId, so that plugins can share state between\n    // their createSession method and other instance methods\n    if (cmd === CREATE_SESSION_COMMAND && this.sessionlessPlugins.length && !res.error) {\n      const sessionId = _.first(res.value);\n      log.info(`Promoting ${this.sessionlessPlugins.length} sessionless plugins to be attached ` +\n               `to session ID ${sessionId}`);\n      this.sessionPlugins[sessionId] = this.sessionlessPlugins;\n      this.sessionlessPlugins = [];\n    }\n\n    return res;\n  }\n\n  wrapCommandWithPlugins ({driver, cmd, args, next, cmdHandledBy, plugins}) {\n    plugins.length && log.info(`Plugins which can handle cmd '${cmd}': ${plugins.map((p) => p.name)}`);\n\n    // now we can go through each plugin and wrap `next` around its own handler, passing the *old*\n    // next in so that it can call it if it wants to\n    for (const plugin of plugins) {\n      // need an IIFE here because we want the value of next that's passed to plugin.handle to be\n      // exactly the value of next here before reassignment; we don't want it to be lazily\n      // evaluated, otherwise we end up with infinite recursion of the last `next` to be defined.\n      cmdHandledBy[plugin.name] = false; // we see a new plugin, so add it to the 'cmdHandledBy' object\n      next = ((_next) => async () => {\n        log.info(`Plugin ${plugin.name} is now handling cmd '${cmd}'`);\n        cmdHandledBy[plugin.name] = true; // if we make it here, this plugin has attempted to handle cmd\n        // first attempt to handle the command via a command-specific handler on the plugin\n        if (plugin[cmd]) {\n          return await plugin[cmd](_next, driver, ...args);\n        }\n        // otherwise, call the generic 'handle' method\n        return await plugin.handle(_next, driver, cmd, ...args);\n      })(next);\n    }\n\n    return next;\n  }\n\n  logPluginHandlerReport (plugins, {cmd, cmdHandledBy}) {\n    if (!plugins.length) {\n      return;\n    }\n\n    // at the end of the day, we have an object representing which plugins ended up getting\n    // their code run as part of handling this command. Because plugins can choose *not* to\n    // pass control to other plugins or to the default driver behavior, this is information\n    // which is probably useful to the user (especially in situations where plugins might not\n    // interact well together, and it would be hard to debug otherwise without this kind of\n    // message).\n    const didHandle = Object.keys(cmdHandledBy).filter((k) => cmdHandledBy[k]);\n    const didntHandle = Object.keys(cmdHandledBy).filter((k) => !cmdHandledBy[k]);\n    if (didntHandle.length > 0) {\n      log.info(`Command '${cmd}' was *not* handled by the following behaviours or plugins, even ` +\n               `though they were registered to handle it: ${JSON.stringify(didntHandle)}. The ` +\n               `command *was* handled by these: ${JSON.stringify(didHandle)}.`);\n    }\n  }\n\n  async executeWrappedCommand ({wrappedCmd, protocol}) {\n    let cmdRes, cmdErr, res = {};\n    try {\n      // At this point, `wrappedCmd` defines a whole sequence of plugin handlers, culminating in\n      // our default handler. Whatever it returns is what we're going to want to send back to the\n      // user.\n      cmdRes = await wrappedCmd();\n    } catch (e) {\n      cmdErr = e;\n    }\n\n    // Sadly, we don't know exactly what kind of object will be returned. It will either be a bare\n    // object, or a protocol-aware object with protocol and error/value keys. So we need to sniff\n    // it and make sure we don't double-wrap it if it's the latter kind.\n    if (_.isPlainObject(cmdRes) && _.has(cmdRes, 'protocol')) {\n      res = cmdRes;\n    } else {\n      res.value = cmdRes;\n      res.error = cmdErr;\n      res.protocol = protocol;\n    }\n    return res;\n  }\n\n  proxyActive (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && _.isFunction(dstSession.proxyActive) && dstSession.proxyActive(sessionId);\n  }\n\n  getProxyAvoidList (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession ? dstSession.getProxyAvoidList() : [];\n  }\n\n  canProxy (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && dstSession.canProxy(sessionId);\n  }\n}\n\n// help decide which commands should be proxied to sub-drivers and which\n// should be handled by this, our umbrella driver\nfunction isAppiumDriverCommand (cmd) {\n  return !isSessionCommand(cmd) || cmd === 'deleteSession';\n}\n\n/**\n * Thrown when Appium tried to proxy a command using a driver's `proxyCommand` method but the\n * method did not exist\n */\nexport class NoDriverProxyCommandError extends Error {\n  /**\n   * @type {Readonly<string>}\n   */\n  code = 'APPIUMERR_NO_DRIVER_PROXYCOMMAND';\n\n  constructor () {\n    super(`The default behavior for this command was to proxy, but the driver ` +\n          `did not have the 'proxyCommand' method defined. To fully support ` +\n          `plugins, drivers should have 'proxyCommand' set to a jwpProxy object's ` +\n          `'command()' method, in addition to the normal 'proxyReqRes'`);\n  }\n}\n\nexport { AppiumDriver };\n"],"file":"lib/appium.js","sourceRoot":"../.."}
566
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/appium.js"],"names":["desiredCapabilityConstraints","automationName","presence","isString","platformName","sessionsListGuard","AsyncLock","pendingDriversGuard","AppiumDriver","BaseDriver","constructor","args","tmpDir","process","env","APPIUM_TMP_DIR","desiredCapConstraints","newCommandTimeoutMs","sessions","pendingDrivers","pluginClasses","sessionPlugins","sessionlessPlugins","isCommandsQueueEnabled","sessionExists","sessionId","dstSession","driverForSession","getStatus","build","_","clone","getSessions","acquire","name","toPairs","map","id","driver","capabilities","caps","printNewSessionAnnouncement","driverName","driverVersion","driverBaseVersion","log","info","APPIUM_VER","baseVersion","_findMatchingDriver","assignCliArgsToExtension","extType","extName","extInstance","cliArgs","isEmpty","createSession","jsonwpCaps","reqCaps","w3cCapabilities","defaultCapabilities","cloneDeep","defaultSettings","jwpSettings","Object","assign","w3cSettings","alwaysMatch","firstMatchEntry","firstMatch","protocol","innerSessionId","dCaps","parsedCaps","desiredCaps","processedJsonwpCapabilities","processedW3CCapabilities","error","InnerDriver","version","driverConfig","sessionOverride","deleteAllSessions","runningDriversData","otherPendingDriversData","driverInstance","relaxedSecurityEnabled","denyInsecure","a","allowInsecure","server","serverHost","address","serverPort","port","serverPath","basePath","curSessionDataForDriver","e","errors","SessionNotCreatedError","message","drv","driverData","push","pull","attachUnexpectedShutdownHandler","startNewCommandTimeout","isW3CProtocol","JSON","stringify","updateSettings","isMjsonwpProtocol","value","onShutdown","cause","Error","warn","plugin","isFunction","onUnexpectedShutdown","debug","data","values","filter","s","datum","deleteSession","otherSessionsData","curConstructorName","key","opts","sessionsCount","size","force","reason","util","pluralize","cleanupPromises","startUnexpectedShutdown","keys","cleanupPromise","pluginsForSession","createPluginInstances","pluginsToHandleCmd","cmd","p","handle","PluginClass","pluginName","executeCommand","isGetStatus","isUmbrellaCmd","isAppiumDriverCommand","isSessionCmd","reqForProxy","last","pop","plugins","cmdHandledBy","default","defaultBehavior","length","proxyCommand","NoDriverProxyCommandError","originalUrl","method","body","wrappedCmd","wrapCommandWithPlugins","next","res","executeWrappedCommand","logPluginHandlerReport","CREATE_SESSION_COMMAND","first","_next","didHandle","k","didntHandle","cmdRes","cmdErr","isPlainObject","has","proxyActive","getProxyAvoidList","canProxy"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport log from './logger';\nimport { getBuildInfo, updateBuildInfo, APPIUM_VER } from './config';\nimport { findMatchingDriver } from './drivers';\nimport { BaseDriver, errors, isSessionCommand,\n         CREATE_SESSION_COMMAND } from '@appium/base-driver';\nimport AsyncLock from 'async-lock';\nimport { parseCapsForInnerDriver, pullSettings } from './utils';\nimport { util } from '@appium/support';\n\nconst desiredCapabilityConstraints = {\n  automationName: {\n    presence: true,\n    isString: true,\n  },\n  platformName: {\n    presence: true,\n    isString: true,\n  },\n};\n\nconst sessionsListGuard = new AsyncLock();\nconst pendingDriversGuard = new AsyncLock();\n\nclass AppiumDriver extends BaseDriver {\n  constructor (args) {\n    // It is necessary to set `--tmp` here since it should be set to\n    // process.env.APPIUM_TMP_DIR once at an initial point in the Appium lifecycle.\n    // The process argument will be referenced by BaseDriver.\n    // Please call @appium/support.tempDir module to apply this benefit.\n    if (args.tmpDir) {\n      process.env.APPIUM_TMP_DIR = args.tmpDir;\n    }\n\n    super(args);\n\n    this.desiredCapConstraints = desiredCapabilityConstraints;\n\n    // the main Appium Driver has no new command timeout\n    this.newCommandTimeoutMs = 0;\n\n    this.args = {...args};\n\n    // Access to sessions list must be guarded with a Semaphore, because\n    // it might be changed by other async calls at any time\n    // It is not recommended to access this property directly from the outside\n    this.sessions = {};\n\n    // Access to pending drivers list must be guarded with a Semaphore, because\n    // it might be changed by other async calls at any time\n    // It is not recommended to access this property directly from the outside\n    this.pendingDrivers = {};\n\n    this.pluginClasses = []; // list of which plugins are active\n    this.sessionPlugins = {}; // map of sessions to actual plugin instances per session\n    this.sessionlessPlugins = []; // some commands are sessionless, so we need a set of plugins for them\n\n    // allow this to happen in the background, so no `await`\n    updateBuildInfo();\n  }\n\n  /** @type {DriverConfig|undefined} */\n  driverConfig;\n\n  /**\n   * Cancel commands queueing for the umbrella Appium driver\n   */\n  get isCommandsQueueEnabled () {\n    return false;\n  }\n\n  sessionExists (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && dstSession.sessionId !== null;\n  }\n\n  driverForSession (sessionId) {\n    return this.sessions[sessionId];\n  }\n\n  async getStatus () { // eslint-disable-line require-await\n    return {\n      build: _.clone(getBuildInfo()),\n    };\n  }\n\n  async getSessions () {\n    const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);\n    return _.toPairs(sessions)\n      .map(([id, driver]) => ({id, capabilities: driver.caps}));\n  }\n\n  printNewSessionAnnouncement (driverName, driverVersion, driverBaseVersion) {\n    log.info(driverVersion\n      ? `Appium v${APPIUM_VER} creating new ${driverName} (v${driverVersion}) session`\n      : `Appium v${APPIUM_VER} creating new ${driverName} session`\n    );\n    log.info(`Checking BaseDriver versions for Appium and ${driverName}`);\n    log.info(AppiumDriver.baseVersion\n      ? `Appium's BaseDriver version is ${AppiumDriver.baseVersion}`\n      : `Could not determine Appium's BaseDriver version`\n    );\n    log.info(driverBaseVersion\n      ? `${driverName}'s BaseDriver version is ${driverBaseVersion}`\n      : `Could not determine ${driverName}'s BaseDriver version`\n    );\n  }\n\n  /**\n   * This is just an alias for driver.js's method, which is necessary for\n   * mocking in the test suite\n   */\n  _findMatchingDriver (...args) {\n    return findMatchingDriver(...args);\n  }\n\n  /**\n   * Validate and assign CLI args for a driver or plugin\n   *\n   * If the extension has provided a schema, validation has already happened.\n   * @param {import('./ext-config-io').ExtensionType} extType 'driver' or 'plugin'\n   * @param {string} extName the name of the extension\n   * @param {Object} extInstance the driver or plugin instance\n   */\n  assignCliArgsToExtension (extType, extName, extInstance) {\n    const cliArgs = this.args[extType]?.[extName];\n    if (!_.isEmpty(cliArgs)) {\n      extInstance.cliArgs = cliArgs;\n    }\n  }\n\n  /**\n   * Create a new session\n   * @param {Object} jsonwpCaps JSONWP formatted desired capabilities\n   * @param {Object} reqCaps Required capabilities (JSONWP standard)\n   * @param {Object} w3cCapabilities W3C capabilities\n   * @return {Array} Unique session ID and capabilities\n   */\n  async createSession (jsonwpCaps, reqCaps, w3cCapabilities) {\n    const defaultCapabilities = _.cloneDeep(this.args.defaultCapabilities);\n    const defaultSettings = pullSettings(defaultCapabilities);\n    jsonwpCaps = _.cloneDeep(jsonwpCaps);\n    const jwpSettings = Object.assign({}, defaultSettings, pullSettings(jsonwpCaps));\n    w3cCapabilities = _.cloneDeep(w3cCapabilities);\n    // It is possible that the client only provides caps using JSONWP standard,\n    // although firstMatch/alwaysMatch properties are still present.\n    // In such case we assume the client understands W3C protocol and merge the given\n    // JSONWP caps to W3C caps\n    const w3cSettings = Object.assign({}, jwpSettings);\n    Object.assign(w3cSettings, pullSettings((w3cCapabilities || {}).alwaysMatch || {}));\n    for (const firstMatchEntry of ((w3cCapabilities || {}).firstMatch || [])) {\n      Object.assign(w3cSettings, pullSettings(firstMatchEntry));\n    }\n\n    let protocol;\n    let innerSessionId, dCaps;\n    try {\n      // Parse the caps into a format that the InnerDriver will accept\n      const parsedCaps = parseCapsForInnerDriver(\n        jsonwpCaps,\n        w3cCapabilities,\n        this.desiredCapConstraints,\n        defaultCapabilities\n      );\n\n      const {desiredCaps, processedJsonwpCapabilities, processedW3CCapabilities, error} = parsedCaps;\n      protocol = parsedCaps.protocol;\n\n      // If the parsing of the caps produced an error, throw it in here\n      if (error) {\n        throw error;\n      }\n\n      const {\n        driver: InnerDriver,\n        version: driverVersion,\n        driverName\n      } = this._findMatchingDriver(this.driverConfig, desiredCaps);\n      this.printNewSessionAnnouncement(InnerDriver.name, driverVersion, InnerDriver.baseVersion);\n\n      if (this.args.sessionOverride) {\n        await this.deleteAllSessions();\n      }\n\n      let runningDriversData, otherPendingDriversData;\n\n      const driverInstance = new InnerDriver(this.args, true);\n\n      // We want to assign security values directly on the driver. The driver\n      // should not read security values from `this.opts` because those values\n      // could have been set by a malicious user via capabilities, whereas we\n      // want a guarantee the values were set by the appium server admin\n      if (this.args.relaxedSecurityEnabled) {\n        log.info(`Applying relaxed security to '${InnerDriver.name}' as per ` +\n                 `server command line argument. All insecure features will be ` +\n                 `enabled unless explicitly disabled by --deny-insecure`);\n        driverInstance.relaxedSecurityEnabled = true;\n      }\n\n      if (!_.isEmpty(this.args.denyInsecure)) {\n        log.info('Explicitly preventing use of insecure features:');\n        this.args.denyInsecure.map((a) => log.info(`    ${a}`));\n        driverInstance.denyInsecure = this.args.denyInsecure;\n      }\n\n      if (!_.isEmpty(this.args.allowInsecure)) {\n        log.info('Explicitly enabling use of insecure features:');\n        this.args.allowInsecure.map((a) => log.info(`    ${a}`));\n        driverInstance.allowInsecure = this.args.allowInsecure;\n      }\n\n      // Likewise, any driver-specific CLI args that were passed in should be assigned directly to\n      // the driver so that they cannot be mimicked by a malicious user sending in capabilities\n      this.assignCliArgsToExtension('driver', driverName, driverInstance);\n\n\n      // This assignment is required for correct web sockets functionality inside the driver\n      driverInstance.server = this.server;\n\n      // Drivers/plugins might also want to know where they are hosted\n      driverInstance.serverHost = this.args.address;\n      driverInstance.serverPort = this.args.port;\n      driverInstance.serverPath = this.args.basePath;\n\n      try {\n        runningDriversData = await this.curSessionDataForDriver(InnerDriver);\n      } catch (e) {\n        throw new errors.SessionNotCreatedError(e.message);\n      }\n      await pendingDriversGuard.acquire(AppiumDriver.name, () => {\n        this.pendingDrivers[InnerDriver.name] = this.pendingDrivers[InnerDriver.name] || [];\n        otherPendingDriversData = this.pendingDrivers[InnerDriver.name].map((drv) => drv.driverData);\n        this.pendingDrivers[InnerDriver.name].push(driverInstance);\n      });\n\n      try {\n        [innerSessionId, dCaps] = await driverInstance.createSession(\n          processedJsonwpCapabilities,\n          reqCaps,\n          processedW3CCapabilities,\n          [...runningDriversData, ...otherPendingDriversData]\n        );\n        protocol = driverInstance.protocol;\n        await sessionsListGuard.acquire(AppiumDriver.name, () => {\n          this.sessions[innerSessionId] = driverInstance;\n        });\n      } finally {\n        await pendingDriversGuard.acquire(AppiumDriver.name, () => {\n          _.pull(this.pendingDrivers[InnerDriver.name], driverInstance);\n        });\n      }\n\n      this.attachUnexpectedShutdownHandler(driverInstance, innerSessionId);\n\n      log.info(`New ${InnerDriver.name} session created successfully, session ` +\n              `${innerSessionId} added to master session list`);\n\n      // set the New Command Timeout for the inner driver\n      driverInstance.startNewCommandTimeout();\n\n      // apply initial values to Appium settings (if provided)\n      if (driverInstance.isW3CProtocol() && !_.isEmpty(w3cSettings)) {\n        log.info(`Applying the initial values to Appium settings parsed from W3C caps: ` +\n          JSON.stringify(w3cSettings));\n        await driverInstance.updateSettings(w3cSettings);\n      } else if (driverInstance.isMjsonwpProtocol() && !_.isEmpty(jwpSettings)) {\n        log.info(`Applying the initial values to Appium settings parsed from MJSONWP caps: ` +\n          JSON.stringify(jwpSettings));\n        await driverInstance.updateSettings(jwpSettings);\n      }\n    } catch (error) {\n      return {\n        protocol,\n        error,\n      };\n    }\n\n    return {\n      protocol,\n      value: [innerSessionId, dCaps, protocol]\n    };\n  }\n\n  attachUnexpectedShutdownHandler (driver, innerSessionId) {\n    const onShutdown = (cause = new Error('Unknown error')) => {\n      log.warn(`Ending session, cause was '${cause.message}'`);\n\n      if (this.sessionPlugins[innerSessionId]) {\n        for (const plugin of this.sessionPlugins[innerSessionId]) {\n          if (_.isFunction(plugin.onUnexpectedShutdown)) {\n            log.debug(`Plugin ${plugin.name} defines an unexpected shutdown handler; calling it now`);\n            try {\n              plugin.onUnexpectedShutdown(driver, cause);\n            } catch (e) {\n              log.warn(`Got an error when running plugin ${plugin.name} shutdown handler: ${e}`);\n            }\n          } else {\n            log.debug(`Plugin ${plugin.name} does not define an unexpected shutdown handler`);\n          }\n        }\n      }\n\n      log.info(`Removing session '${innerSessionId}' from our master session list`);\n      delete this.sessions[innerSessionId];\n      delete this.sessionPlugins[innerSessionId];\n    };\n\n    if (_.isFunction(driver.onUnexpectedShutdown)) {\n      driver.onUnexpectedShutdown(onShutdown);\n    } else {\n      log.warn(`Failed to attach the unexpected shutdown listener. ` +\n        `Is 'onUnexpectedShutdown' method available for '${driver.constructor.name}'?`);\n    }\n  }\n\n  async curSessionDataForDriver (InnerDriver) {\n    const sessions = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions);\n    const data = _.values(sessions)\n                   .filter((s) => s.constructor.name === InnerDriver.name)\n                   .map((s) => s.driverData);\n    for (let datum of data) {\n      if (!datum) {\n        throw new Error(`Problem getting session data for driver type ` +\n                        `${InnerDriver.name}; does it implement 'get ` +\n                        `driverData'?`);\n      }\n    }\n    return data;\n  }\n\n  async deleteSession (sessionId) {\n    let protocol;\n    try {\n      let otherSessionsData = null;\n      let dstSession = null;\n      await sessionsListGuard.acquire(AppiumDriver.name, () => {\n        if (!this.sessions[sessionId]) {\n          return;\n        }\n        const curConstructorName = this.sessions[sessionId].constructor.name;\n        otherSessionsData = _.toPairs(this.sessions)\n              .filter(([key, value]) => value.constructor.name === curConstructorName && key !== sessionId)\n              .map(([, value]) => value.driverData);\n        dstSession = this.sessions[sessionId];\n        protocol = dstSession.protocol;\n        log.info(`Removing session ${sessionId} from our master session list`);\n        // regardless of whether the deleteSession completes successfully or not\n        // make the session unavailable, because who knows what state it might\n        // be in otherwise\n        delete this.sessions[sessionId];\n        delete this.sessionPlugins[sessionId];\n      });\n      return {\n        protocol,\n        value: await dstSession.deleteSession(sessionId, otherSessionsData),\n      };\n    } catch (e) {\n      log.error(`Had trouble ending session ${sessionId}: ${e.message}`);\n      return {\n        protocol,\n        error: e,\n      };\n    }\n  }\n\n  async deleteAllSessions (opts = {}) {\n    const sessionsCount = _.size(this.sessions);\n    if (0 === sessionsCount) {\n      log.debug('There are no active sessions for cleanup');\n      return;\n    }\n\n    const {\n      force = false,\n      reason,\n    } = opts;\n    log.debug(`Cleaning up ${util.pluralize('active session', sessionsCount, true)}`);\n    const cleanupPromises = force\n      ? _.values(this.sessions).map((drv) => drv.startUnexpectedShutdown(reason && new Error(reason)))\n      : _.keys(this.sessions).map((id) => this.deleteSession(id));\n    for (const cleanupPromise of cleanupPromises) {\n      try {\n        await cleanupPromise;\n      } catch (e) {\n        log.debug(e);\n      }\n    }\n  }\n\n  /**\n   * Get the appropriate plugins for a session (or sessionless plugins)\n   *\n   * @param {?string} sessionId - the sessionId (or null) to use to find plugins\n   * @returns {Array} - array of plugin instances\n   */\n  pluginsForSession (sessionId = null) {\n    if (sessionId) {\n      if (!this.sessionPlugins[sessionId]) {\n        this.sessionPlugins[sessionId] = this.createPluginInstances();\n      }\n      return this.sessionPlugins[sessionId];\n    }\n\n    if (_.isEmpty(this.sessionlessPlugins)) {\n      this.sessionlessPlugins = this.createPluginInstances();\n    }\n    return this.sessionlessPlugins;\n  }\n\n  /**\n   * To get plugins for a command, we either get the plugin instances associated with the\n   * particular command's session, or in the case of sessionless plugins, pull from the set of\n   * plugin instances reserved for sessionless commands (and we lazily create plugin instances on\n   * first use)\n   *\n   * @param {string} cmd - the name of the command to find a plugin to handle\n   * @param {?string} sessionId - the particular session for which to find a plugin, or null if\n   * sessionless\n   */\n  pluginsToHandleCmd (cmd, sessionId = null) {\n    // to handle a given command, a plugin should either implement that command as a plugin\n    // instance method or it should implement a generic 'handle' method\n    return this.pluginsForSession(sessionId)\n      .filter((p) => _.isFunction(p[cmd]) || _.isFunction(p.handle));\n  }\n\n  createPluginInstances () {\n    return this.pluginClasses.map((PluginClass) => {\n      const name = PluginClass.pluginName;\n      const plugin = new PluginClass(name);\n      this.assignCliArgsToExtension('plugin', name, plugin);\n      return plugin;\n    });\n  }\n\n  async executeCommand (cmd, ...args) {\n    // We have basically three cases for how to handle commands:\n    // 1. handle getStatus (we do this as a special out of band case so it doesn't get added to an\n    //    execution queue, and can be called while e.g. createSession is in progress)\n    // 2. handle commands that this umbrella driver should handle, rather than the actual session\n    //    driver (for example, deleteSession, or other non-session commands)\n    // 3. handle session driver commands.\n    // The tricky part is that because we support command plugins, we need to wrap any of these\n    // cases with plugin handling.\n\n    const isGetStatus = cmd === 'getStatus';\n    const isUmbrellaCmd = !isGetStatus && isAppiumDriverCommand(cmd);\n    const isSessionCmd = !isGetStatus && !isUmbrellaCmd;\n\n    // if a plugin override proxying for this command and that is why we are here instead of just\n    // letting the protocol proxy the command entirely, determine that, get the request object for\n    // use later on, then clean up the args\n    const reqForProxy = _.last(args)?.reqForProxy;\n    if (reqForProxy) {\n      args.pop();\n    }\n\n\n    // first do some error checking. If we're requesting a session command execution, then make\n    // sure that session actually exists on the session driver, and set the session driver itself\n    let sessionId = null;\n    let dstSession = null;\n    let protocol = null;\n    let driver = this;\n    if (isSessionCmd) {\n      sessionId = _.last(args);\n      dstSession = await sessionsListGuard.acquire(AppiumDriver.name, () => this.sessions[sessionId]);\n      if (!dstSession) {\n        throw new Error(`The session with id '${sessionId}' does not exist`);\n      }\n      // now save the response protocol given that the session driver's protocol might differ\n      protocol = dstSession.protocol;\n      driver = dstSession;\n    }\n\n    // get any plugins which are registered as handling this command\n    const plugins = this.pluginsToHandleCmd(cmd, sessionId);\n\n    // now we define a 'cmdHandledBy' object which will keep track of which plugins have handled this\n    // command. we care about this because (a) multiple plugins can handle the same command, and\n    // (b) there's no guarantee that a plugin will actually call the next() method which runs the\n    // original command execution. This results in a situation where the command might be handled\n    // by some but not all plugins, or by plugin(s) but not by the default behavior. So start out\n    // this object declaring that the default handler has not been executed.\n    const cmdHandledBy = {default: false};\n\n    // now we define an async function which will be passed to plugins, and successively wrapped\n    // if there is more than one plugin that can handle the command. To start off with, the async\n    // function is defined as calling the default behavior, i.e., whichever of the 3 cases above is\n    // the appropriate one\n    const defaultBehavior = async () => {\n      // if we're running with plugins, make sure we log that the default behavior is actually\n      // happening so we can tell when the plugin call chain is unwrapping to the default behavior\n      // if that's what happens\n      plugins.length && log.info(`Executing default handling behavior for command '${cmd}'`);\n\n      // if we make it here, we know that the default behavior is handled\n      cmdHandledBy.default = true;\n\n      if (reqForProxy) {\n        // we would have proxied this command had a plugin not handled it, so the default behavior\n        // is to do the proxy and retrieve the result internally so it can be passed to the plugin\n        // in case it calls 'await next()'. This requires that the driver have defined\n        // 'proxyCommand' and not just 'proxyReqRes'.\n        if (!dstSession.proxyCommand) {\n          throw new NoDriverProxyCommandError();\n        }\n        return await dstSession.proxyCommand(reqForProxy.originalUrl, reqForProxy.method,\n          reqForProxy.body);\n      }\n\n      if (isGetStatus) {\n        return await this.getStatus();\n      }\n\n      if (isUmbrellaCmd) {\n        // some commands, like deleteSession, we want to make sure to handle on *this* driver,\n        // not the platform driver\n        return await super.executeCommand(cmd, ...args);\n      }\n\n      // here we know that we are executing a session command, and have a valid session driver\n      return await dstSession.executeCommand(cmd, ...args);\n    };\n\n    // now take our default behavior, wrap it with any number of plugin behaviors, and run it\n    const wrappedCmd = this.wrapCommandWithPlugins({\n      driver, cmd, args, plugins, cmdHandledBy, next: defaultBehavior\n    });\n    const res = await this.executeWrappedCommand({wrappedCmd, protocol});\n\n    // if we had plugins, make sure to log out the helpful report about which plugins ended up\n    // handling the command and which didn't\n    this.logPluginHandlerReport(plugins, {cmd, cmdHandledBy});\n\n    // And finally, if the command was createSession, we want to migrate any plugins which were\n    // previously sessionless to use the new sessionId, so that plugins can share state between\n    // their createSession method and other instance methods\n    if (cmd === CREATE_SESSION_COMMAND && this.sessionlessPlugins.length && !res.error) {\n      const sessionId = _.first(res.value);\n      log.info(`Promoting ${this.sessionlessPlugins.length} sessionless plugins to be attached ` +\n               `to session ID ${sessionId}`);\n      this.sessionPlugins[sessionId] = this.sessionlessPlugins;\n      this.sessionlessPlugins = [];\n    }\n\n    return res;\n  }\n\n  wrapCommandWithPlugins ({driver, cmd, args, next, cmdHandledBy, plugins}) {\n    plugins.length && log.info(`Plugins which can handle cmd '${cmd}': ${plugins.map((p) => p.name)}`);\n\n    // now we can go through each plugin and wrap `next` around its own handler, passing the *old*\n    // next in so that it can call it if it wants to\n    for (const plugin of plugins) {\n      // need an IIFE here because we want the value of next that's passed to plugin.handle to be\n      // exactly the value of next here before reassignment; we don't want it to be lazily\n      // evaluated, otherwise we end up with infinite recursion of the last `next` to be defined.\n      cmdHandledBy[plugin.name] = false; // we see a new plugin, so add it to the 'cmdHandledBy' object\n      next = ((_next) => async () => {\n        log.info(`Plugin ${plugin.name} is now handling cmd '${cmd}'`);\n        cmdHandledBy[plugin.name] = true; // if we make it here, this plugin has attempted to handle cmd\n        // first attempt to handle the command via a command-specific handler on the plugin\n        if (plugin[cmd]) {\n          return await plugin[cmd](_next, driver, ...args);\n        }\n        // otherwise, call the generic 'handle' method\n        return await plugin.handle(_next, driver, cmd, ...args);\n      })(next);\n    }\n\n    return next;\n  }\n\n  logPluginHandlerReport (plugins, {cmd, cmdHandledBy}) {\n    if (!plugins.length) {\n      return;\n    }\n\n    // at the end of the day, we have an object representing which plugins ended up getting\n    // their code run as part of handling this command. Because plugins can choose *not* to\n    // pass control to other plugins or to the default driver behavior, this is information\n    // which is probably useful to the user (especially in situations where plugins might not\n    // interact well together, and it would be hard to debug otherwise without this kind of\n    // message).\n    const didHandle = Object.keys(cmdHandledBy).filter((k) => cmdHandledBy[k]);\n    const didntHandle = Object.keys(cmdHandledBy).filter((k) => !cmdHandledBy[k]);\n    if (didntHandle.length > 0) {\n      log.info(`Command '${cmd}' was *not* handled by the following behaviours or plugins, even ` +\n               `though they were registered to handle it: ${JSON.stringify(didntHandle)}. The ` +\n               `command *was* handled by these: ${JSON.stringify(didHandle)}.`);\n    }\n  }\n\n  async executeWrappedCommand ({wrappedCmd, protocol}) {\n    let cmdRes, cmdErr, res = {};\n    try {\n      // At this point, `wrappedCmd` defines a whole sequence of plugin handlers, culminating in\n      // our default handler. Whatever it returns is what we're going to want to send back to the\n      // user.\n      cmdRes = await wrappedCmd();\n    } catch (e) {\n      cmdErr = e;\n    }\n\n    // Sadly, we don't know exactly what kind of object will be returned. It will either be a bare\n    // object, or a protocol-aware object with protocol and error/value keys. So we need to sniff\n    // it and make sure we don't double-wrap it if it's the latter kind.\n    if (_.isPlainObject(cmdRes) && _.has(cmdRes, 'protocol')) {\n      res = cmdRes;\n    } else {\n      res.value = cmdRes;\n      res.error = cmdErr;\n      res.protocol = protocol;\n    }\n    return res;\n  }\n\n  proxyActive (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && _.isFunction(dstSession.proxyActive) && dstSession.proxyActive(sessionId);\n  }\n\n  getProxyAvoidList (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession ? dstSession.getProxyAvoidList() : [];\n  }\n\n  canProxy (sessionId) {\n    const dstSession = this.sessions[sessionId];\n    return dstSession && dstSession.canProxy(sessionId);\n  }\n}\n\n// help decide which commands should be proxied to sub-drivers and which\n// should be handled by this, our umbrella driver\nfunction isAppiumDriverCommand (cmd) {\n  return !isSessionCommand(cmd) || cmd === 'deleteSession';\n}\n\n/**\n * Thrown when Appium tried to proxy a command using a driver's `proxyCommand` method but the\n * method did not exist\n */\nexport class NoDriverProxyCommandError extends Error {\n  /**\n   * @type {Readonly<string>}\n   */\n  code = 'APPIUMERR_NO_DRIVER_PROXYCOMMAND';\n\n  constructor () {\n    super(`The default behavior for this command was to proxy, but the driver ` +\n          `did not have the 'proxyCommand' method defined. To fully support ` +\n          `plugins, drivers should have 'proxyCommand' set to a jwpProxy object's ` +\n          `'command()' method, in addition to the normal 'proxyReqRes'`);\n  }\n}\n\nexport { AppiumDriver };\n"],"file":"lib/appium.js","sourceRoot":"../.."}
@@ -50,7 +50,7 @@ class NPM {
50
50
  args.unshift(cmd);
51
51
 
52
52
  if (json) {
53
- args.push('-json');
53
+ args.push('--json');
54
54
  }
55
55
 
56
56
  const npmCmd = _support.system.isWindows() ? 'npm.cmd' : 'npm';
@@ -65,31 +65,42 @@ class NPM {
65
65
  runner = async () => await acquireLock(_runner);
66
66
  }
67
67
 
68
- const {
69
- stdout,
70
- stderr,
71
- code
72
- } = await runner();
73
- const ret = {
74
- stdout,
75
- stderr,
76
- code,
77
- json: null
78
- };
68
+ let ret;
69
+
70
+ try {
71
+ const {
72
+ stdout,
73
+ stderr,
74
+ code
75
+ } = await runner();
76
+ ret = {
77
+ stdout,
78
+ stderr,
79
+ code
80
+ };
79
81
 
80
- if (json) {
81
82
  try {
82
83
  ret.json = JSON.parse(stdout);
83
84
  } catch (ign) {}
85
+ } catch (e) {
86
+ const {
87
+ stdout,
88
+ stderr,
89
+ code
90
+ } = e;
91
+ const err = new Error(`npm command '${cmd} ${args.join(' ')}' failed with code ${code}.\n\nSTDOUT:\n${stdout.trim()}\n\nSTDERR:\n${stderr.trim()}`);
92
+ throw err;
84
93
  }
85
94
 
86
95
  return ret;
87
96
  }
88
97
 
89
98
  async getLatestVersion(pkg) {
90
- return (await this.exec('view', [pkg, 'dist-tags'], {
99
+ var _await$this$exec$json;
100
+
101
+ return (_await$this$exec$json = (await this.exec('view', [pkg, 'dist-tags'], {
91
102
  json: true
92
- })).json.latest;
103
+ })).json) === null || _await$this$exec$json === void 0 ? void 0 : _await$this$exec$json.latest;
93
104
  }
94
105
 
95
106
  async getLatestSafeUpgradeVersion(pkg, curVersion) {
@@ -206,4 +217,4 @@ class NPM {
206
217
  exports.default = NPM;require('source-map-support').install();
207
218
 
208
219
 
209
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/cli/npm.js"],"names":["INSTALL_LOCKFILE","LINK_LOCKFILE","NPM","constructor","appiumHome","exec","cmd","args","opts","execOpts","cwd","json","lockFile","dummyPkgJson","path","resolve","fs","exists","writeFile","unshift","push","npmCmd","system","isWindows","runner","acquireLock","util","getLockFileGuard","_runner","stdout","stderr","code","ret","JSON","parse","ign","getLatestVersion","pkg","latest","getLatestSafeUpgradeVersion","curVersion","allVersions","getLatestSafeUpgradeFromVersions","safeUpgradeVer","curSemver","semver","Error","testVer","testSemver","prerelease","length","compare","major","format","installPackage","pkgDir","pkgName","pkgVer","res","error","pkgJson","require","linkPackage","pkgPath","name","process","pkgHome","uninstallPackage"],"mappings":"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","sourcesContent":["import path from 'path';\nimport semver from 'semver';\nimport { exec } from 'teen_process';\nimport { fs, mkdirp, util, system } from '@appium/support';\n\nconst INSTALL_LOCKFILE = '.appium.install.lock';\nconst LINK_LOCKFILE = '.appium.link.lock';\n\nexport default class NPM {\n\n  constructor (appiumHome) {\n    this.appiumHome = appiumHome;\n  }\n\n  async exec (cmd, args, opts, execOpts = {}) {\n    let { cwd, json, lockFile } = opts;\n    if (!cwd) {\n      cwd = this.appiumHome;\n    }\n    // ensure the directory we want to install inside of exists\n    await mkdirp(cwd);\n\n    // not only this, this directory needs a 'package.json' inside of it, otherwise, if any\n    // directory in the filesystem tree ABOVE cwd happens to have a package.json or a node_modules\n    // dir in it, NPM will install the module up there instead (silly NPM)\n    const dummyPkgJson = path.resolve(cwd, 'package.json');\n    if (!await fs.exists(dummyPkgJson)) {\n      await fs.writeFile(dummyPkgJson, '{}');\n    }\n\n    // make sure we perform the current operation in cwd\n    execOpts = {...execOpts, cwd};\n\n    args.unshift(cmd);\n    if (json) {\n      args.push('-json');\n    }\n    const npmCmd = system.isWindows() ? 'npm.cmd' : 'npm';\n    let runner = async () => await exec(npmCmd, args, execOpts);\n    if (lockFile) {\n      const acquireLock = util.getLockFileGuard(path.resolve(cwd, lockFile));\n      const _runner = runner;\n      runner = async () => await acquireLock(_runner);\n    }\n    const {stdout, stderr, code} = await runner();\n    const ret = {stdout, stderr, code, json: null};\n\n    if (json) {\n      // if possible, parse NPM's json output. During NPM install 3rd-party\n      // packages can write to stdout, so sometimes the json output can't be\n      // guaranteed to be parseable\n      try {\n        ret.json = JSON.parse(stdout);\n      } catch (ign) {}\n    }\n\n    return ret;\n  }\n\n  async getLatestVersion (pkg) {\n    return (await this.exec('view', [pkg, 'dist-tags'], {\n      json: true\n    })).json.latest;\n  }\n\n  async getLatestSafeUpgradeVersion (pkg, curVersion) {\n    const allVersions = (await this.exec('view', [pkg, 'versions'], {\n      json: true\n    })).json;\n    return this.getLatestSafeUpgradeFromVersions(curVersion, allVersions);\n  }\n\n  /**\n   * Given a current version and a list of all versions for a package, return the version which is\n   * the highest safely-upgradable version (meaning not crossing any major revision boundaries, and\n   * not including any alpha/beta/rc versions)\n   *\n   * @param {string} curVersion - the current version of a package\n   * @param {Array<string>} allVersions - a list of version strings\n   *\n   * @return {string|null} - the highest safely-upgradable version, or null if there isn't one\n   */\n  getLatestSafeUpgradeFromVersions (curVersion, allVersions) {\n    let safeUpgradeVer = null;\n    const curSemver = semver.parse(curVersion);\n    if (curSemver === null) {\n      throw new Error(`Could not parse current version '${curVersion}'`);\n    }\n    for (const testVer of allVersions) {\n      const testSemver = semver.parse(testVer);\n      if (testSemver === null) {\n        throw new Error(`Could not parse version to test against: '${testVer}'`);\n      }\n      // if the test version is a prerelease, ignore it\n      if (testSemver.prerelease.length > 0) {\n        continue;\n      }\n      // if the current version is later than the test version, skip this test version\n      if (curSemver.compare(testSemver) === 1) {\n        continue;\n      }\n      // if the test version is newer, but crosses a major revision boundary, also skip it\n      if (testSemver.major > curSemver.major) {\n        continue;\n      }\n      // otherwise this version is safe to upgrade to. But there might be multiple ones of this\n      // kind, so keep iterating and keeping the highest\n      if (safeUpgradeVer === null || testSemver.compare(safeUpgradeVer) === 1) {\n        safeUpgradeVer = testSemver;\n      }\n    }\n    if (safeUpgradeVer) {\n      safeUpgradeVer = safeUpgradeVer.format();\n    }\n    return safeUpgradeVer;\n  }\n\n  async installPackage ({pkgDir, pkgName, pkgVer}) {\n    const res = await this.exec('install', [\n      '--no-save',\n      '--no-package-lock',\n      pkgVer ? `${pkgName}@${pkgVer}` : pkgName\n    ], {\n      cwd: pkgDir,\n      json: true,\n      lockFile: INSTALL_LOCKFILE\n    });\n\n    if (res.json) {\n      // we parsed a valid json response, so if we got an error here, return that\n      // message straightaway\n      if (res.json.error) {\n        throw new Error(res.json.error);\n      }\n    }\n\n    // Now read package data from the installed package to return, and make sure\n    // everything got installed ok. Remember, pkgName might end up with a / in it due to an npm\n    // org, so if so, that will get correctly exploded into multiple directories, by path.resolve here\n    // (even on Windows!)\n    const pkgJson = path.resolve(pkgDir, 'node_modules', pkgName, 'package.json');\n    try {\n      return require(pkgJson);\n    } catch {\n      throw new Error('The package was not downloaded correctly; its package.json ' +\n                      'did not exist or was unreadable. We looked for it at ' +\n                      pkgJson);\n    }\n  }\n\n  async linkPackage (pkgPath) {\n    // from the path alone we don't know the npm package name, so we need to\n    // look in package.json\n    let pkgName;\n    try {\n      pkgName = require(path.resolve(pkgPath, 'package.json')).name;\n    } catch {\n      throw new Error('Could not find package.json inside the package path ' +\n                      `provided: ${pkgPath}`);\n    }\n\n    // this is added to handle commands with relative paths\n    // ie: \"node . driver install --source=local ../fake-driver\"\n    pkgPath = path.resolve(process.cwd(), pkgPath);\n\n    const pkgHome = path.resolve(this.appiumHome, pkgName);\n\n    // call link with --no-package-lock to ensure no corruption while installing local packages\n    const res = await this.exec('link', ['--no-package-lock', pkgPath], {cwd: pkgHome, lockFile: LINK_LOCKFILE});\n    if (res.json && res.json.error) {\n      throw new Error(res.json.error);\n    }\n\n    // now ensure it was linked to the correct place\n    try {\n      return require(path.resolve(pkgHome, 'node_modules', pkgName, 'package.json'));\n    } catch {\n      throw new Error('The package was not linked correctly; its package.json ' +\n                      'did not exist or was unreadable');\n    }\n  }\n\n  async uninstallPackage (pkgDir, pkg) {\n    await this.exec('uninstall', [pkg], {\n      cwd: pkgDir,\n      lockFile: INSTALL_LOCKFILE\n    });\n  }\n}\n"],"file":"lib/cli/npm.js","sourceRoot":"../../.."}
220
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/cli/npm.js"],"names":["INSTALL_LOCKFILE","LINK_LOCKFILE","NPM","constructor","appiumHome","exec","cmd","args","opts","execOpts","cwd","json","lockFile","dummyPkgJson","path","resolve","fs","exists","writeFile","unshift","push","npmCmd","system","isWindows","runner","acquireLock","util","getLockFileGuard","_runner","ret","stdout","stderr","code","JSON","parse","ign","e","err","Error","join","trim","getLatestVersion","pkg","latest","getLatestSafeUpgradeVersion","curVersion","allVersions","getLatestSafeUpgradeFromVersions","safeUpgradeVer","curSemver","semver","testVer","testSemver","prerelease","length","compare","major","format","installPackage","pkgDir","pkgName","pkgVer","res","error","pkgJson","require","linkPackage","pkgPath","name","process","pkgHome","uninstallPackage"],"mappings":"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","sourcesContent":["// @ts-check\n\nimport path from 'path';\nimport semver from 'semver';\nimport { exec } from 'teen_process';\nimport { fs, mkdirp, util, system } from '@appium/support';\n\nconst INSTALL_LOCKFILE = '.appium.install.lock';\nconst LINK_LOCKFILE = '.appium.link.lock';\n\nexport default class NPM {\n\n  /**\n   * @param {string} appiumHome\n   */\n  constructor (appiumHome) {\n    /** @type {string} */\n    this.appiumHome = appiumHome;\n  }\n\n  /**\n   * Execute `npm` with given args.\n   *\n   * If the process exits with a nonzero code, the contents of `STDOUT` and `STDERR` will be in the\n   * `message` of the {@link TeenProcessExecError} rejected.\n   * @param {string} cmd\n   * @param {string[]} args\n   * @param {{ json?: boolean; cwd?: string; lockFile?: string; }} opts\n   */\n  async exec (cmd, args, opts, execOpts = {}) {\n    let { cwd, json, lockFile } = opts;\n    if (!cwd) {\n      cwd = this.appiumHome;\n    }\n    // ensure the directory we want to install inside of exists\n    await mkdirp(cwd);\n\n    // not only this, this directory needs a 'package.json' inside of it, otherwise, if any\n    // directory in the filesystem tree ABOVE cwd happens to have a package.json or a node_modules\n    // dir in it, NPM will install the module up there instead (silly NPM)\n    const dummyPkgJson = path.resolve(cwd, 'package.json');\n    if (!await fs.exists(dummyPkgJson)) {\n      await fs.writeFile(dummyPkgJson, '{}');\n    }\n\n    // make sure we perform the current operation in cwd\n    execOpts = {...execOpts, cwd};\n\n    args.unshift(cmd);\n    if (json) {\n      args.push('--json');\n    }\n    const npmCmd = system.isWindows() ? 'npm.cmd' : 'npm';\n    let runner = async () => await exec(npmCmd, args, execOpts);\n    if (lockFile) {\n      const acquireLock = util.getLockFileGuard(path.resolve(cwd, lockFile));\n      const _runner = runner;\n      runner = async () => await acquireLock(_runner);\n    }\n\n    let ret;\n    try {\n      const {stdout, stderr, code} = await runner();\n      ret = /** @type {TeenProcessExecResult} */({stdout, stderr, code});\n      // if possible, parse NPM's json output. During NPM install 3rd-party\n      // packages can write to stdout, so sometimes the json output can't be\n      // guaranteed to be parseable\n      try {\n        ret.json = JSON.parse(stdout);\n      } catch (ign) {}\n    } catch (e) {\n      const {stdout, stderr, code} = /** @type {TeenProcessExecError} */(e);\n      const err = new Error(`npm command '${cmd} ${args.join(' ')}' failed with code ${code}.\\n\\nSTDOUT:\\n${stdout.trim()}\\n\\nSTDERR:\\n${stderr.trim()}`);\n      throw err;\n    }\n    return ret;\n  }\n\n  /**\n   * @param {string} pkg\n   */\n  async getLatestVersion (pkg) {\n    return (await this.exec('view', [pkg, 'dist-tags'], {\n      json: true\n    })).json?.latest;\n  }\n\n  /**\n   * @param {string} pkg\n   * @param {string} curVersion\n   */\n  async getLatestSafeUpgradeVersion (pkg, curVersion) {\n    const allVersions = (await this.exec('view', [pkg, 'versions'], {\n      json: true\n    })).json;\n    return this.getLatestSafeUpgradeFromVersions(curVersion, allVersions);\n  }\n\n  /**\n   * Given a current version and a list of all versions for a package, return the version which is\n   * the highest safely-upgradable version (meaning not crossing any major revision boundaries, and\n   * not including any alpha/beta/rc versions)\n   *\n   * @param {string} curVersion - the current version of a package\n   * @param {Array<string>} allVersions - a list of version strings\n   *\n   * @return {string|null} - the highest safely-upgradable version, or null if there isn't one\n   */\n  getLatestSafeUpgradeFromVersions (curVersion, allVersions) {\n    let safeUpgradeVer = null;\n    const curSemver = semver.parse(curVersion);\n    if (curSemver === null) {\n      throw new Error(`Could not parse current version '${curVersion}'`);\n    }\n    for (const testVer of allVersions) {\n      const testSemver = semver.parse(testVer);\n      if (testSemver === null) {\n        throw new Error(`Could not parse version to test against: '${testVer}'`);\n      }\n      // if the test version is a prerelease, ignore it\n      if (testSemver.prerelease.length > 0) {\n        continue;\n      }\n      // if the current version is later than the test version, skip this test version\n      if (curSemver.compare(testSemver) === 1) {\n        continue;\n      }\n      // if the test version is newer, but crosses a major revision boundary, also skip it\n      if (testSemver.major > curSemver.major) {\n        continue;\n      }\n      // otherwise this version is safe to upgrade to. But there might be multiple ones of this\n      // kind, so keep iterating and keeping the highest\n      if (safeUpgradeVer === null || testSemver.compare(safeUpgradeVer) === 1) {\n        safeUpgradeVer = testSemver;\n      }\n    }\n    if (safeUpgradeVer) {\n      safeUpgradeVer = safeUpgradeVer.format();\n    }\n    return safeUpgradeVer;\n  }\n\n  /**\n   *\n   * @param {{pkgDir: string, pkgName: string, pkgVer?: string}} param0\n   * @returns {Promise<import('type-fest').PackageJson>}\n   */\n  async installPackage ({pkgDir, pkgName, pkgVer}) {\n    const res = await this.exec('install', [\n      '--no-save',\n      '--no-package-lock',\n      pkgVer ? `${pkgName}@${pkgVer}` : pkgName\n    ], {\n      cwd: pkgDir,\n      json: true,\n      lockFile: INSTALL_LOCKFILE\n    });\n\n    if (res.json) {\n      // we parsed a valid json response, so if we got an error here, return that\n      // message straightaway\n      if (res.json.error) {\n        throw new Error(res.json.error);\n      }\n    }\n\n    // Now read package data from the installed package to return, and make sure\n    // everything got installed ok. Remember, pkgName might end up with a / in it due to an npm\n    // org, so if so, that will get correctly exploded into multiple directories, by path.resolve here\n    // (even on Windows!)\n    const pkgJson = path.resolve(pkgDir, 'node_modules', pkgName, 'package.json');\n    try {\n      return require(pkgJson);\n    } catch {\n      throw new Error('The package was not downloaded correctly; its package.json ' +\n                      'did not exist or was unreadable. We looked for it at ' +\n                      pkgJson);\n    }\n  }\n\n  /**\n   * @param {string} pkgPath\n   */\n  async linkPackage (pkgPath) {\n    // from the path alone we don't know the npm package name, so we need to\n    // look in package.json\n    let pkgName;\n    try {\n      pkgName = require(path.resolve(pkgPath, 'package.json')).name;\n    } catch {\n      throw new Error('Could not find package.json inside the package path ' +\n                      `provided: ${pkgPath}`);\n    }\n\n    // this is added to handle commands with relative paths\n    // ie: \"node . driver install --source=local ../fake-driver\"\n    pkgPath = path.resolve(process.cwd(), pkgPath);\n\n    const pkgHome = path.resolve(this.appiumHome, pkgName);\n\n    // call link with --no-package-lock to ensure no corruption while installing local packages\n    const res = await this.exec('link', ['--no-package-lock', pkgPath], {cwd: pkgHome, lockFile: LINK_LOCKFILE});\n    if (res.json && res.json.error) {\n      throw new Error(res.json.error);\n    }\n\n    // now ensure it was linked to the correct place\n    try {\n      return require(path.resolve(pkgHome, 'node_modules', pkgName, 'package.json'));\n    } catch {\n      throw new Error('The package was not linked correctly; its package.json ' +\n                      'did not exist or was unreadable');\n    }\n  }\n\n  /**\n   * @param {string} pkgDir\n   * @param {string} pkg\n   */\n  async uninstallPackage (pkgDir, pkg) {\n    await this.exec('uninstall', [pkg], {\n      cwd: pkgDir,\n      lockFile: INSTALL_LOCKFILE\n    });\n  }\n}\n\n\n/**\n * Result from a non-zero-exit execution of `appium`\n * @typedef {Object} TeenProcessExecResult\n * @property {string} stdout - Stdout\n * @property {string} stderr - Stderr\n * @property {number?} code - Exit code\n * @property {any} json - JSON parsed from stdout\n */\n\n/**\n * Extra props `teen_process.exec` adds to its error objects\n * @typedef {Object} TeenProcessExecErrorProps\n * @property {string} stdout - STDOUT\n * @property {string} stderr - STDERR\n * @property {number?} code - Exit code\n */\n\n\n/**\n * Error thrown by `teen_process.exec`\n * @typedef {Error & TeenProcessExecErrorProps} TeenProcessExecError\n */\n"],"file":"lib/cli/npm.js","sourceRoot":"../../.."}
@@ -83,4 +83,4 @@ const installLocalExtension = _lodash.default.curry(async (appiumHome, type, pat
83
83
  exports.installLocalExtension = installLocalExtension;require('source-map-support').install();
84
84
 
85
85
 
86
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["test/cli/cli-helpers.js"],"names":["EXECUTABLE","path","join","PROJECT_ROOT","run","appiumHome","args","env","APPIUM_HOME","PATH","process","execPath","cwd","err","stdout","stderr","runErr","Object","assign","originalMessage","message","trim","command","runAppium","_","curry","runAppiumRaw","runAppiumJson","includes","push","result","JSON","parse","installLocalExtension","type","pathToExtension"],"mappings":";;;;;;;;;;;AAEA;;AACA;;AACA;;AACA;;AAKO,MAAMA,UAAU,GAAGC,cAAKC,IAAL,CACxBC,qBADwB,EAExB,UAFwB,EAGxB,QAHwB,EAIxB,OAJwB,EAKxB,KALwB,EAMxB,SANwB,CAAnB;;;;AAiBP,eAAeC,GAAf,CAAoBC,UAApB,EAAgCC,IAAhC,EAAsC;AACpC,QAAMC,GAAG,GAAG;AACVC,IAAAA,WAAW,EAAEH,UADH;AAEVI,IAAAA,IAAI,EAAEC,OAAO,CAACH,GAAR,CAAYE;AAFR,GAAZ;;AAIA,MAAI;AAIF,WAAO,MAAM,wBAAKC,OAAO,CAACC,QAAb,EAAuB,CAACX,UAAD,EAAa,GAAGM,IAAhB,CAAvB,EAA8C;AACzDM,MAAAA,GAAG,EAAET,qBADoD;AAEzDI,MAAAA;AAFyD,KAA9C,CAAb;AAID,GARD,CAQE,OAAOM,GAAP,EAAY;AACZ,UAAM;AAACC,MAAAA,MAAD;AAASC,MAAAA;AAAT,QAAuDF,GAA7D;AAIA,UAAMG,MAAM,GAAGC,MAAM,CAACC,MAAP,CACbL,GADa,EAEb;AACEM,MAAAA,eAAe,EAAEN,GAAG,CAACO,OADvB;AAEEA,MAAAA,OAAO,EAAG,GAAEN,MAAM,CAACO,IAAP,EAAc,OAAMN,MAAM,CAACM,IAAP,EAAc,EAFhD;AAGEC,MAAAA,OAAO,EAAG,GAAEZ,OAAO,CAACC,QAAS,IAAGX,UAAW,IAAGM,IAAI,CAACJ,IAAL,CAAU,GAAV,CAAe,EAH/D;AAIEK,MAAAA,GAJF;AAKEK,MAAAA,GAAG,EAAET;AALP,KAFa,CAAf;AASA,UAAMa,MAAN;AACD;AACF;;AAOM,MAAMO,SAAS,GAAGC,gBAAEC,KAAF,CAMvB,OAAOpB,UAAP,EAAmBC,IAAnB,KAA4B;AAC1B,QAAM;AAACQ,IAAAA;AAAD,MAAW,MAAMV,GAAG,CAACC,UAAD,EAAaC,IAAb,CAA1B;AACA,SAAOQ,MAAP;AACD,CATsB,CAAlB;;;;AAgBA,MAAMY,YAAY,GAAGF,gBAAEC,KAAF,CAM1B,OAAOpB,UAAP,EAAmBC,IAAnB,KAA4B,MAAMF,GAAG,CAACC,UAAD,EAAaC,IAAb,CANX,CAArB;;;;AAaA,MAAMqB,aAAa,GAAGH,gBAAEC,KAAF,CAM3B,OAAOpB,UAAP,EAAmBC,IAAnB,KAA4B;AAC1B,MAAI,CAACA,IAAI,CAACsB,QAAL,CAAc,QAAd,CAAL,EAA8B;AAC5BtB,IAAAA,IAAI,CAACuB,IAAL,CAAU,QAAV;AACD;;AACD,QAAMC,MAAM,GAAG,MAAMP,SAAS,CAAClB,UAAD,EAAaC,IAAb,CAA9B;;AACA,MAAI;AACF,WAAOyB,IAAI,CAACC,KAAL,CAAWF,MAAX,CAAP;AACD,GAFD,CAEE,OAAOjB,GAAP,EAAY;AACZA,IAAAA,GAAG,CAACO,OAAJ,GAAe,2CAA0CU,MAAO,EAAhE;AACA,UAAMjB,GAAN;AACD;AACF,CAjB0B,CAAtB;;;;AAuBA,MAAMoB,qBAAqB,GAAGT,gBAAEC,KAAF,CAOnC,OAAOpB,UAAP,EAAmB6B,IAAnB,EAAyBC,eAAzB,KACE,MAAMR,aAAa,CAACtB,UAAD,EAAa,CAC9B6B,IAD8B,EAE9B,SAF8B,EAG9B,UAH8B,EAI9B,OAJ8B,EAK9BC,eAL8B,CAAb,CARc,CAA9B","sourcesContent":["// @ts-check\n\nimport _ from 'lodash';\nimport {exec} from 'teen_process';\nimport {PROJECT_ROOT} from '../helpers';\nimport path from 'path';\n\n/**\n * Path to the `appium` executable. Sort of.\n */\nexport const EXECUTABLE = path.join(\n  PROJECT_ROOT,\n  'packages',\n  'appium',\n  'build',\n  'lib',\n  'main.js',\n);\n\n/**\n * Runs the `appium` executable with the given args.\n *\n * If the process exits with a nonzero code, the error will be a\n * @param {string} appiumHome - Path to `APPIUM_HOME`\n * @param {string[]} args - Args, including commands\n * @returns {Promise<TeenProcessExecResult>}\n */\nasync function run (appiumHome, args) {\n  const env = {\n    APPIUM_HOME: appiumHome,\n    PATH: process.env.PATH,\n  };\n  try {\n    /**\n     * @type {TeenProcessExecResult}\n     */\n    return await exec(process.execPath, [EXECUTABLE, ...args], {\n      cwd: PROJECT_ROOT,\n      env,\n    });\n  } catch (err) {\n    const {stdout, stderr} = /** @type {TeenProcessExecError} */(err);\n    /**\n     * @type {AppiumRunError}\n     */\n    const runErr = Object.assign(\n      err,\n      {\n        originalMessage: err.message,\n        message: `${stdout.trim()}\\n\\n${stderr.trim()}`,\n        command: `${process.execPath} ${EXECUTABLE} ${args.join(' ')}`,\n        env,\n        cwd: PROJECT_ROOT,\n      });\n    throw runErr;\n  }\n}\n\n/**\n * Runs the `appium` executable with the given args and returns contents of `STDOUT`.\n *\n * Do not use this when testing error output, as it will likely be in `STDERR`. Use {@link runAppiumRaw} instead.\n */\nexport const runAppium = _.curry(\n  /**\n   * @param {string} appiumHome - Path to `APPIUM_HOME`\n   * @param {string[]} args - Args, including commands\n   * @returns {Promise<string>} Contents of ``STDOUT``\n   */\n  async (appiumHome, args) => {\n    const {stdout} = await run(appiumHome, args);\n    return stdout;\n  },\n);\n\n/**\n * Runs the `appium` executable with the given args and returns the entire\n * {@link TeenProcessExecResult} object.\n */\nexport const runAppiumRaw = _.curry(\n  /**\n   * @param {string} appiumHome - Path to `APPIUM_HOME`\n   * @param {string[]} args - Args, including commands\n   * @returns {Promise<TeenProcessExecResult>} Raw result of `exec`\n   */\n  async (appiumHome, args) => await run(appiumHome, args),\n);\n\n/**\n * Runs the `appium` executable with the given args in JSON mode.\n * Will reject with the contents of `STDOUT` (which were to be pasred) if parsing into JSON fails.\n */\nexport const runAppiumJson = _.curry(\n  /**\n   * @param {string} appiumHome - Path to `APPIUM_HOME`\n   * @param {string[]} args - Args, including commands\n   * @returns {Promise<TeenProcessExecResult>} Raw result of `exec`\n   */\n  async (appiumHome, args) => {\n    if (!args.includes('--json')) {\n      args.push('--json');\n    }\n    const result = await runAppium(appiumHome, args);\n    try {\n      return JSON.parse(result);\n    } catch (err) {\n      err.message = `Error parsing JSON. Contents of STDOUT: ${result}`;\n      throw err;\n    }\n  },\n);\n\n/**\n * Given a path to a local extension, install it into `APPIUM_HOME` via CLI.\n */\nexport const installLocalExtension = _.curry(\n  /**\n   * @param {string} appiumHome\n   * @param {import('../../lib/ext-config-io').ExtensionType} type\n   * @param {string} pathToExtension\n   * @returns {Promise<object>}\n   */\n  async (appiumHome, type, pathToExtension) =>\n    await runAppiumJson(appiumHome, [\n      type,\n      'install',\n      '--source',\n      'local',\n      pathToExtension,\n    ]),\n);\n\n/**\n * Options for {@link runAppium}.\n * @private\n * @typedef {Object} RunAppiumOptions\n * @property {boolean} [raw] - Whether to return the raw output from `teen_process`\n */\n\n/**\n * Result from a non-zero-exit execution of `appium`\n * @typedef {Object} TeenProcessExecResult\n * @property {string} stdout - Stdout\n * @property {string} stderr - Stderr\n * @property {number?} code - Exit code\n */\n\n/**\n * Extra props `teen_process.exec` adds to its error objects\n * @typedef {Object} TeenProcessExecErrorProps\n * @property {string} stdout - STDOUT\n * @property {string} stderr - STDERR\n * @property {number?} code - Exit code\n */\n\n/**\n * Error thrown by `teen_process.exec`\n * @typedef {Error & TeenProcessExecErrorProps} TeenProcessExecError\n */\n\n/**\n * Error thrown by all of the functions in this file which execute `appium`.\n * @typedef {Error & AppiumRunErrorProps & TeenProcessExecErrorProps} AppiumRunError\n */\n\n/**\n * Wraps the error returned by {@link exec}.\n * @typedef {Object} AppiumRunErrorProps\n * @property {string} originalMessage - Original error message\n * @property {string} command - Command that was run\n * @property {string} env - Environment variables\n * @property {string} cwd - Current working directory\n */\n"],"file":"test/cli/cli-helpers.js","sourceRoot":"../../.."}
86
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["test/cli/cli-helpers.js"],"names":["EXECUTABLE","path","join","PROJECT_ROOT","run","appiumHome","args","env","APPIUM_HOME","PATH","process","execPath","cwd","err","stdout","stderr","runErr","Object","assign","originalMessage","message","trim","command","runAppium","_","curry","runAppiumRaw","runAppiumJson","includes","push","result","JSON","parse","installLocalExtension","type","pathToExtension"],"mappings":";;;;;;;;;;;AAEA;;AACA;;AACA;;AACA;;AAKO,MAAMA,UAAU,GAAGC,cAAKC,IAAL,CACxBC,qBADwB,EAExB,UAFwB,EAGxB,QAHwB,EAIxB,OAJwB,EAKxB,KALwB,EAMxB,SANwB,CAAnB;;;;AAiBP,eAAeC,GAAf,CAAoBC,UAApB,EAAgCC,IAAhC,EAAsC;AACpC,QAAMC,GAAG,GAAG;AACVC,IAAAA,WAAW,EAAEH,UADH;AAEVI,IAAAA,IAAI,EAAEC,OAAO,CAACH,GAAR,CAAYE;AAFR,GAAZ;;AAIA,MAAI;AAIF,WAAO,MAAM,wBAAKC,OAAO,CAACC,QAAb,EAAuB,CAACX,UAAD,EAAa,GAAGM,IAAhB,CAAvB,EAA8C;AACzDM,MAAAA,GAAG,EAAET,qBADoD;AAEzDI,MAAAA;AAFyD,KAA9C,CAAb;AAID,GARD,CAQE,OAAOM,GAAP,EAAY;AACZ,UAAM;AAACC,MAAAA,MAAD;AAASC,MAAAA;AAAT,QAAuDF,GAA7D;AAIA,UAAMG,MAAM,GAAGC,MAAM,CAACC,MAAP,CACbL,GADa,EAEb;AACEM,MAAAA,eAAe,EAAEN,GAAG,CAACO,OADvB;AAEEA,MAAAA,OAAO,EAAG,GAAEN,MAAM,CAACO,IAAP,EAAc,OAAMN,MAAM,CAACM,IAAP,EAAc,EAFhD;AAGEC,MAAAA,OAAO,EAAG,GAAEZ,OAAO,CAACC,QAAS,IAAGX,UAAW,IAAGM,IAAI,CAACJ,IAAL,CAAU,GAAV,CAAe,EAH/D;AAIEK,MAAAA,GAJF;AAKEK,MAAAA,GAAG,EAAET;AALP,KAFa,CAAf;AASA,UAAMa,MAAN;AACD;AACF;;AAOM,MAAMO,SAAS,GAAGC,gBAAEC,KAAF,CAMvB,OAAOpB,UAAP,EAAmBC,IAAnB,KAA4B;AAC1B,QAAM;AAACQ,IAAAA;AAAD,MAAW,MAAMV,GAAG,CAACC,UAAD,EAAaC,IAAb,CAA1B;AACA,SAAOQ,MAAP;AACD,CATsB,CAAlB;;;;AAgBA,MAAMY,YAAY,GAAGF,gBAAEC,KAAF,CAM1B,OAAOpB,UAAP,EAAmBC,IAAnB,KAA4B,MAAMF,GAAG,CAACC,UAAD,EAAaC,IAAb,CANX,CAArB;;;;AAaA,MAAMqB,aAAa,GAAGH,gBAAEC,KAAF,CAM3B,OAAOpB,UAAP,EAAmBC,IAAnB,KAA4B;AAC1B,MAAI,CAACA,IAAI,CAACsB,QAAL,CAAc,QAAd,CAAL,EAA8B;AAC5BtB,IAAAA,IAAI,CAACuB,IAAL,CAAU,QAAV;AACD;;AACD,QAAMC,MAAM,GAAG,MAAMP,SAAS,CAAClB,UAAD,EAAaC,IAAb,CAA9B;;AACA,MAAI;AACF,WAAOyB,IAAI,CAACC,KAAL,CAAWF,MAAX,CAAP;AACD,GAFD,CAEE,OAAOjB,GAAP,EAAY;AACZA,IAAAA,GAAG,CAACO,OAAJ,GAAe,2CAA0CU,MAAO,EAAhE;AACA,UAAMjB,GAAN;AACD;AACF,CAjB0B,CAAtB;;;;AAuBA,MAAMoB,qBAAqB,GAAGT,gBAAEC,KAAF,CAOnC,OAAOpB,UAAP,EAAmB6B,IAAnB,EAAyBC,eAAzB,KACE,MAAMR,aAAa,CAACtB,UAAD,EAAa,CAC9B6B,IAD8B,EAE9B,SAF8B,EAG9B,UAH8B,EAI9B,OAJ8B,EAK9BC,eAL8B,CAAb,CARc,CAA9B","sourcesContent":["// @ts-check\n\nimport _ from 'lodash';\nimport {exec} from 'teen_process';\nimport {PROJECT_ROOT} from '../helpers';\nimport path from 'path';\n\n/**\n * Path to the `appium` executable. Sort of.\n */\nexport const EXECUTABLE = path.join(\n  PROJECT_ROOT,\n  'packages',\n  'appium',\n  'build',\n  'lib',\n  'main.js',\n);\n\n/**\n * Runs the `appium` executable with the given args.\n *\n * If the process exits with a nonzero code, the error will be a\n * @param {string} appiumHome - Path to `APPIUM_HOME`\n * @param {string[]} args - Args, including commands\n * @returns {Promise<TeenProcessExecResult>}\n */\nasync function run (appiumHome, args) {\n  const env = {\n    APPIUM_HOME: appiumHome,\n    PATH: process.env.PATH,\n  };\n  try {\n    /**\n     * @type {TeenProcessExecResult}\n     */\n    return await exec(process.execPath, [EXECUTABLE, ...args], {\n      cwd: PROJECT_ROOT,\n      env,\n    });\n  } catch (err) {\n    const {stdout, stderr} = /** @type {TeenProcessExecError} */(err);\n    /**\n     * @type {AppiumRunError}\n     */\n    const runErr = Object.assign(\n      err,\n      {\n        originalMessage: err.message,\n        message: `${stdout.trim()}\\n\\n${stderr.trim()}`,\n        command: `${process.execPath} ${EXECUTABLE} ${args.join(' ')}`,\n        env,\n        cwd: PROJECT_ROOT,\n      });\n    throw runErr;\n  }\n}\n\n/**\n * Runs the `appium` executable with the given args and returns contents of `STDOUT`.\n *\n * Do not use this when testing error output, as it will likely be in `STDERR`. Use {@link runAppiumRaw} instead.\n */\nexport const runAppium = _.curry(\n  /**\n   * @param {string} appiumHome - Path to `APPIUM_HOME`\n   * @param {string[]} args - Args, including commands\n   * @returns {Promise<string>} Contents of ``STDOUT``\n   */\n  async (appiumHome, args) => {\n    const {stdout} = await run(appiumHome, args);\n    return stdout;\n  },\n);\n\n/**\n * Runs the `appium` executable with the given args and returns the entire\n * {@link TeenProcessExecResult} object.\n */\nexport const runAppiumRaw = _.curry(\n  /**\n   * @param {string} appiumHome - Path to `APPIUM_HOME`\n   * @param {string[]} args - Args, including commands\n   * @returns {Promise<TeenProcessExecResult>} Raw result of `exec`\n   */\n  async (appiumHome, args) => await run(appiumHome, args),\n);\n\n/**\n * Runs the `appium` executable with the given args in JSON mode.\n * Will reject with the contents of `STDOUT` (which were to be pasred) if parsing into JSON fails.\n */\nexport const runAppiumJson = _.curry(\n  /**\n   * @param {string} appiumHome - Path to `APPIUM_HOME`\n   * @param {string[]} args - Args, including commands\n   * @returns {Promise<TeenProcessExecResult>} Raw result of `exec`\n   */\n  async (appiumHome, args) => {\n    if (!args.includes('--json')) {\n      args.push('--json');\n    }\n    const result = await runAppium(appiumHome, args);\n    try {\n      return JSON.parse(result);\n    } catch (err) {\n      err.message = `Error parsing JSON. Contents of STDOUT: ${result}`;\n      throw err;\n    }\n  },\n);\n\n/**\n * Given a path to a local extension, install it into `APPIUM_HOME` via CLI.\n */\nexport const installLocalExtension = _.curry(\n  /**\n   * @param {string} appiumHome\n   * @param {import('../../lib/ext-config-io').ExtensionType} type\n   * @param {string} pathToExtension\n   * @returns {Promise<object>}\n   */\n  async (appiumHome, type, pathToExtension) =>\n    await runAppiumJson(appiumHome, [\n      type,\n      'install',\n      '--source',\n      'local',\n      pathToExtension,\n    ]),\n);\n\n/**\n * Options for {@link runAppium}.\n * @private\n * @typedef {Object} RunAppiumOptions\n * @property {boolean} [raw] - Whether to return the raw output from `teen_process`\n */\n\n\n/**\n * Error thrown by all of the functions in this file which execute `appium`.\n * @typedef {Error & AppiumRunErrorProps & import('../../lib/cli/npm').TeenProcessExecErrorProps} AppiumRunError\n */\n\n/**\n * Wraps the error returned by {@link exec}.\n * @typedef {Object} AppiumRunErrorProps\n * @property {string} originalMessage - Original error message\n * @property {string} command - Command that was run\n * @property {string} env - Environment variables\n * @property {string} cwd - Current working directory\n */\n"],"file":"test/cli/cli-helpers.js","sourceRoot":"../../.."}
package/lib/appium.js CHANGED
@@ -216,6 +216,12 @@ class AppiumDriver extends BaseDriver {
216
216
 
217
217
  // This assignment is required for correct web sockets functionality inside the driver
218
218
  driverInstance.server = this.server;
219
+
220
+ // Drivers/plugins might also want to know where they are hosted
221
+ driverInstance.serverHost = this.args.address;
222
+ driverInstance.serverPort = this.args.port;
223
+ driverInstance.serverPath = this.args.basePath;
224
+
219
225
  try {
220
226
  runningDriversData = await this.curSessionDataForDriver(InnerDriver);
221
227
  } catch (e) {
package/lib/cli/npm.js CHANGED
@@ -1,3 +1,5 @@
1
+ // @ts-check
2
+
1
3
  import path from 'path';
2
4
  import semver from 'semver';
3
5
  import { exec } from 'teen_process';
@@ -8,10 +10,23 @@ const LINK_LOCKFILE = '.appium.link.lock';
8
10
 
9
11
  export default class NPM {
10
12
 
13
+ /**
14
+ * @param {string} appiumHome
15
+ */
11
16
  constructor (appiumHome) {
17
+ /** @type {string} */
12
18
  this.appiumHome = appiumHome;
13
19
  }
14
20
 
21
+ /**
22
+ * Execute `npm` with given args.
23
+ *
24
+ * If the process exits with a nonzero code, the contents of `STDOUT` and `STDERR` will be in the
25
+ * `message` of the {@link TeenProcessExecError} rejected.
26
+ * @param {string} cmd
27
+ * @param {string[]} args
28
+ * @param {{ json?: boolean; cwd?: string; lockFile?: string; }} opts
29
+ */
15
30
  async exec (cmd, args, opts, execOpts = {}) {
16
31
  let { cwd, json, lockFile } = opts;
17
32
  if (!cwd) {
@@ -33,7 +48,7 @@ export default class NPM {
33
48
 
34
49
  args.unshift(cmd);
35
50
  if (json) {
36
- args.push('-json');
51
+ args.push('--json');
37
52
  }
38
53
  const npmCmd = system.isWindows() ? 'npm.cmd' : 'npm';
39
54
  let runner = async () => await exec(npmCmd, args, execOpts);
@@ -42,27 +57,38 @@ export default class NPM {
42
57
  const _runner = runner;
43
58
  runner = async () => await acquireLock(_runner);
44
59
  }
45
- const {stdout, stderr, code} = await runner();
46
- const ret = {stdout, stderr, code, json: null};
47
60
 
48
- if (json) {
61
+ let ret;
62
+ try {
63
+ const {stdout, stderr, code} = await runner();
64
+ ret = /** @type {TeenProcessExecResult} */({stdout, stderr, code});
49
65
  // if possible, parse NPM's json output. During NPM install 3rd-party
50
66
  // packages can write to stdout, so sometimes the json output can't be
51
67
  // guaranteed to be parseable
52
68
  try {
53
69
  ret.json = JSON.parse(stdout);
54
70
  } catch (ign) {}
71
+ } catch (e) {
72
+ const {stdout, stderr, code} = /** @type {TeenProcessExecError} */(e);
73
+ const err = new Error(`npm command '${cmd} ${args.join(' ')}' failed with code ${code}.\n\nSTDOUT:\n${stdout.trim()}\n\nSTDERR:\n${stderr.trim()}`);
74
+ throw err;
55
75
  }
56
-
57
76
  return ret;
58
77
  }
59
78
 
79
+ /**
80
+ * @param {string} pkg
81
+ */
60
82
  async getLatestVersion (pkg) {
61
83
  return (await this.exec('view', [pkg, 'dist-tags'], {
62
84
  json: true
63
- })).json.latest;
85
+ })).json?.latest;
64
86
  }
65
87
 
88
+ /**
89
+ * @param {string} pkg
90
+ * @param {string} curVersion
91
+ */
66
92
  async getLatestSafeUpgradeVersion (pkg, curVersion) {
67
93
  const allVersions = (await this.exec('view', [pkg, 'versions'], {
68
94
  json: true
@@ -115,6 +141,11 @@ export default class NPM {
115
141
  return safeUpgradeVer;
116
142
  }
117
143
 
144
+ /**
145
+ *
146
+ * @param {{pkgDir: string, pkgName: string, pkgVer?: string}} param0
147
+ * @returns {Promise<import('type-fest').PackageJson>}
148
+ */
118
149
  async installPackage ({pkgDir, pkgName, pkgVer}) {
119
150
  const res = await this.exec('install', [
120
151
  '--no-save',
@@ -148,6 +179,9 @@ export default class NPM {
148
179
  }
149
180
  }
150
181
 
182
+ /**
183
+ * @param {string} pkgPath
184
+ */
151
185
  async linkPackage (pkgPath) {
152
186
  // from the path alone we don't know the npm package name, so we need to
153
187
  // look in package.json
@@ -180,6 +214,10 @@ export default class NPM {
180
214
  }
181
215
  }
182
216
 
217
+ /**
218
+ * @param {string} pkgDir
219
+ * @param {string} pkg
220
+ */
183
221
  async uninstallPackage (pkgDir, pkg) {
184
222
  await this.exec('uninstall', [pkg], {
185
223
  cwd: pkgDir,
@@ -187,3 +225,27 @@ export default class NPM {
187
225
  });
188
226
  }
189
227
  }
228
+
229
+
230
+ /**
231
+ * Result from a non-zero-exit execution of `appium`
232
+ * @typedef {Object} TeenProcessExecResult
233
+ * @property {string} stdout - Stdout
234
+ * @property {string} stderr - Stderr
235
+ * @property {number?} code - Exit code
236
+ * @property {any} json - JSON parsed from stdout
237
+ */
238
+
239
+ /**
240
+ * Extra props `teen_process.exec` adds to its error objects
241
+ * @typedef {Object} TeenProcessExecErrorProps
242
+ * @property {string} stdout - STDOUT
243
+ * @property {string} stderr - STDERR
244
+ * @property {number?} code - Exit code
245
+ */
246
+
247
+
248
+ /**
249
+ * Error thrown by `teen_process.exec`
250
+ * @typedef {Error & TeenProcessExecErrorProps} TeenProcessExecError
251
+ */
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "appium",
3
- "version": "2.0.0-beta.22",
3
+ "version": "2.0.0-beta.23",
4
4
  "description": "Automation for Apps.",
5
5
  "keywords": [
6
6
  "automation",
@@ -43,12 +43,12 @@
43
43
  "zip-and-upload": "npm run zip && npm run upload"
44
44
  },
45
45
  "dependencies": {
46
- "@appium/base-driver": "^8.2.1",
46
+ "@appium/base-driver": "^8.2.2",
47
47
  "@appium/base-plugin": "1.8.0",
48
48
  "@appium/support": "^2.55.2",
49
49
  "@babel/runtime": "7.16.3",
50
50
  "@sidvind/better-ajv-errors": "0.9.2",
51
- "ajv": "8.8.1",
51
+ "ajv": "8.8.2",
52
52
  "ajv-formats": "2.1.1",
53
53
  "argparse": "2.0.1",
54
54
  "async-lock": "1.3.0",
@@ -79,5 +79,5 @@
79
79
  "tag": "next"
80
80
  },
81
81
  "homepage": "https://appium.io",
82
- "gitHead": "e36a70105880b845e99e2c8a958403f6253f45b6"
82
+ "gitHead": "280d409df6c02d36b4ffc4d02a95ab3f4649b08c"
83
83
  }