appium-ios-simulator 4.0.1 → 4.0.2

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.
@@ -486,16 +486,16 @@ class SimulatorXcode6 extends _events.EventEmitter {
486
486
  }
487
487
 
488
488
  async launchAndQuit(safari = false, startupTimeout = this.startupTimeout) {
489
- _logger.default.debug('Attempting to launch and quit the simulator, to create directory structure');
490
-
491
- _logger.default.debug(`Will launch with Safari? ${safari}`);
489
+ _logger.default.debug('Attempting to launch and quit the simulator to create the directory structure');
492
490
 
493
491
  await this.run({
494
492
  startupTimeout
495
493
  });
496
494
 
497
495
  if (safari) {
498
- await this.openUrl('http://www.appium.io');
496
+ _logger.default.debug('Spawning Safari browser in order to create the necessary file system items');
497
+
498
+ await (0, _utils.launchApp)(this.simctl, _utils.MOBILE_SAFARI_BUNDLE_ID);
499
499
  }
500
500
 
501
501
  try {
@@ -616,11 +616,11 @@ class SimulatorXcode6 extends _events.EventEmitter {
616
616
  _logger.default.debug('Deleting Safari apps from simulator');
617
617
 
618
618
  let dirs = [];
619
- dirs.push(await this.getAppDir('com.apple.mobilesafari'));
619
+ dirs.push(await this.getAppDir(_utils.MOBILE_SAFARI_BUNDLE_ID));
620
620
  let pv = await this.getPlatformVersion();
621
621
 
622
622
  if (pv >= 8) {
623
- dirs.push(await this.getAppDir('com.apple.mobilesafari', 'Bundle'));
623
+ dirs.push(await this.getAppDir(_utils.MOBILE_SAFARI_BUNDLE_ID, 'Bundle'));
624
624
  }
625
625
 
626
626
  let deletePromises = [];
@@ -645,7 +645,7 @@ class SimulatorXcode6 extends _events.EventEmitter {
645
645
 
646
646
  let libraryDir = _path.default.resolve(this.getDir(), 'Library');
647
647
 
648
- let safariRoot = await this.getAppDir('com.apple.mobilesafari');
648
+ let safariRoot = await this.getAppDir(_utils.MOBILE_SAFARI_BUNDLE_ID);
649
649
 
650
650
  if (!safariRoot) {
651
651
  _logger.default.info('Could not find Safari support directories to clean out old ' + 'data. Probably there is nothing to clean out');
@@ -655,7 +655,7 @@ class SimulatorXcode6 extends _events.EventEmitter {
655
655
 
656
656
  let safariLibraryDir = _path.default.resolve(safariRoot, 'Library');
657
657
 
658
- let filesToDelete = ['Caches/Snapshots/com.apple.mobilesafari', 'Caches/com.apple.mobilesafari/*', 'Caches/com.apple.WebAppCache/*', 'Caches/com.apple.WebKit.Networking/*', 'Caches/com.apple.WebKit.WebContent/*', 'Image Cache/*', 'WebKit/com.apple.mobilesafari/*', 'WebKit/GeolocationSites.plist', 'WebKit/LocalStorage/*.*', 'Safari/*', 'Cookies/*.binarycookies', 'Caches/com.apple.UIStatusBar/*', 'Caches/com.apple.keyboards/images/*', 'Caches/com.apple.Safari.SafeBrowsing/*', '../tmp/com.apple.mobilesafari/*'];
658
+ let filesToDelete = [`Caches/Snapshots/${_utils.MOBILE_SAFARI_BUNDLE_ID}`, `Caches/${_utils.MOBILE_SAFARI_BUNDLE_ID}/*`, 'Caches/com.apple.WebAppCache/*', 'Caches/com.apple.WebKit.Networking/*', 'Caches/com.apple.WebKit.WebContent/*', 'Image Cache/*', `WebKit/${_utils.MOBILE_SAFARI_BUNDLE_ID}/*`, 'WebKit/GeolocationSites.plist', 'WebKit/LocalStorage/*.*', 'Safari/*', 'Cookies/*.binarycookies', 'Caches/com.apple.UIStatusBar/*', 'Caches/com.apple.keyboards/images/*', 'Caches/com.apple.Safari.SafeBrowsing/*', `../tmp/${_utils.MOBILE_SAFARI_BUNDLE_ID}/*`];
659
659
  let deletePromises = [];
660
660
 
661
661
  for (let file of filesToDelete) {
@@ -1141,4 +1141,4 @@ var _default = SimulatorXcode6;
1141
1141
  exports.default = _default;require('source-map-support').install();
1142
1142
 
1143
1143
 
1144
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/simulator-xcode-6.js"],"names":["STARTUP_TIMEOUT","EXTRA_STARTUP_TIME","UI_CLIENT_ACCESS_GUARD","AsyncLock","UI_CLIENT_BUNDLE_ID","SPRINGBOARD_BUNDLE_ID","BOOT_COMPLETED_EVENT","SimulatorXcode6","EventEmitter","constructor","udid","xcodeVersion","String","simctl","Simctl","_platformVersion","keychainPath","path","resolve","getDir","simulatorApp","appDataBundlePaths","isFreshFiles","extraStartupTime","calendar","Calendar","permissions","Permissions","uiClientBundleId","devicesSetPath","value","getUIClientPid","stdout","e","isNaN","parseInt","trim","log","debug","isUIClientRunning","_","isNull","startupTimeout","getPlatformVersion","sdk","stat","getRootDir","home","process","env","HOME","getLogDir","installApp","app","isAppInstalled","bundleId","appFile","appDirs","getAppDirs","length","getUserInstalledBundleIdsByBundleName","bundleName","rootUserAppDir","buildBundlePathMap","bundleIds","isEmpty","userAppDirPath","Object","entries","fs","readdir","find","file","extname","toLowerCase","infoPlistPath","exists","infoPlist","plist","parsePlistFile","CFBundleName","push","err","warn","message","getAppDir","id","subDir","isFresh","applicationList","pathBundlePair","dir","appFiles","glob","match","readBundleId","metadata","settings","read","MCMMetadataIdentifier","bundlePathDirs","bundlePathPairs","reduce","bundleMap","bundlePath","deviceArr","toPairs","getDevices","device","files","pv","map","s","existences","f","hasAccess","fresh","compact","isRunning","getEnv","isShutdown","includes","stderr","waitForBoot","bootedIndicator","getBootedIndicatorString","tailLogsUntil","B","delay","emit","indicator","platformVersion","startUIClient","opts","cloneDeep","defaultsDeep","scaleFactor","args","name","formattedDeviceName","replace","argumentName","info","join","timeout","run","assign","isServerRunning","timer","timing","Timer","start","shutdown","getDuration","asSeconds","toFixed","clean","endSimulatorDaemon","eraseDevice","scrubCustomApp","appBundleId","cleanCustomApp","scrub","deletePromises","rimraf","relRmPath","rmPath","all","dirs","data","bundle","undefined","src","launchAndQuit","safari","openUrl","Error","launchctlCmd","stopCmd","removeCmd","waitMs","intervalMs","shutdownDevice","bind","delete","deleteDevice","updateSettings","updates","updateLocationSettings","authorized","setReduceMotion","reduceMotion","setAppearance","getAppearance","updateSafariSettings","updated","updateSafariUserSettings","updateSafariGlobalSettings","updateLocale","language","locale","calendarFormat","deleteSafari","cleanSafari","keepPrefs","libraryDir","safariRoot","safariLibraryDir","filesToDelete","removeApp","moveBuiltInApp","appName","appPath","newAppPath","copyFile","url","SAFARI_BOOTED_INDICATOR","SAFARI_STARTUP_TIMEOUT","clearCaches","folderNames","cachesRoot","itemsToRemove","x","filter","isDirectory","util","pluralize","timeoutMs","simLog","enableCalendarAccess","bundleID","disableCalendarAccess","hasCalendarAccess","_activateWindow","pid","executeUIClientScript","appleScript","windowActivationScript","resultScript","acquire","errorAndThrow","isBiometricEnrolled","output","isString","enrollBiometric","isEnabled","sendBiometricMatch","shouldMatch","dismissDatabaseAlert","increase","button","backupKeychains","backupPath","tempDir","prefix","Math","floor","random","toString","substring","suffix","zipArgs","sep","_keychainsBackupPath","unlink","restoreKeychains","excludePatterns","split","plistPath","getLaunchDaemonsRoot","spawnProcess","unzipArgs","flatMap","clearKeychains","ps","servicesMatch","exec","result","pattern","group","trimEnd","setPermission","permission","setPermissions","permissionsMapping","JSON","stringify","setAccess","getPermission","serviceName","getAccess","addCertificate","payload","truncate","pushNotification","devRoot","_getDeviceStringPlatformVersion","reqVersion","xcode","getMaxIOSSDK","_getDeviceStringVersionString","_getDeviceStringConfigFix","getDeviceString","deviceName","forceIphone","forceIpad","logOpts","isiPhone","indexOf","iosDeviceString","test","CONFIG_FIX","configFix","getWebInspectorSocket","cmd","fn","extensions","prototype"],"mappings":"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","sourcesContent":["import path from 'path';\nimport { default as xcode, getPath as getXcodePath } from 'appium-xcode';\nimport log from './logger';\nimport { fs, tempDir, mkdirp, plist, timing, util } from '@appium/support';\nimport B from 'bluebird';\nimport _ from 'lodash';\nimport AsyncLock from 'async-lock';\nimport {\n  safeRimRaf, getDeveloperRoot, installSSLCert, hasSSLCert, activateApp,\n} from './utils.js';\nimport { asyncmap, retryInterval, waitForCondition, retry } from 'asyncbox';\nimport * as settings from './settings';\nimport { exec } from 'teen_process';\nimport { tailUntil } from './tail-until.js';\nimport extensions from './extensions/index';\nimport { EventEmitter } from 'events';\nimport Calendar from './calendar';\nimport Permissions from './permissions';\nimport Simctl from 'node-simctl';\n\n\nconst STARTUP_TIMEOUT = 60 * 1000;\nconst EXTRA_STARTUP_TIME = 2000;\nconst UI_CLIENT_ACCESS_GUARD = new AsyncLock();\nconst UI_CLIENT_BUNDLE_ID = 'com.apple.iphonesimulator';\nconst SPRINGBOARD_BUNDLE_ID = 'com.apple.SpringBoard';\n\n/*\n * This event is emitted as soon as iOS Simulator\n * has finished booting and it is ready to accept xcrun commands.\n * The event handler is called after 'run' method is completed\n * for Xcode 7 and older and is only useful in Xcode 8+,\n * since one can start doing stuff (for example install/uninstall an app) in parallel\n * with Simulator UI startup, which shortens session startup time.\n */\nconst BOOT_COMPLETED_EVENT = 'bootCompleted';\n\n\nclass SimulatorXcode6 extends EventEmitter {\n\n  /**\n   * Constructs the object with the `udid` and version of Xcode. Use the exported `getSimulator(udid)` method instead.\n   *\n   * @param {string} udid - The Simulator ID.\n   * @param {object} xcodeVersion - The target Xcode version in format {major, minor, build}.\n   */\n  constructor (udid, xcodeVersion) {\n    super();\n\n    this.udid = String(udid);\n    this.simctl = new Simctl({\n      udid: this.udid,\n    });\n    this.xcodeVersion = xcodeVersion;\n\n    // platformVersion cannot be found initially, since getting it has side effects for\n    // our logic for figuring out if a sim has been run\n    // it will be set when it is needed\n    this._platformVersion = null;\n\n    this.keychainPath = path.resolve(this.getDir(), 'Library', 'Keychains');\n    this.simulatorApp = 'iOS Simulator.app';\n\n    this.appDataBundlePaths = {};\n\n    // list of files to check for when seeing if a simulator is \"fresh\"\n    // (meaning it has never been booted).\n    // If these files are present, we assume it's been successfully booted\n    this.isFreshFiles = [\n      'Library/ConfigurationProfiles',\n      'Library/Cookies',\n      'Library/Preferences/.GlobalPreferences.plist',\n      'Library/Preferences/com.apple.springboard.plist',\n      'var/run/syslog.pid'\n    ];\n\n    // extra time to wait for simulator to be deemed booted\n    this.extraStartupTime = EXTRA_STARTUP_TIME;\n\n    this.calendar = new Calendar(xcodeVersion, this.getDir());\n    this.permissions = new Permissions(xcodeVersion, this.getDir(), this.udid);\n  }\n\n  /**\n   * @return {string} Bundle identifier of Simulator UI client.\n   */\n  get uiClientBundleId () {\n    return UI_CLIENT_BUNDLE_ID;\n  }\n\n  /**\n   * @return {?string} The full path to the devices set where the current simulator is located.\n   * `null` value means that the default path is used, which is usually `~/Library/Developer/CoreSimulator/Devices`\n   */\n  get devicesSetPath () {\n    return this.simctl.devicesSetPath;\n  }\n\n  /**\n   * Set the full path to the devices set. It is recommended to set this value\n   * once right after Simulator instance is created and to not change it during\n   * the instance lifecycle\n   *\n   * @param {?string} value The full path to the devices set root on the\n   * local file system\n   */\n  set devicesSetPath (value) {\n    this.simctl.devicesSetPath = value;\n  }\n\n  /**\n   * Retrieves the current process id of the UI client\n   *\n   * @return {?string} The process ID or null if the UI client is not running\n   */\n  async getUIClientPid () {\n    let stdout;\n    try {\n      ({stdout} = await exec('pgrep', ['-fn', `${this.simulatorApp}/Contents/MacOS/`]));\n    } catch (e) {\n      return null;\n    }\n    if (isNaN(parseInt(stdout, 10))) {\n      return null;\n    }\n    stdout = stdout.trim();\n    log.debug(`Got Simulator UI client PID: ${stdout}`);\n    return stdout;\n  }\n\n  /**\n   * Check the state of Simulator UI client.\n   *\n   * @return {boolean} True of if UI client is running or false otherwise.\n   */\n  async isUIClientRunning () {\n    return !_.isNull(await this.getUIClientPid());\n  }\n\n  /**\n   * How long to wait before throwing an error about Simulator startup timeout happened.\n   *\n   * @return {number} The number of milliseconds.\n   */\n  get startupTimeout () {\n    return STARTUP_TIMEOUT;\n  }\n\n  /**\n   * Get the platform version of the current Simulator.\n   *\n   * @return {string} SDK version, for example '8.3'.\n   */\n  async getPlatformVersion () {\n    if (!this._platformVersion) {\n      let {sdk} = await this.stat();\n      this._platformVersion = sdk;\n    }\n    return this._platformVersion;\n  }\n\n  /**\n   * Retrieve the full path to the directory where Simulator stuff is located.\n   *\n   * @return {string} The path string.\n   */\n  getRootDir () {\n    let home = process.env.HOME;\n    return path.resolve(home, 'Library', 'Developer', 'CoreSimulator', 'Devices');\n  }\n\n  /**\n   * Retrieve the full path to the directory where Simulator applications data is located.\n   *\n   * @return {string} The path string.\n   */\n  getDir () {\n    return path.resolve(this.getRootDir(), this.udid, 'data');\n  }\n\n  /**\n   * Retrieve the full path to the directory where Simulator logs are stored.\n   *\n   * @return {string} The path string.\n   */\n  getLogDir () {\n    let home = process.env.HOME;\n    return path.resolve(home, 'Library', 'Logs', 'CoreSimulator', this.udid);\n  }\n\n  /**\n   * Install valid .app package on Simulator.\n   *\n   * @param {string} app - The path to the .app package.\n   */\n  async installApp (app) {\n    return await this.simctl.installApp(app);\n  }\n\n  /**\n   * Verify whether the particular application is installed on Simulator.\n   *\n   * @param {string} bundleId - The bundle id of the application to be checked.\n   * @param {string} appFule - Application name minus \".app\" (for iOS 7.1)\n   * @return {boolean} True if the given application is installed\n   */\n  async isAppInstalled (bundleId, appFile = null) {\n    // `appFile` argument only necessary for iOS below version 8\n    let appDirs = await this.getAppDirs(appFile, bundleId);\n    return appDirs.length !== 0;\n  }\n\n  /**\n   * Returns user installed bundle ids which has 'bundleName' in their Info.Plist as 'CFBundleName'\n   * @param {string} bundleId - The bundle id of the application to be checked.\n   * @return {array<string>} - The list of bundle ids which have 'bundleName'\n   */\n  async getUserInstalledBundleIdsByBundleName (bundleName) {\n    const rootUserAppDir = await this.buildBundlePathMap('Bundle');\n    const bundleIds = [];\n    if (_.isEmpty(rootUserAppDir)) {\n      return bundleIds;\n    }\n\n    for (const [bundleId, userAppDirPath] of Object.entries(rootUserAppDir)) {\n      const appFile = (await fs.readdir(userAppDirPath)).find(\n        (file) => path.extname(file).toLowerCase() === '.app');\n      const infoPlistPath = path.resolve(userAppDirPath, appFile, 'Info.plist');\n      if (!await fs.exists(infoPlistPath)) {\n        continue;\n      }\n      try {\n        const infoPlist = await plist.parsePlistFile(infoPlistPath, false);\n        if (infoPlist.CFBundleName === bundleName) {\n          bundleIds.push(bundleId);\n        }\n      } catch (err) {\n        log.warn(`Failed to read plist ${infoPlistPath}. Original error '${err.message}'`);\n        continue;\n      }\n    }\n    log.debug(`The simulator has '${bundleIds.length}' bundles which have '${bundleName}' as their 'CFBundleName':`);\n    for (const bundleId of bundleIds) {\n      log.debug(`    '${bundleId}'`);\n    }\n    return bundleIds;\n  }\n\n  /**\n   * Retrieve the directory for a particular application's data.\n   *\n   * @param {string} id - Either a bundleId (e.g., com.apple.mobilesafari) or, for iOS 7.1, the app name without `.app` (e.g., MobileSafari)\n   * @param {string} subdir - The sub-directory we expect to be within the application directory. Defaults to \"Data\".\n   * @return {string} The root application folder.\n   */\n  async getAppDir (id, subDir = 'Data') {\n    this.appDataBundlePaths[subDir] = this.appDataBundlePaths[subDir] || {};\n    if (_.isEmpty(this.appDataBundlePaths[subDir]) && !await this.isFresh()) {\n      this.appDataBundlePaths[subDir] = await this.buildBundlePathMap(subDir);\n    }\n    return this.appDataBundlePaths[subDir][id];\n  }\n\n  /**\n   * The xcode 6 simulators are really annoying, and bury the main app\n   * directories inside directories just named with Hashes.\n   * This function finds the proper directory by traversing all of them\n   * and reading a metadata plist (Mobile Container Manager) to get the\n   * bundle id.\n   *\n   * @param {string} subdir - The sub-directory we expect to be within the application directory. Defaults to \"Data\".\n   * @return {object} The list of path-bundle pairs to an object where bundleIds are mapped to paths.\n   */\n  async buildBundlePathMap (subDir = 'Data') {\n    log.debug('Building bundle path map');\n    let applicationList;\n    let pathBundlePair;\n    if (await this.getPlatformVersion() === '7.1') {\n      // apps available\n      //   Web.app,\n      //   WebViewService.app,\n      //   MobileSafari.app,\n      //   WebContentAnalysisUI.app,\n      //   DDActionsService.app,\n      //   StoreKitUIService.app\n      applicationList = path.resolve(this.getDir(), 'Applications');\n      pathBundlePair = async (dir) => {\n        dir = path.resolve(applicationList, dir);\n        let appFiles = await fs.glob(`${dir}/*.app`);\n        let bundleId = appFiles[0].match(/.*\\/(.*)\\.app/)[1];\n        return {path: dir, bundleId};\n      };\n    } else {\n      applicationList = path.resolve(this.getDir(), 'Containers', subDir, 'Application');\n      // given a directory, find the plist file and pull the bundle id from it\n      let readBundleId = async (dir) => {\n        let plist = path.resolve(dir, '.com.apple.mobile_container_manager.metadata.plist');\n        let metadata = await settings.read(plist);\n        return metadata.MCMMetadataIdentifier;\n      };\n      // given a directory, return the path and bundle id associated with it\n      pathBundlePair = async (dir) => {\n        dir = path.resolve(applicationList, dir);\n        let bundleId = await readBundleId(dir);\n        return {path: dir, bundleId};\n      };\n    }\n\n    if (!await fs.exists(applicationList)) {\n      log.warn(`No directory path '${applicationList}'`);\n      return {};\n    }\n\n    let bundlePathDirs = await fs.readdir(applicationList);\n    let bundlePathPairs = await asyncmap(bundlePathDirs, async function (dir) {\n      return await pathBundlePair(dir);\n    }, false);\n\n    // reduce the list of path-bundle pairs to an object where bundleIds are mapped to paths\n    return bundlePathPairs.reduce((bundleMap, bundlePath) => {\n      bundleMap[bundlePath.bundleId] = bundlePath.path;\n      return bundleMap;\n    }, {});\n  }\n\n  /**\n   * Get the state and specifics of this sim.\n   *\n   * @return {object} Simulator stats mapping, for example:\n   * { name: 'iPhone 4s',\n   *   udid: 'C09B34E5-7DCB-442E-B79C-AB6BC0357417',\n   *   state: 'Shutdown',\n   *   sdk: '8.3'\n   * }\n   */\n  async stat () {\n    for (let [sdk, deviceArr] of _.toPairs(await this.simctl.getDevices())) {\n      for (let device of deviceArr) {\n        if (device.udid === this.udid) {\n          device.sdk = sdk;\n          return device;\n        }\n      }\n    }\n\n    return {};\n  }\n\n  /**\n   * This is a best-bet heuristic for whether or not a sim has been booted\n   * before. We usually want to start a simulator to \"warm\" it up, have\n   * Xcode populate it with plists for us to manipulate before a real\n   * test run.\n   *\n   * @return {boolean} True if the current Simulator has never been started before\n   */\n  async isFresh () {\n    // if the following files don't exist, it hasn't been booted.\n    // THIS IS NOT AN EXHAUSTIVE LIST\n    let files = this.isFreshFiles;\n\n    let pv = await this.getPlatformVersion();\n    if (pv !== '7.1') {\n      files.push('Library/Preferences/com.apple.Preferences.plist');\n    } else {\n      files.push('Applications');\n    }\n\n    const dir = this.getDir();\n    files = files.map((s) => path.resolve(dir, s));\n\n    const existences = await asyncmap(files, async (f) => await fs.hasAccess(f));\n    const fresh = _.compact(existences).length !== files.length;\n    log.debug(`Checking whether simulator has been run before: ${fresh ? 'no' : 'yes'}`);\n\n    return fresh;\n  }\n\n  /**\n   * Retrieves the state of the current Simulator. One should distinguish the\n   * states of Simulator UI and the Simulator itself.\n   *\n   * @return {boolean} True if the current Simulator is running.\n   */\n  async isRunning () {\n    try {\n      await this.simctl.getEnv('dummy');\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Checks if the simulator is in shutdown state.\n   * This method is necessary, because Simulator might also be\n   * in the transitional Shutting Down state right after the `shutdown`\n   * command has been issued.\n   *\n   * @return {boolean} True if the current Simulator is shut down.\n   */\n  async isShutdown () {\n    try {\n      await this.simctl.getEnv('dummy');\n      return false;\n    } catch (e) {\n      return _.includes(e.stderr, 'Current state: Shutdown');\n    }\n  }\n\n  /**\n   * Verify whether the Simulator booting is completed and/or wait for it\n   * until the timeout expires.\n   *\n   * @param {number} startupTimeout - the number of milliseconds to wait until booting is completed.\n   * @emits BOOT_COMPLETED_EVENT if the current Simulator is ready to accept simctl commands, like 'install'.\n   */\n  async waitForBoot (startupTimeout) {\n    // wait for the simulator to boot\n    // waiting for the simulator status to be 'booted' isn't good enough\n    // it claims to be booted way before finishing loading\n    // let's tail the simulator system log until we see a magic line (this.bootedIndicator)\n    let bootedIndicator = await this.getBootedIndicatorString();\n    await this.tailLogsUntil(bootedIndicator, startupTimeout);\n\n    // so sorry, but we should wait another two seconds, just to make sure we've really started\n    // we can't look for another magic log line, because they seem to be app-dependent (not system dependent)\n    log.debug(`Waiting an extra ${this.extraStartupTime}ms for the simulator to really finish booting`);\n    await B.delay(this.extraStartupTime);\n    log.debug('Done waiting extra time for simulator');\n\n    this.emit(BOOT_COMPLETED_EVENT);\n  }\n\n  /**\n   * Returns a magic string, which, if present in logs, reflects the fact that simulator booting has been completed.\n   *\n   * @return {string} The magic log string.\n   */\n  async getBootedIndicatorString () {\n    let indicator;\n    let platformVersion = await this.getPlatformVersion();\n    switch (platformVersion) {\n      case '7.1':\n      case '8.1':\n      case '8.2':\n      case '8.3':\n      case '8.4':\n        indicator = 'profiled: Service starting...';\n        break;\n      case '9.0':\n      case '9.1':\n      case '9.2':\n      case '9.3':\n        indicator = 'System app \"com.apple.springboard\" finished startup';\n        break;\n      case '10.0':\n        indicator = 'Switching to keyboard';\n        break;\n      default:\n        log.warn(`No boot indicator case for platform version '${platformVersion}'`);\n        indicator = 'no boot indicator string available';\n    }\n    return indicator;\n  }\n\n\n  /**\n   * @typedef {Object} SimulatorOptions\n   * @property {?string} scaleFactor [null] - Defines the window scale value for the UI client window for the current Simulator.\n   *   Equals to null by default, which keeps the current scale unchanged.\n   *   It should be one of ['1.0', '0.75', '0.5', '0.33', '0.25'].\n   * @property {number} startupTimeout [60000] - Number of milliseconds to wait until Simulator booting\n   *   process is completed. The default timeout will be used if not set explicitly.\n   */\n\n  /**\n   * Start the Simulator UI client with the given arguments\n   * @param {SimulatorOptions} opts - Simulator startup options\n   */\n  async startUIClient (opts = {}) {\n    opts = _.cloneDeep(opts);\n    _.defaultsDeep(opts, {\n      scaleFactor: null,\n      startupTimeout: this.startupTimeout,\n    });\n\n    const simulatorApp = path.resolve(await getXcodePath(), 'Applications', this.simulatorApp);\n    const args = [\n      '-Fn', simulatorApp,\n      '--args', '-CurrentDeviceUDID', this.udid,\n    ];\n\n    if (opts.scaleFactor) {\n      const {name} = await this.stat();\n      const formattedDeviceName = name.replace(/\\s+/g, '-');\n      const argumentName = `-SimulatorWindowLastScale-com.apple.CoreSimulator.SimDeviceType.${formattedDeviceName}`;\n      args.push(argumentName, opts.scaleFactor);\n    }\n\n    log.info(`Starting Simulator UI with command: open ${args.join(' ')}`);\n    try {\n      await exec('open', args, {timeout: opts.startupTimeout});\n    } catch (err) {\n      if (!(err.stdout || '').includes('-10825') && !(err.stderr || '').includes('-10825')) {\n        throw err;\n      }\n      log.warn(`Error while opening UI: ${err.stdout || err.stderr}. Continuing`);\n    }\n  }\n\n  /**\n   * Executes given Simulator with options. The Simulator will not be restarted if\n   * it is already running.\n   *\n   * @param {object} opts - One or more of available Simulator options.\n   *   See {#startUIClient(opts)} documentation for more details on other supported keys.\n   */\n  async run (opts = {}) {\n    opts = Object.assign({\n      startupTimeout: this.startupTimeout,\n    }, opts);\n    const isServerRunning = await this.isRunning();\n    const isUIClientRunning = await this.isUIClientRunning();\n    if (isServerRunning && isUIClientRunning) {\n      log.info(`Both Simulator with UDID ${this.udid} and the UI client are currently running`);\n      return;\n    }\n    const timer = new timing.Timer().start();\n    try {\n      await this.shutdown();\n    } catch (err) {\n      log.warn(`Error on Simulator shutdown: ${err.message}`);\n    }\n    await this.startUIClient(opts);\n\n    await this.waitForBoot(opts.startupTimeout);\n    log.info(`Simulator with UDID ${this.udid} booted in ${timer.getDuration().asSeconds.toFixed(3)}s`);\n  }\n\n  // TODO keep keychains\n  /**\n   * Reset the current Simulator to the clean state.\n   */\n  async clean () {\n    await this.endSimulatorDaemon();\n    log.info(`Cleaning simulator ${this.udid}`);\n    await this.simctl.eraseDevice(10000);\n  }\n\n  /**\n   * Scrub (delete the preferences and changed files) the particular application on Simulator.\n   *\n   * @param {string} appFile - Application name minus \".app\".\n   * @param {string} appBundleId - Bundle identifier of the application.\n   */\n  async scrubCustomApp (appFile, appBundleId) {\n    return await this.cleanCustomApp(appFile, appBundleId, true);\n  }\n\n  /**\n   * Clean/scrub the particular application on Simulator.\n   *\n   * @param {string} appFile - Application name minus \".app\".\n   * @param {string} appBundleId - Bundle identifier of the application.\n   * @param {boolean} scrub - If `scrub` is false, we want to clean by deleting the app and all\n   *   files associated with it. If `scrub` is true, we just want to delete the preferences and\n   *   changed files.\n   */\n  async cleanCustomApp (appFile, appBundleId, scrub = false) {\n    log.debug(`Cleaning app data files for '${appFile}', '${appBundleId}'`);\n    if (!scrub) {\n      log.debug(`Deleting app altogether`);\n    }\n\n    // get the directories to be deleted\n    let appDirs = await this.getAppDirs(appFile, appBundleId, scrub);\n\n    if (appDirs.length === 0) {\n      log.debug('Could not find app directories to delete. It is probably not installed');\n      return;\n    }\n\n    let deletePromises = [];\n\n    for (let dir of appDirs) {\n      log.debug(`Deleting directory: '${dir}'`);\n      deletePromises.push(fs.rimraf(dir));\n    }\n\n    if (await this.getPlatformVersion() >= 8) {\n      let relRmPath = `Library/Preferences/${appBundleId}.plist`;\n      let rmPath = path.resolve(this.getRootDir(), relRmPath);\n      log.debug(`Deleting file: '${rmPath}'`);\n      deletePromises.push(fs.rimraf(rmPath));\n    }\n\n    await B.all(deletePromises);\n  }\n\n  /**\n   * Retrieve paths to dirs where application data is stored. iOS 8+ stores app data in two places,\n   * and iOS 7.1 has only one directory\n   *\n   * @param {string} appFile - Application name minus \".app\".\n   * @param {string} appBundleId - Bundle identifier of the application.\n   * @param {boolean} scrub - The `Bundle` directory has the actual app in it. If we are just scrubbing,\n   *   we want this to stay. If we are cleaning we delete.\n   * @return {array<string>} Array of application data paths.\n   */\n  async getAppDirs (appFile, appBundleId, scrub = false) {\n    let dirs = [];\n    if (await this.getPlatformVersion() >= 8) {\n      let data = await this.getAppDir(appBundleId);\n      if (!data) return dirs; // eslint-disable-line curly\n\n      let bundle = !scrub ? await this.getAppDir(appBundleId, 'Bundle') : undefined;\n\n      for (let src of [data, bundle]) {\n        if (src) {\n          dirs.push(src);\n        }\n      }\n    } else {\n      let data = await this.getAppDir(appFile);\n      if (data) {\n        dirs.push(data);\n      }\n    }\n    return dirs;\n  }\n\n  /**\n   * Execute the Simulator in order to have the initial file structure created and shutdown it afterwards.\n   *\n   * @param {boolean} safari - Whether to execute mobile Safari after startup.\n   * @param {number} startupTimeout - How long to wait until Simulator booting is completed (in milliseconds).\n   */\n  async launchAndQuit (safari = false, startupTimeout = this.startupTimeout) {\n    log.debug('Attempting to launch and quit the simulator, to create directory structure');\n    log.debug(`Will launch with Safari? ${safari}`);\n\n    await this.run({startupTimeout});\n\n    if (safari) {\n      await this.openUrl('http://www.appium.io');\n    }\n\n    // wait for the system to create the files we will manipulate\n    // need quite a high retry number, in order to accommodate iOS 7.1\n    // locally, 7.1 averages 8.5 retries (from 6 - 12)\n    //          8 averages 0.6 retries (from 0 - 2)\n    //          9 averages 14 retries\n    try {\n      await retryInterval(60, 250, async () => {\n        if (await this.isFresh()) {\n          throw new Error('Simulator files not fully created. Waiting a bit');\n        }\n      });\n    } catch (err) {\n      log.warn(`Timeout waiting for simulator files to be created. Continuing`);\n    }\n\n    // and quit\n    await this.shutdown();\n  }\n\n  /**\n   * Looks for launchd daemons corresponding to the sim udid and tries to stop them cleanly\n   * This prevents xcrun simctl erase from hanging.\n   */\n  async endSimulatorDaemon () {\n    log.debug(`Killing any simulator daemons for ${this.udid}`);\n\n    let launchctlCmd = `launchctl list | grep ${this.udid} | cut -f 3 | xargs -n 1 launchctl`;\n    try {\n      let stopCmd = `${launchctlCmd} stop`;\n      await exec('bash', ['-c', stopCmd]);\n    } catch (err) {\n      log.warn(`Could not stop simulator daemons: ${err.message}`);\n      log.debug('Carrying on anyway!');\n    }\n    try {\n      let removeCmd = `${launchctlCmd} remove`;\n      await exec('bash', ['-c', removeCmd]);\n    } catch (err) {\n      log.warn(`Could not remove simulator daemons: ${err.message}`);\n      log.debug('Carrying on anyway!');\n    }\n    try {\n      // Waits 10 sec for the simulator launchd services to stop.\n      await waitForCondition(async () => {\n        let {stdout} = await exec('bash', ['-c',\n          `ps -e  | grep ${this.udid} | grep launchd_sim | grep -v bash | grep -v grep | awk {'print$1'}`]);\n        return stdout.trim().length === 0;\n      }, {waitMs: 10000, intervalMs: 500});\n    } catch (err) {\n      log.warn(`Could not end simulator daemon for ${this.udid}: ${err.message}`);\n      log.debug('Carrying on anyway!');\n    }\n  }\n\n  /**\n   * @typedef {Object} ShutdownOptions\n   * @property {?number|string} timeout The number of milliseconds to wait until\n   * Simulator is shut down completely. No wait happens if the timeout value is not set\n   */\n\n  /**\n   * Shut down the current Simulator.\n   *\n   * @param {?ShutdownOptions} opts\n   * @throws {Error} If Simulator fails to transition into Shutdown state after\n   * the given timeout\n   */\n  async shutdown (opts = {}) {\n    if (await this.isShutdown()) {\n      return;\n    }\n\n    await retryInterval(5, 500, this.simctl.shutdownDevice.bind(this.simctl));\n    const waitMs = parseInt(opts.timeout, 10);\n    if (waitMs > 0) {\n      try {\n        await waitForCondition(async () => await this.isShutdown(), {\n          waitMs,\n          intervalMs: 100,\n        });\n      } catch (err) {\n        throw new Error(`Simulator is not in 'Shutdown' state after ${waitMs}ms`);\n      }\n    }\n  }\n\n  /**\n   * Delete the particular Simulator from devices list\n   */\n  async delete () {\n    await this.simctl.deleteDevice();\n  }\n\n  /**\n   * Update the particular preference file with the given key/value pairs.\n   *\n   * @param {string} plist - The preferences file to update.\n   * @param {object} updates - The key/value pairs to update.\n   */\n  async updateSettings (plist, updates) {\n    return await settings.updateSettings(this, plist, updates);\n  }\n\n  /**\n   * Authorize/de-authorize location settings for a particular application.\n   *\n   * @param {string} bundleId - The application ID to update.\n   * @param {boolean} authorized - Whether or not to authorize.\n   */\n  async updateLocationSettings (bundleId, authorized) {\n    return await settings.updateLocationSettings(this, bundleId, authorized);\n  }\n\n  /**\n   * Enable/Disable reduce motion.\n   *\n   * @param {boolean} reduceMotion - Whether or not to enable it.\n   */\n  async setReduceMotion (reduceMotion = true) {\n    if (await this.isFresh()) {\n      await this.launchAndQuit(false, STARTUP_TIMEOUT);\n    }\n\n    await settings.setReduceMotion(this, reduceMotion);\n  }\n\n  /**\n   * Sets UI appearance style.\n   * This function can only be called on a booted simulator.\n   *\n   * @since Xcode SDK 11.4\n   */\n  async setAppearance (/* value */) { // eslint-disable-line require-await\n    throw new Error(`Xcode SDK '${this.xcodeVersion}' is too old to set UI appearance`);\n  }\n\n  /**\n   * Gets the current UI appearance style\n   * This function can only be called on a booted simulator.\n   *\n   * @since Xcode SDK 11.4\n   */\n  async getAppearance () { // eslint-disable-line require-await\n    throw new Error(`Xcode SDK '${this.xcodeVersion}' is too old to get UI appearance`);\n  }\n\n  /**\n   * Update settings for Safari.\n   *\n   * @param {object} updates - The hash of key/value pairs to update for Safari.\n   */\n  async updateSafariSettings (updates) {\n    let updated = await settings.updateSafariUserSettings(this, updates);\n    return await settings.updateSettings(this, 'mobileSafari', updates) || updated;\n  }\n\n  /**\n   * Update global settings for Safari.\n   *\n   * @param {object} updates - The hash of key/value pairs to update for Safari.\n   */\n  async updateSafariGlobalSettings (updates) {\n    return await settings.updateSafariGlobalSettings(this, updates);\n  }\n\n  /**\n   * Update the locale for the Simulator.\n   *\n   * @param {string} language - The language for the simulator. E.g., `\"fr_US\"`.\n   * @param {string} locale - The locale to set for the simulator. E.g., `\"en\"`.\n   * @param {string} calendarFormat - The format of the calendar.\n   */\n  async updateLocale (language, locale, calendarFormat) {\n    return await settings.updateLocale(this, language, locale, calendarFormat);\n  }\n\n  /**\n   * Completely delete mobile Safari application from the current Simulator.\n   */\n  async deleteSafari () {\n    log.debug('Deleting Safari apps from simulator');\n\n    let dirs = [];\n\n    // get the data directory\n    dirs.push(await this.getAppDir('com.apple.mobilesafari'));\n\n    let pv = await this.getPlatformVersion();\n    if (pv >= 8) {\n      // get the bundle directory\n      dirs.push(await this.getAppDir('com.apple.mobilesafari', 'Bundle'));\n    }\n\n    let deletePromises = [];\n    for (let dir of _.compact(dirs)) {\n      log.debug(`Deleting directory: '${dir}'`);\n      deletePromises.push(fs.rimraf(dir));\n    }\n    await B.all(deletePromises);\n  }\n\n  /**\n   * Clean up the directories for mobile Safari.\n   *\n   * @param {boolean} keepPrefs - Whether to keep Safari preferences from being deleted.\n   */\n  async cleanSafari (keepPrefs = true) {\n    log.debug('Cleaning mobile safari data files');\n    if (await this.isFresh()) {\n      log.info('Could not find Safari support directories to clean out old ' +\n               'data. Probably there is nothing to clean out');\n      return;\n    }\n\n    let libraryDir = path.resolve(this.getDir(), 'Library');\n    let safariRoot = await this.getAppDir('com.apple.mobilesafari');\n    if (!safariRoot) {\n      log.info('Could not find Safari support directories to clean out old ' +\n               'data. Probably there is nothing to clean out');\n      return;\n    }\n    let safariLibraryDir = path.resolve(safariRoot, 'Library');\n    let filesToDelete = [\n      'Caches/Snapshots/com.apple.mobilesafari',\n      'Caches/com.apple.mobilesafari/*',\n      'Caches/com.apple.WebAppCache/*',\n      'Caches/com.apple.WebKit.Networking/*',\n      'Caches/com.apple.WebKit.WebContent/*',\n      'Image Cache/*',\n      'WebKit/com.apple.mobilesafari/*',\n      'WebKit/GeolocationSites.plist',\n      'WebKit/LocalStorage/*.*',\n      'Safari/*',\n      'Cookies/*.binarycookies',\n      'Caches/com.apple.UIStatusBar/*',\n      'Caches/com.apple.keyboards/images/*',\n      'Caches/com.apple.Safari.SafeBrowsing/*',\n      '../tmp/com.apple.mobilesafari/*'\n    ];\n    let deletePromises = [];\n\n    for (let file of filesToDelete) {\n      deletePromises.push(fs.rimraf(path.resolve(libraryDir, file)));\n      deletePromises.push(fs.rimraf(path.resolve(safariLibraryDir, file)));\n    }\n\n    if (!keepPrefs) {\n      deletePromises.push(fs.rimraf(path.resolve(safariLibraryDir, 'Preferences/*.plist')));\n    }\n\n    await B.all(deletePromises);\n  }\n\n  /**\n   * Uninstall the given application from the current Simulator.\n   *\n   * @param {string} bundleId - The buindle ID of the application to be removed.\n   */\n  async removeApp (bundleId) {\n    await this.simctl.removeApp(bundleId);\n  }\n\n  /**\n   * Move a built-in application to a new place (actually, rename it).\n   *\n   * @param {string} appName - The name of the app to be moved.\n   * @param {string} appPath - The current path to the application.\n   * @param {string} newAppPath - The new path to the application.\n   *   If some application already exists by this path then it's going to be removed.\n   */\n  async moveBuiltInApp (appName, appPath, newAppPath) {\n    await safeRimRaf(newAppPath);\n    await fs.copyFile(appPath, newAppPath);\n    log.debug(`Copied '${appName}' to '${newAppPath}'`);\n\n    await fs.rimraf(appPath);\n    log.debug(`Temporarily deleted original app at '${appPath}'`);\n\n    return [newAppPath, appPath];\n  }\n\n  /**\n   * Open the given URL in mobile Safari browser.\n   * The browser will be started automatically if it is not running.\n   *\n   * @param {string} url - The URL to be opened.\n   */\n  async openUrl (url) {\n    const SAFARI_BOOTED_INDICATOR = 'MobileSafari[';\n    const SAFARI_STARTUP_TIMEOUT = 15 * 1000;\n    const EXTRA_STARTUP_TIME = 3 * 1000;\n\n    if (await this.isRunning()) {\n      await retry(5000, this.simctl.openUrl.bind(this.simctl), url);\n      await this.tailLogsUntil(SAFARI_BOOTED_INDICATOR, SAFARI_STARTUP_TIMEOUT);\n      // So sorry, but the logs have nothing else for Safari starting.. just delay a little bit\n      log.debug(`Safari started, waiting ${EXTRA_STARTUP_TIME}ms for it to fully start`);\n      await B.delay(EXTRA_STARTUP_TIME);\n      log.debug('Done waiting for Safari');\n      return;\n    } else {\n      throw new Error('Tried to open a url, but the Simulator is not Booted');\n    }\n  }\n\n  /**\n   * Perform Simulator caches cleanup.\n   *\n   * @param {...string} folderNames - The names of Caches subfolders to be cleaned.\n   *   Non-accessible/non-existing subfolders will be skipped.\n   *   All existing subfolders under Caches will be deleted if this parameter is omitted.\n   * @returns {number} The count of cleaned cache items.\n   *   Zero is returned if no items were matched for cleanup (either not accessible or not directories).\n   */\n  async clearCaches (...folderNames) {\n    const cachesRoot = path.resolve(this.getDir(), 'Library', 'Caches');\n    if (!(await fs.hasAccess(cachesRoot))) {\n      log.debug(`Caches root at '${cachesRoot}' does not exist or is not accessible. Nothing to do there`);\n      return 0;\n    }\n\n    let itemsToRemove = folderNames.length ? folderNames : (await fs.readdir(cachesRoot));\n    itemsToRemove = itemsToRemove.map((x) => path.resolve(cachesRoot, x));\n    if (folderNames.length) {\n      itemsToRemove = await B.filter(itemsToRemove, (x) => fs.hasAccess(x));\n    }\n    itemsToRemove = await B.filter(itemsToRemove, async (x) => (await fs.stat(x)).isDirectory());\n    if (!itemsToRemove.length) {\n      log.debug(`No Simulator cache items for cleanup were matched in '${cachesRoot}'`);\n      return 0;\n    }\n\n    log.debug(`Matched ${util.pluralize('simulator cache item', itemsToRemove.length, true)} ` +\n      `for cleanup: ${itemsToRemove}`);\n    try {\n      await B.all(itemsToRemove, (x) => fs.rimraf(x));\n    } catch (e) {\n      log.warn(`Got an exception while cleaning Simulator caches: ${e.message}`);\n    }\n    return itemsToRemove.length;\n  }\n\n  /**\n   * Blocks until the given indicater string appears in Simulator logs.\n   *\n   * @param {string} bootedIndicator - The magic string, which appears in logs after Simulator booting is completed.\n   * @param {number} timeoutMs - The maximumm number of milliseconds to wait for the string indicator presence.\n   * @returns {Promise} A promise that resolves when the ios simulator logs output a line matching `bootedIndicator`\n   * times out after timeoutMs\n   */\n  async tailLogsUntil (bootedIndicator, timeoutMs) {\n    let simLog = path.resolve(this.getLogDir(), 'system.log');\n\n    // we need to make sure log file exists before we can tail it\n    await retryInterval(200, 200, async () => {\n      let exists = await fs.exists(simLog);\n      if (!exists) {\n        throw new Error(`Could not find Simulator log: '${simLog}'`);\n      }\n    });\n\n    log.info(`Simulator log at '${simLog}'`);\n    log.info(`Tailing simulator logs until we encounter the string \"${bootedIndicator}\"`);\n    log.info(`We will time out after ${timeoutMs}ms`);\n    try {\n      await tailUntil(simLog, bootedIndicator, timeoutMs);\n    } catch (err) {\n      log.debug('Simulator startup timed out. Continuing anyway.');\n    }\n  }\n\n  /**\n   * Enable Calendar access for the given application.\n   *\n   * @param {string} bundleID - Bundle ID of the application, for which the access should be granted.\n   */\n  async enableCalendarAccess (bundleID) {\n    await this.calendar.enableCalendarAccess(bundleID);\n  }\n\n  /**\n   * Disable Calendar access for the given application.\n   *\n   * @param {string} bundleID - Bundle ID of the application, for which the access should be denied.\n   */\n  async disableCalendarAccess (bundleID) {\n    await this.calendar.disableCalendarAccess(bundleID);\n  }\n\n  /**\n   * Check whether the given application has access to Calendar.\n   *\n   * @return {boolean} True if the given application has the access.\n   */\n  async hasCalendarAccess (bundleID) {\n    return await this.calendar.hasCalendarAccess(bundleID);\n  }\n\n  /**\n   * Activates Simulator window.\n   *\n   * @private\n   * @returns {?string} If the method returns a string then it should be a valid Apple Script which\n   * is appended before each UI client command is executed. Otherwise the method should activate the window\n   * itself and return nothing.\n   */\n  async _activateWindow () { // eslint-disable-line require-await\n    const pid = await this.getUIClientPid();\n    if (pid) {\n      try {\n        return await activateApp(pid);\n      } catch (e) {\n        log.debug(e.stderr || e.message);\n      }\n    }\n    return `\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set frontmost to false\n          set frontmost to true\n        end tell\n      end tell\n    `;\n  }\n\n  /**\n   * Execute given Apple Script inside a critical section, so other\n   * sessions cannot influence the UI client at the same time.\n   *\n   * @param {string} appleScript - The valid Apple Script snippet to be executed.\n   * @return {string} The stdout output produced by the script.\n   * @throws {Error} If osascript tool returns non-zero exit code.\n   */\n  async executeUIClientScript (appleScript) {\n    const windowActivationScript = await this._activateWindow();\n    const resultScript = `${windowActivationScript ? windowActivationScript + '\\n' : ''}${appleScript}`;\n    log.debug(`Executing UI Apple Script on Simulator with UDID ${this.udid}: ${resultScript}`);\n    return await UI_CLIENT_ACCESS_GUARD.acquire(this.simulatorApp, async () => {\n      try {\n        const {stdout} = await exec('osascript', ['-e', resultScript]);\n        return stdout;\n      } catch (err) {\n        log.errorAndThrow(`Could not complete operation. Make sure Simulator UI is running and the parent Appium application (e. g. Appium.app or Terminal.app) ` +\n                          `is present in System Preferences > Security & Privacy > Privacy > Accessibility list. If the operation is still unsuccessful then ` +\n                          `it is not supported by this Simulator. ` +\n                          `Original error: ${err.message}`);\n      }\n    });\n  }\n\n  /**\n   * Get the current state of Biometric Enrollment feature.\n   *\n   * @returns {boolean} Either true or false\n   * @throws {Error} If Enrollment state cannot be determined\n   */\n  async isBiometricEnrolled () {\n    const output = await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"Touch ID Enrolled\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          set isChecked to (value of attribute \"AXMenuItemMarkChar\" of dstMenuItem) is \"✓\"\n        end tell\n      end tell\n    `);\n    log.debug(`Touch ID enrolled state: ${output}`);\n    return _.isString(output) && output.trim() === 'true';\n  }\n\n  /**\n   * Enrolls biometric (TouchId, FaceId) feature testing in Simulator UI client.\n   *\n   * @param {boolean} isEnabled - Defines whether biometric state is enabled/disabled\n   * @throws {Error} If the enrolled state cannot be changed\n   */\n  async enrollBiometric (isEnabled = true) {\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"Touch ID Enrolled\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          set isChecked to (value of attribute \"AXMenuItemMarkChar\" of dstMenuItem) is \"✓\"\n          if ${isEnabled ? 'not ' : ''}isChecked then\n            click dstMenuItem\n          end if\n        end tell\n      end tell\n    `);\n  }\n\n  /**\n   * Sends a notification to match/not match the touch id.\n   *\n   * @param {?boolean} shouldMatch [true] - Set it to true or false in order to emulate\n   * matching/not matching the corresponding biometric\n   */\n  async sendBiometricMatch (shouldMatch = true) {\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"${shouldMatch ? 'Matching' : 'Non-matching'}\" of menu 1 of menu item \"Simulate Finger Touch\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          click dstMenuItem\n        end tell\n      end tell\n    `);\n  }\n\n  /**\n   * Execute a special Apple script, which clicks the particular button on Database alert.\n   *\n   * @param {boolean} increase - Click the button with 'Increase' title on the alert if this\n   *   parameter is true. The 'Cancel' button will be clicked otherwise.\n   */\n  async dismissDatabaseAlert (increase = true) {\n    let button = increase ? 'Increase' : 'Cancel';\n    log.debug(`Attempting to dismiss database alert with '${button}' button`);\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          click button \"${button}\" of window 1\n        end tell\n      end tell\n    `);\n  }\n\n  //region Keychains Interaction\n  /**\n   * Create the backup of keychains folder.\n   * The previously created backup will be automatically\n   * deleted if this method was called twice in a row without\n   * `restoreKeychains` being invoked.\n   *\n   * @returns {boolean} True if the backup operation was successfull.\n   */\n  async backupKeychains () {\n    if (!await fs.exists(this.keychainPath)) {\n      return false;\n    }\n\n    const backupPath = await tempDir.path({\n      prefix: `keychains_backup_${Math.floor((1 + Math.random()) * 0x10000).toString(16).substring(1)}`,\n      suffix: '.zip',\n    });\n    const zipArgs = [\n      '-r', backupPath,\n      `${this.keychainPath}${path.sep}`\n    ];\n    log.debug(`Creating keychains backup with 'zip ${zipArgs.join(' ')}' command`);\n    await exec('zip', zipArgs);\n    if (_.isString(this._keychainsBackupPath) && await fs.exists(this._keychainsBackupPath)) {\n      await fs.unlink(this._keychainsBackupPath);\n    }\n    this._keychainsBackupPath = backupPath;\n    return true;\n  }\n\n  /**\n   * Restore the previsouly created keychains backup.\n   *\n   * @param {?string|Array<string>} excludePatterns - The list\n   * of file name patterns to be excluded from restore. The format\n   * of each item should be the same as '-x' option format for\n   * 'unzip' utility. This can also be a comma-separated string,\n   * which is going be transformed into a list automatically,\n   * for example: '*.db*,blabla.sqlite'\n   * @returns {boolean} If the restore opration was successful.\n   * @throws {Error} If there is no keychains backup available for restore.\n   */\n  async restoreKeychains (excludePatterns = []) {\n    if (!_.isString(this._keychainsBackupPath) || !await fs.exists(this._keychainsBackupPath)) {\n      throw new Error(`The keychains backup archive does not exist. ` +\n                      `Are you sure it was created before?`);\n    }\n\n    if (_.isString(excludePatterns)) {\n      excludePatterns = excludePatterns.split(',').map((x) => x.trim());\n    }\n    const isServerRunning = await this.isRunning();\n    let plistPath;\n    if (isServerRunning) {\n      plistPath = path.resolve(await this.getLaunchDaemonsRoot(), 'com.apple.securityd.plist');\n      if (!await fs.exists(plistPath)) {\n        throw new Error(`Cannot clear keychains because '${plistPath}' does not exist`);\n      }\n      await this.simctl.spawnProcess(['launchctl', 'unload', plistPath]);\n    }\n    try {\n      await fs.rimraf(this.keychainPath);\n      await mkdirp(this.keychainPath);\n      const unzipArgs = [\n        '-o', this._keychainsBackupPath,\n        ...(_.flatMap(excludePatterns.map((x) => ['-x', x]))),\n        '-d', '/'\n      ];\n      log.debug(`Restoring keychains with 'unzip ${unzipArgs.join(' ')}' command`);\n      await exec('unzip', unzipArgs);\n      await fs.unlink(this._keychainsBackupPath);\n      this._keychainsBackupPath = null;\n    } finally {\n      if (isServerRunning && plistPath) {\n        await this.simctl.spawnProcess(['launchctl', 'load', plistPath]);\n      }\n    }\n    return true;\n  }\n\n  /**\n   * Clears Keychains for the particular simulator in runtime (there is no need to stop it).\n   *\n   * @throws {Error} If keychain cleanup has failed.\n   */\n  async clearKeychains () {\n    const plistPath = path.resolve(await this.getLaunchDaemonsRoot(), 'com.apple.securityd.plist');\n    if (!await fs.exists(plistPath)) {\n      throw new Error(`Cannot clear keychains because '${plistPath}' does not exist`);\n    }\n    await this.simctl.spawnProcess(['launchctl', 'unload', plistPath]);\n    try {\n      if (await fs.exists(this.keychainPath)) {\n        await fs.rimraf(this.keychainPath);\n        await mkdirp(this.keychainPath);\n      }\n    } finally {\n      await this.simctl.spawnProcess(['launchctl', 'load', plistPath]);\n    }\n  }\n\n  //endregion\n\n  /**\n   * @typedef {Object} ProcessInfo\n   * @property {number} pid The actual process identifier.\n   * Could be zero if the process is the system one.\n   * @property {?string} group The process group identifier.\n   * This could be `null` if the process is not a part of the\n   * particular group. For `normal` application processes the group\n   * name usually equals to `UIKitApplication`.\n   * @property {string} name The process name, for example\n   * `com.apple.Preferences`\n   */\n\n  /**\n   * Lists processes that are currently running on the given Simulator.\n   * The simulator must be in running state in order for this\n   * method to work properly.\n   *\n   * @return {Array<ProcessInfo>} The list of retrieved process\n   * information\n   * @throws {Error} if no process information could be retrieved.\n   */\n  async ps () {\n    const {stdout} = await this.simctl.spawnProcess([\n      'launchctl',\n      'print',\n      'system',\n    ]);\n\n    const servicesMatch = /^\\s*services\\s*=\\s*{([^}]+)/m.exec(stdout);\n    if (!servicesMatch) {\n      log.debug(stdout);\n      throw new Error(`The list of active processes cannot be retrieved`);\n    }\n    /*\n    Example match:\n        0     78 \tcom.apple.resourcegrabberd\n    82158      - \tcom.apple.assistant_service\n    82120      - \tcom.apple.nanoregistryd\n    82087      - \tcom.apple.notifyd\n    82264      - \tUIKitApplication:com.apple.Preferences[704b][rb-legacy]\n    */\n    const result = [];\n    const pattern = /^\\s*(\\d+)\\s+[\\d-]+\\s+([\\w\\-.]+:)?([\\w\\-.]+)/gm;\n    let match;\n    while ((match = pattern.exec(servicesMatch[1]))) {\n      result.push({\n        pid: parseInt(match[1], 10),\n        group: _.trimEnd(match[2], ':') || null,\n        name: match[3],\n      });\n    }\n    return result;\n  }\n\n  /**\n   * Sets the particular permission to the application bundle. See\n   * https://github.com/wix/AppleSimulatorUtils for more details on\n   * the available service names and statuses.\n   *\n   * @param {string} bundleId - Application bundle identifier.\n   * @param {string} permission - Service name to be set.\n   * @param {string} value - The desired status for the service.\n   * @throws {Error} If there was an error while changing permission.\n   */\n  async setPermission (bundleId, permission, value) {\n    await this.setPermissions(bundleId, {[permission]: value});\n  }\n\n  /**\n   * Sets the permissions for the particular application bundle.\n   *\n   * @param {string} bundleId - Application bundle identifier.\n   * @param {Object} permissionsMapping - A mapping where kays\n   * are service names and values are their corresponding status values.\n   * See https://github.com/wix/AppleSimulatorUtils\n   * for more details on available service names and statuses.\n   * @throws {Error} If there was an error while changing permissions.\n   */\n  async setPermissions (bundleId, permissionsMapping) {\n    log.debug(`Setting access for '${bundleId}': ` +\n      JSON.stringify(permissionsMapping, null, 2));\n    await this.permissions.setAccess(bundleId, permissionsMapping);\n  }\n\n  /**\n   * Retrieves current permission status for the given application bundle.\n   *\n   * @param {string} bundleId - Application bundle identifier.\n   * @param {string} serviceName - One of available service names.\n   * @throws {Error} If there was an error while retrieving permissions.\n   */\n  async getPermission (bundleId, serviceName) {\n    const result = await this.permissions.getAccess(bundleId, serviceName);\n    log.debug(`Got ${serviceName} access status for '${bundleId}': ${result}`);\n    return result;\n  }\n\n  /**\n   * Adds the given certificate into the Trusted Root Store on the simulator.\n   * The simulator must be shut down in order for this method to work properly.\n   *\n   * @param {string} payload the content of the PEM certificate\n   * @returns {boolean} `true` if the certificate has been successfully installed\n   * or `false` if it has already been there\n   */\n  async addCertificate (payload, /* opts = {} */) {\n    if (await hasSSLCert(payload, this.udid)) {\n      log.info(`SSL certificate '${_.truncate(payload, {length: 20})}' already installed`);\n      return false;\n    }\n    log.info(`Installing SSL root certificate '${_.truncate(payload, {length: 20})}'`);\n    await installSSLCert(payload, this.udid);\n    return true;\n  }\n\n  /**\n   * Simulates push notification delivery\n   *\n   * @since Xcode SDK 11.4\n   */\n  async pushNotification (/* payload */) { // eslint-disable-line require-await\n    throw new Error(`Xcode SDK '${this.xcodeVersion}' is too old to push notifications`);\n  }\n\n  async getLaunchDaemonsRoot () {\n    const devRoot = await getDeveloperRoot();\n    return path.resolve(devRoot,\n      'Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator.sdk/System/Library/LaunchDaemons');\n  }\n\n  static async _getDeviceStringPlatformVersion (platformVersion) {\n    let reqVersion = platformVersion;\n    if (!reqVersion) {\n      reqVersion = await xcode.getMaxIOSSDK();\n      log.warn(`No platform version set. Using max SDK version: ${reqVersion}`);\n      // this will be a number, and possibly an integer (e.g., if max iOS SDK is 9)\n      // so turn it into a string and add a .0 if necessary\n      if (!_.isString(reqVersion)) {\n        reqVersion = (reqVersion % 1) ? String(reqVersion) : `${reqVersion}.0`;\n      }\n    }\n    return reqVersion;\n  }\n\n  // change the format in subclasses, as necessary\n  static async _getDeviceStringVersionString (platformVersion) {\n    let reqVersion = await this._getDeviceStringPlatformVersion(platformVersion);\n\n    return `(${reqVersion} Simulator)`;\n  }\n\n  // change the format in subclasses, as necessary\n  static _getDeviceStringConfigFix () {\n    // some devices need to be updated\n    return {\n      'iPad Simulator (7.1 Simulator)': 'iPad 2 (7.1 Simulator)',\n      'iPad Simulator (8.0 Simulator)': 'iPad 2 (8.0 Simulator)',\n      'iPad Simulator (8.1 Simulator)': 'iPad 2 (8.1 Simulator)',\n      'iPad Simulator (8.2 Simulator)': 'iPad 2 (8.2 Simulator)',\n      'iPad Simulator (8.3 Simulator)': 'iPad 2 (8.3 Simulator)',\n      'iPad Simulator (8.4 Simulator)': 'iPad 2 (8.4 Simulator)',\n      'iPhone Simulator (7.1 Simulator)': 'iPhone 5s (7.1 Simulator)',\n      'iPhone Simulator (8.4 Simulator)': 'iPhone 6 (8.4 Simulator)',\n      'iPhone Simulator (8.3 Simulator)': 'iPhone 6 (8.3 Simulator)',\n      'iPhone Simulator (8.2 Simulator)': 'iPhone 6 (8.2 Simulator)',\n      'iPhone Simulator (8.1 Simulator)': 'iPhone 6 (8.1 Simulator)',\n      'iPhone Simulator (8.0 Simulator)': 'iPhone 6 (8.0 Simulator)'\n    };\n  }\n\n  /**\n   * Takes a set of options and finds the correct device string in order for Instruments to\n   * identify the correct simulator.\n   *\n   * @param {object} opts - The options available are:\n   *   - `deviceName` - a name for the device. If the given device name starts with `=`, the name, less the equals sign, is returned.\n   *   - `platformVersion` - the version of iOS to use. Defaults to the current Xcode's maximum SDK version.\n   *   - `forceIphone` - force the configuration of the device string to iPhone. Defaults to `false`.\n   *   - `forceIpad` - force the configuration of the device string to iPad. Defaults to `false`.\n   *   If both `forceIphone` and `forceIpad` are true, the device will be forced to iPhone.\n   *\n   * @return {string} The found device string.\n   */\n  static async getDeviceString (opts) {\n    opts = Object.assign({}, {\n      deviceName: null,\n      platformVersion: null,\n      forceIphone: false,\n      forceIpad: false\n    }, opts);\n    let logOpts = {\n      deviceName: opts.deviceName,\n      platformVersion: opts.platformVersion,\n      forceIphone: opts.forceIphone,\n      forceIpad: opts.forceIpad\n    };\n    log.debug(`Getting device string from options: ${JSON.stringify(logOpts)}`);\n\n    // short circuit if we already have a device name\n    if ((opts.deviceName || '')[0] === '=') {\n      return opts.deviceName.substring(1);\n    }\n\n    let isiPhone = !!opts.forceIphone || !opts.forceIpad;\n\n    if (opts.deviceName) {\n      let device = opts.deviceName.toLowerCase();\n      if (device.indexOf('iphone') !== -1) {\n        isiPhone = true;\n      } else if (device.indexOf('ipad') !== -1) {\n        isiPhone = false;\n      }\n    }\n\n    let iosDeviceString = opts.deviceName || (isiPhone ? 'iPhone Simulator' : 'iPad Simulator');\n\n    // if someone passes in just \"iPhone\", make that \"iPhone Simulator\" to\n    // conform to all the logic below\n    if (/^(iPhone|iPad)$/.test(iosDeviceString)) {\n      iosDeviceString += ' Simulator';\n    }\n\n    // we support deviceName: \"iPhone Simulator\", and also want to support\n    // \"iPhone XYZ Simulator\", but these strings aren't in the device list.\n    // So, if someone sent in \"iPhone XYZ Simulator\", strip off \" Simulator\"\n    // in order to allow the default \"iPhone XYZ\" match\n    if (/[^(iPhone|iPad)] Simulator/.test(iosDeviceString)) {\n      iosDeviceString = iosDeviceString.replace(' Simulator', '');\n    }\n    iosDeviceString += ` ${await this._getDeviceStringVersionString(opts.platformVersion)}`;\n\n    let CONFIG_FIX = this._getDeviceStringConfigFix();\n\n    let configFix = CONFIG_FIX;\n    if (configFix[iosDeviceString]) {\n      iosDeviceString = configFix[iosDeviceString];\n      log.debug(`Fixing device. Changed from '${opts.deviceName}' ` +\n                `to '${iosDeviceString}'`);\n    }\n\n    log.debug(`Final device string is '${iosDeviceString}'`);\n    return iosDeviceString;\n  }\n\n  /**\n   * @return {?string} The full path to the simulator's WebInspector Unix Domain Socket\n   *   or `null` if there is no socket.\n   */\n  async getWebInspectorSocket () { // eslint-disable-line require-await\n    // there is no WebInspector socket for this version of Xcode\n    return null;\n  }\n}\n\nfor (let [cmd, fn] of _.toPairs(extensions)) {\n  SimulatorXcode6.prototype[cmd] = fn;\n}\n\nexport default SimulatorXcode6;\nexport { SimulatorXcode6, BOOT_COMPLETED_EVENT, SPRINGBOARD_BUNDLE_ID };\n"],"file":"lib/simulator-xcode-6.js","sourceRoot":"../.."}
1144
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/simulator-xcode-6.js"],"names":["STARTUP_TIMEOUT","EXTRA_STARTUP_TIME","UI_CLIENT_ACCESS_GUARD","AsyncLock","UI_CLIENT_BUNDLE_ID","SPRINGBOARD_BUNDLE_ID","BOOT_COMPLETED_EVENT","SimulatorXcode6","EventEmitter","constructor","udid","xcodeVersion","String","simctl","Simctl","_platformVersion","keychainPath","path","resolve","getDir","simulatorApp","appDataBundlePaths","isFreshFiles","extraStartupTime","calendar","Calendar","permissions","Permissions","uiClientBundleId","devicesSetPath","value","getUIClientPid","stdout","e","isNaN","parseInt","trim","log","debug","isUIClientRunning","_","isNull","startupTimeout","getPlatformVersion","sdk","stat","getRootDir","home","process","env","HOME","getLogDir","installApp","app","isAppInstalled","bundleId","appFile","appDirs","getAppDirs","length","getUserInstalledBundleIdsByBundleName","bundleName","rootUserAppDir","buildBundlePathMap","bundleIds","isEmpty","userAppDirPath","Object","entries","fs","readdir","find","file","extname","toLowerCase","infoPlistPath","exists","infoPlist","plist","parsePlistFile","CFBundleName","push","err","warn","message","getAppDir","id","subDir","isFresh","applicationList","pathBundlePair","dir","appFiles","glob","match","readBundleId","metadata","settings","read","MCMMetadataIdentifier","bundlePathDirs","bundlePathPairs","reduce","bundleMap","bundlePath","deviceArr","toPairs","getDevices","device","files","pv","map","s","existences","f","hasAccess","fresh","compact","isRunning","getEnv","isShutdown","includes","stderr","waitForBoot","bootedIndicator","getBootedIndicatorString","tailLogsUntil","B","delay","emit","indicator","platformVersion","startUIClient","opts","cloneDeep","defaultsDeep","scaleFactor","args","name","formattedDeviceName","replace","argumentName","info","join","timeout","run","assign","isServerRunning","timer","timing","Timer","start","shutdown","getDuration","asSeconds","toFixed","clean","endSimulatorDaemon","eraseDevice","scrubCustomApp","appBundleId","cleanCustomApp","scrub","deletePromises","rimraf","relRmPath","rmPath","all","dirs","data","bundle","undefined","src","launchAndQuit","safari","MOBILE_SAFARI_BUNDLE_ID","Error","launchctlCmd","stopCmd","removeCmd","waitMs","intervalMs","shutdownDevice","bind","delete","deleteDevice","updateSettings","updates","updateLocationSettings","authorized","setReduceMotion","reduceMotion","setAppearance","getAppearance","updateSafariSettings","updated","updateSafariUserSettings","updateSafariGlobalSettings","updateLocale","language","locale","calendarFormat","deleteSafari","cleanSafari","keepPrefs","libraryDir","safariRoot","safariLibraryDir","filesToDelete","removeApp","moveBuiltInApp","appName","appPath","newAppPath","copyFile","openUrl","url","SAFARI_BOOTED_INDICATOR","SAFARI_STARTUP_TIMEOUT","clearCaches","folderNames","cachesRoot","itemsToRemove","x","filter","isDirectory","util","pluralize","timeoutMs","simLog","enableCalendarAccess","bundleID","disableCalendarAccess","hasCalendarAccess","_activateWindow","pid","executeUIClientScript","appleScript","windowActivationScript","resultScript","acquire","errorAndThrow","isBiometricEnrolled","output","isString","enrollBiometric","isEnabled","sendBiometricMatch","shouldMatch","dismissDatabaseAlert","increase","button","backupKeychains","backupPath","tempDir","prefix","Math","floor","random","toString","substring","suffix","zipArgs","sep","_keychainsBackupPath","unlink","restoreKeychains","excludePatterns","split","plistPath","getLaunchDaemonsRoot","spawnProcess","unzipArgs","flatMap","clearKeychains","ps","servicesMatch","exec","result","pattern","group","trimEnd","setPermission","permission","setPermissions","permissionsMapping","JSON","stringify","setAccess","getPermission","serviceName","getAccess","addCertificate","payload","truncate","pushNotification","devRoot","_getDeviceStringPlatformVersion","reqVersion","xcode","getMaxIOSSDK","_getDeviceStringVersionString","_getDeviceStringConfigFix","getDeviceString","deviceName","forceIphone","forceIpad","logOpts","isiPhone","indexOf","iosDeviceString","test","CONFIG_FIX","configFix","getWebInspectorSocket","cmd","fn","extensions","prototype"],"mappings":"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","sourcesContent":["import path from 'path';\nimport { default as xcode, getPath as getXcodePath } from 'appium-xcode';\nimport log from './logger';\nimport { fs, tempDir, mkdirp, plist, timing, util } from '@appium/support';\nimport B from 'bluebird';\nimport _ from 'lodash';\nimport AsyncLock from 'async-lock';\nimport {\n  safeRimRaf, getDeveloperRoot, installSSLCert, hasSSLCert, activateApp,\n  MOBILE_SAFARI_BUNDLE_ID, launchApp\n} from './utils.js';\nimport { asyncmap, retryInterval, waitForCondition, retry } from 'asyncbox';\nimport * as settings from './settings';\nimport { exec } from 'teen_process';\nimport { tailUntil } from './tail-until.js';\nimport extensions from './extensions/index';\nimport { EventEmitter } from 'events';\nimport Calendar from './calendar';\nimport Permissions from './permissions';\nimport Simctl from 'node-simctl';\n\n\nconst STARTUP_TIMEOUT = 60 * 1000;\nconst EXTRA_STARTUP_TIME = 2000;\nconst UI_CLIENT_ACCESS_GUARD = new AsyncLock();\nconst UI_CLIENT_BUNDLE_ID = 'com.apple.iphonesimulator';\nconst SPRINGBOARD_BUNDLE_ID = 'com.apple.SpringBoard';\n\n/*\n * This event is emitted as soon as iOS Simulator\n * has finished booting and it is ready to accept xcrun commands.\n * The event handler is called after 'run' method is completed\n * for Xcode 7 and older and is only useful in Xcode 8+,\n * since one can start doing stuff (for example install/uninstall an app) in parallel\n * with Simulator UI startup, which shortens session startup time.\n */\nconst BOOT_COMPLETED_EVENT = 'bootCompleted';\n\n\nclass SimulatorXcode6 extends EventEmitter {\n\n  /**\n   * Constructs the object with the `udid` and version of Xcode. Use the exported `getSimulator(udid)` method instead.\n   *\n   * @param {string} udid - The Simulator ID.\n   * @param {object} xcodeVersion - The target Xcode version in format {major, minor, build}.\n   */\n  constructor (udid, xcodeVersion) {\n    super();\n\n    this.udid = String(udid);\n    this.simctl = new Simctl({\n      udid: this.udid,\n    });\n    this.xcodeVersion = xcodeVersion;\n\n    // platformVersion cannot be found initially, since getting it has side effects for\n    // our logic for figuring out if a sim has been run\n    // it will be set when it is needed\n    this._platformVersion = null;\n\n    this.keychainPath = path.resolve(this.getDir(), 'Library', 'Keychains');\n    this.simulatorApp = 'iOS Simulator.app';\n\n    this.appDataBundlePaths = {};\n\n    // list of files to check for when seeing if a simulator is \"fresh\"\n    // (meaning it has never been booted).\n    // If these files are present, we assume it's been successfully booted\n    this.isFreshFiles = [\n      'Library/ConfigurationProfiles',\n      'Library/Cookies',\n      'Library/Preferences/.GlobalPreferences.plist',\n      'Library/Preferences/com.apple.springboard.plist',\n      'var/run/syslog.pid'\n    ];\n\n    // extra time to wait for simulator to be deemed booted\n    this.extraStartupTime = EXTRA_STARTUP_TIME;\n\n    this.calendar = new Calendar(xcodeVersion, this.getDir());\n    this.permissions = new Permissions(xcodeVersion, this.getDir(), this.udid);\n  }\n\n  /**\n   * @return {string} Bundle identifier of Simulator UI client.\n   */\n  get uiClientBundleId () {\n    return UI_CLIENT_BUNDLE_ID;\n  }\n\n  /**\n   * @return {?string} The full path to the devices set where the current simulator is located.\n   * `null` value means that the default path is used, which is usually `~/Library/Developer/CoreSimulator/Devices`\n   */\n  get devicesSetPath () {\n    return this.simctl.devicesSetPath;\n  }\n\n  /**\n   * Set the full path to the devices set. It is recommended to set this value\n   * once right after Simulator instance is created and to not change it during\n   * the instance lifecycle\n   *\n   * @param {?string} value The full path to the devices set root on the\n   * local file system\n   */\n  set devicesSetPath (value) {\n    this.simctl.devicesSetPath = value;\n  }\n\n  /**\n   * Retrieves the current process id of the UI client\n   *\n   * @return {?string} The process ID or null if the UI client is not running\n   */\n  async getUIClientPid () {\n    let stdout;\n    try {\n      ({stdout} = await exec('pgrep', ['-fn', `${this.simulatorApp}/Contents/MacOS/`]));\n    } catch (e) {\n      return null;\n    }\n    if (isNaN(parseInt(stdout, 10))) {\n      return null;\n    }\n    stdout = stdout.trim();\n    log.debug(`Got Simulator UI client PID: ${stdout}`);\n    return stdout;\n  }\n\n  /**\n   * Check the state of Simulator UI client.\n   *\n   * @return {boolean} True of if UI client is running or false otherwise.\n   */\n  async isUIClientRunning () {\n    return !_.isNull(await this.getUIClientPid());\n  }\n\n  /**\n   * How long to wait before throwing an error about Simulator startup timeout happened.\n   *\n   * @return {number} The number of milliseconds.\n   */\n  get startupTimeout () {\n    return STARTUP_TIMEOUT;\n  }\n\n  /**\n   * Get the platform version of the current Simulator.\n   *\n   * @return {string} SDK version, for example '8.3'.\n   */\n  async getPlatformVersion () {\n    if (!this._platformVersion) {\n      let {sdk} = await this.stat();\n      this._platformVersion = sdk;\n    }\n    return this._platformVersion;\n  }\n\n  /**\n   * Retrieve the full path to the directory where Simulator stuff is located.\n   *\n   * @return {string} The path string.\n   */\n  getRootDir () {\n    let home = process.env.HOME;\n    return path.resolve(home, 'Library', 'Developer', 'CoreSimulator', 'Devices');\n  }\n\n  /**\n   * Retrieve the full path to the directory where Simulator applications data is located.\n   *\n   * @return {string} The path string.\n   */\n  getDir () {\n    return path.resolve(this.getRootDir(), this.udid, 'data');\n  }\n\n  /**\n   * Retrieve the full path to the directory where Simulator logs are stored.\n   *\n   * @return {string} The path string.\n   */\n  getLogDir () {\n    let home = process.env.HOME;\n    return path.resolve(home, 'Library', 'Logs', 'CoreSimulator', this.udid);\n  }\n\n  /**\n   * Install valid .app package on Simulator.\n   *\n   * @param {string} app - The path to the .app package.\n   */\n  async installApp (app) {\n    return await this.simctl.installApp(app);\n  }\n\n  /**\n   * Verify whether the particular application is installed on Simulator.\n   *\n   * @param {string} bundleId - The bundle id of the application to be checked.\n   * @param {string} appFule - Application name minus \".app\" (for iOS 7.1)\n   * @return {boolean} True if the given application is installed\n   */\n  async isAppInstalled (bundleId, appFile = null) {\n    // `appFile` argument only necessary for iOS below version 8\n    let appDirs = await this.getAppDirs(appFile, bundleId);\n    return appDirs.length !== 0;\n  }\n\n  /**\n   * Returns user installed bundle ids which has 'bundleName' in their Info.Plist as 'CFBundleName'\n   * @param {string} bundleId - The bundle id of the application to be checked.\n   * @return {array<string>} - The list of bundle ids which have 'bundleName'\n   */\n  async getUserInstalledBundleIdsByBundleName (bundleName) {\n    const rootUserAppDir = await this.buildBundlePathMap('Bundle');\n    const bundleIds = [];\n    if (_.isEmpty(rootUserAppDir)) {\n      return bundleIds;\n    }\n\n    for (const [bundleId, userAppDirPath] of Object.entries(rootUserAppDir)) {\n      const appFile = (await fs.readdir(userAppDirPath)).find(\n        (file) => path.extname(file).toLowerCase() === '.app');\n      const infoPlistPath = path.resolve(userAppDirPath, appFile, 'Info.plist');\n      if (!await fs.exists(infoPlistPath)) {\n        continue;\n      }\n      try {\n        const infoPlist = await plist.parsePlistFile(infoPlistPath, false);\n        if (infoPlist.CFBundleName === bundleName) {\n          bundleIds.push(bundleId);\n        }\n      } catch (err) {\n        log.warn(`Failed to read plist ${infoPlistPath}. Original error '${err.message}'`);\n        continue;\n      }\n    }\n    log.debug(`The simulator has '${bundleIds.length}' bundles which have '${bundleName}' as their 'CFBundleName':`);\n    for (const bundleId of bundleIds) {\n      log.debug(`    '${bundleId}'`);\n    }\n    return bundleIds;\n  }\n\n  /**\n   * Retrieve the directory for a particular application's data.\n   *\n   * @param {string} id - Either a bundleId (e.g., com.apple.mobilesafari) or, for iOS 7.1, the app name without `.app` (e.g., MobileSafari)\n   * @param {string} subdir - The sub-directory we expect to be within the application directory. Defaults to \"Data\".\n   * @return {string} The root application folder.\n   */\n  async getAppDir (id, subDir = 'Data') {\n    this.appDataBundlePaths[subDir] = this.appDataBundlePaths[subDir] || {};\n    if (_.isEmpty(this.appDataBundlePaths[subDir]) && !await this.isFresh()) {\n      this.appDataBundlePaths[subDir] = await this.buildBundlePathMap(subDir);\n    }\n    return this.appDataBundlePaths[subDir][id];\n  }\n\n  /**\n   * The xcode 6 simulators are really annoying, and bury the main app\n   * directories inside directories just named with Hashes.\n   * This function finds the proper directory by traversing all of them\n   * and reading a metadata plist (Mobile Container Manager) to get the\n   * bundle id.\n   *\n   * @param {string} subdir - The sub-directory we expect to be within the application directory. Defaults to \"Data\".\n   * @return {object} The list of path-bundle pairs to an object where bundleIds are mapped to paths.\n   */\n  async buildBundlePathMap (subDir = 'Data') {\n    log.debug('Building bundle path map');\n    let applicationList;\n    let pathBundlePair;\n    if (await this.getPlatformVersion() === '7.1') {\n      // apps available\n      //   Web.app,\n      //   WebViewService.app,\n      //   MobileSafari.app,\n      //   WebContentAnalysisUI.app,\n      //   DDActionsService.app,\n      //   StoreKitUIService.app\n      applicationList = path.resolve(this.getDir(), 'Applications');\n      pathBundlePair = async (dir) => {\n        dir = path.resolve(applicationList, dir);\n        let appFiles = await fs.glob(`${dir}/*.app`);\n        let bundleId = appFiles[0].match(/.*\\/(.*)\\.app/)[1];\n        return {path: dir, bundleId};\n      };\n    } else {\n      applicationList = path.resolve(this.getDir(), 'Containers', subDir, 'Application');\n      // given a directory, find the plist file and pull the bundle id from it\n      let readBundleId = async (dir) => {\n        let plist = path.resolve(dir, '.com.apple.mobile_container_manager.metadata.plist');\n        let metadata = await settings.read(plist);\n        return metadata.MCMMetadataIdentifier;\n      };\n      // given a directory, return the path and bundle id associated with it\n      pathBundlePair = async (dir) => {\n        dir = path.resolve(applicationList, dir);\n        let bundleId = await readBundleId(dir);\n        return {path: dir, bundleId};\n      };\n    }\n\n    if (!await fs.exists(applicationList)) {\n      log.warn(`No directory path '${applicationList}'`);\n      return {};\n    }\n\n    let bundlePathDirs = await fs.readdir(applicationList);\n    let bundlePathPairs = await asyncmap(bundlePathDirs, async function (dir) {\n      return await pathBundlePair(dir);\n    }, false);\n\n    // reduce the list of path-bundle pairs to an object where bundleIds are mapped to paths\n    return bundlePathPairs.reduce((bundleMap, bundlePath) => {\n      bundleMap[bundlePath.bundleId] = bundlePath.path;\n      return bundleMap;\n    }, {});\n  }\n\n  /**\n   * Get the state and specifics of this sim.\n   *\n   * @return {object} Simulator stats mapping, for example:\n   * { name: 'iPhone 4s',\n   *   udid: 'C09B34E5-7DCB-442E-B79C-AB6BC0357417',\n   *   state: 'Shutdown',\n   *   sdk: '8.3'\n   * }\n   */\n  async stat () {\n    for (let [sdk, deviceArr] of _.toPairs(await this.simctl.getDevices())) {\n      for (let device of deviceArr) {\n        if (device.udid === this.udid) {\n          device.sdk = sdk;\n          return device;\n        }\n      }\n    }\n\n    return {};\n  }\n\n  /**\n   * This is a best-bet heuristic for whether or not a sim has been booted\n   * before. We usually want to start a simulator to \"warm\" it up, have\n   * Xcode populate it with plists for us to manipulate before a real\n   * test run.\n   *\n   * @return {boolean} True if the current Simulator has never been started before\n   */\n  async isFresh () {\n    // if the following files don't exist, it hasn't been booted.\n    // THIS IS NOT AN EXHAUSTIVE LIST\n    let files = this.isFreshFiles;\n\n    let pv = await this.getPlatformVersion();\n    if (pv !== '7.1') {\n      files.push('Library/Preferences/com.apple.Preferences.plist');\n    } else {\n      files.push('Applications');\n    }\n\n    const dir = this.getDir();\n    files = files.map((s) => path.resolve(dir, s));\n\n    const existences = await asyncmap(files, async (f) => await fs.hasAccess(f));\n    const fresh = _.compact(existences).length !== files.length;\n    log.debug(`Checking whether simulator has been run before: ${fresh ? 'no' : 'yes'}`);\n\n    return fresh;\n  }\n\n  /**\n   * Retrieves the state of the current Simulator. One should distinguish the\n   * states of Simulator UI and the Simulator itself.\n   *\n   * @return {boolean} True if the current Simulator is running.\n   */\n  async isRunning () {\n    try {\n      await this.simctl.getEnv('dummy');\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Checks if the simulator is in shutdown state.\n   * This method is necessary, because Simulator might also be\n   * in the transitional Shutting Down state right after the `shutdown`\n   * command has been issued.\n   *\n   * @return {boolean} True if the current Simulator is shut down.\n   */\n  async isShutdown () {\n    try {\n      await this.simctl.getEnv('dummy');\n      return false;\n    } catch (e) {\n      return _.includes(e.stderr, 'Current state: Shutdown');\n    }\n  }\n\n  /**\n   * Verify whether the Simulator booting is completed and/or wait for it\n   * until the timeout expires.\n   *\n   * @param {number} startupTimeout - the number of milliseconds to wait until booting is completed.\n   * @emits BOOT_COMPLETED_EVENT if the current Simulator is ready to accept simctl commands, like 'install'.\n   */\n  async waitForBoot (startupTimeout) {\n    // wait for the simulator to boot\n    // waiting for the simulator status to be 'booted' isn't good enough\n    // it claims to be booted way before finishing loading\n    // let's tail the simulator system log until we see a magic line (this.bootedIndicator)\n    let bootedIndicator = await this.getBootedIndicatorString();\n    await this.tailLogsUntil(bootedIndicator, startupTimeout);\n\n    // so sorry, but we should wait another two seconds, just to make sure we've really started\n    // we can't look for another magic log line, because they seem to be app-dependent (not system dependent)\n    log.debug(`Waiting an extra ${this.extraStartupTime}ms for the simulator to really finish booting`);\n    await B.delay(this.extraStartupTime);\n    log.debug('Done waiting extra time for simulator');\n\n    this.emit(BOOT_COMPLETED_EVENT);\n  }\n\n  /**\n   * Returns a magic string, which, if present in logs, reflects the fact that simulator booting has been completed.\n   *\n   * @return {string} The magic log string.\n   */\n  async getBootedIndicatorString () {\n    let indicator;\n    let platformVersion = await this.getPlatformVersion();\n    switch (platformVersion) {\n      case '7.1':\n      case '8.1':\n      case '8.2':\n      case '8.3':\n      case '8.4':\n        indicator = 'profiled: Service starting...';\n        break;\n      case '9.0':\n      case '9.1':\n      case '9.2':\n      case '9.3':\n        indicator = 'System app \"com.apple.springboard\" finished startup';\n        break;\n      case '10.0':\n        indicator = 'Switching to keyboard';\n        break;\n      default:\n        log.warn(`No boot indicator case for platform version '${platformVersion}'`);\n        indicator = 'no boot indicator string available';\n    }\n    return indicator;\n  }\n\n\n  /**\n   * @typedef {Object} SimulatorOptions\n   * @property {?string} scaleFactor [null] - Defines the window scale value for the UI client window for the current Simulator.\n   *   Equals to null by default, which keeps the current scale unchanged.\n   *   It should be one of ['1.0', '0.75', '0.5', '0.33', '0.25'].\n   * @property {number} startupTimeout [60000] - Number of milliseconds to wait until Simulator booting\n   *   process is completed. The default timeout will be used if not set explicitly.\n   */\n\n  /**\n   * Start the Simulator UI client with the given arguments\n   * @param {SimulatorOptions} opts - Simulator startup options\n   */\n  async startUIClient (opts = {}) {\n    opts = _.cloneDeep(opts);\n    _.defaultsDeep(opts, {\n      scaleFactor: null,\n      startupTimeout: this.startupTimeout,\n    });\n\n    const simulatorApp = path.resolve(await getXcodePath(), 'Applications', this.simulatorApp);\n    const args = [\n      '-Fn', simulatorApp,\n      '--args', '-CurrentDeviceUDID', this.udid,\n    ];\n\n    if (opts.scaleFactor) {\n      const {name} = await this.stat();\n      const formattedDeviceName = name.replace(/\\s+/g, '-');\n      const argumentName = `-SimulatorWindowLastScale-com.apple.CoreSimulator.SimDeviceType.${formattedDeviceName}`;\n      args.push(argumentName, opts.scaleFactor);\n    }\n\n    log.info(`Starting Simulator UI with command: open ${args.join(' ')}`);\n    try {\n      await exec('open', args, {timeout: opts.startupTimeout});\n    } catch (err) {\n      if (!(err.stdout || '').includes('-10825') && !(err.stderr || '').includes('-10825')) {\n        throw err;\n      }\n      log.warn(`Error while opening UI: ${err.stdout || err.stderr}. Continuing`);\n    }\n  }\n\n  /**\n   * Executes given Simulator with options. The Simulator will not be restarted if\n   * it is already running.\n   *\n   * @param {object} opts - One or more of available Simulator options.\n   *   See {#startUIClient(opts)} documentation for more details on other supported keys.\n   */\n  async run (opts = {}) {\n    opts = Object.assign({\n      startupTimeout: this.startupTimeout,\n    }, opts);\n    const isServerRunning = await this.isRunning();\n    const isUIClientRunning = await this.isUIClientRunning();\n    if (isServerRunning && isUIClientRunning) {\n      log.info(`Both Simulator with UDID ${this.udid} and the UI client are currently running`);\n      return;\n    }\n    const timer = new timing.Timer().start();\n    try {\n      await this.shutdown();\n    } catch (err) {\n      log.warn(`Error on Simulator shutdown: ${err.message}`);\n    }\n    await this.startUIClient(opts);\n\n    await this.waitForBoot(opts.startupTimeout);\n    log.info(`Simulator with UDID ${this.udid} booted in ${timer.getDuration().asSeconds.toFixed(3)}s`);\n  }\n\n  // TODO keep keychains\n  /**\n   * Reset the current Simulator to the clean state.\n   */\n  async clean () {\n    await this.endSimulatorDaemon();\n    log.info(`Cleaning simulator ${this.udid}`);\n    await this.simctl.eraseDevice(10000);\n  }\n\n  /**\n   * Scrub (delete the preferences and changed files) the particular application on Simulator.\n   *\n   * @param {string} appFile - Application name minus \".app\".\n   * @param {string} appBundleId - Bundle identifier of the application.\n   */\n  async scrubCustomApp (appFile, appBundleId) {\n    return await this.cleanCustomApp(appFile, appBundleId, true);\n  }\n\n  /**\n   * Clean/scrub the particular application on Simulator.\n   *\n   * @param {string} appFile - Application name minus \".app\".\n   * @param {string} appBundleId - Bundle identifier of the application.\n   * @param {boolean} scrub - If `scrub` is false, we want to clean by deleting the app and all\n   *   files associated with it. If `scrub` is true, we just want to delete the preferences and\n   *   changed files.\n   */\n  async cleanCustomApp (appFile, appBundleId, scrub = false) {\n    log.debug(`Cleaning app data files for '${appFile}', '${appBundleId}'`);\n    if (!scrub) {\n      log.debug(`Deleting app altogether`);\n    }\n\n    // get the directories to be deleted\n    let appDirs = await this.getAppDirs(appFile, appBundleId, scrub);\n\n    if (appDirs.length === 0) {\n      log.debug('Could not find app directories to delete. It is probably not installed');\n      return;\n    }\n\n    let deletePromises = [];\n\n    for (let dir of appDirs) {\n      log.debug(`Deleting directory: '${dir}'`);\n      deletePromises.push(fs.rimraf(dir));\n    }\n\n    if (await this.getPlatformVersion() >= 8) {\n      let relRmPath = `Library/Preferences/${appBundleId}.plist`;\n      let rmPath = path.resolve(this.getRootDir(), relRmPath);\n      log.debug(`Deleting file: '${rmPath}'`);\n      deletePromises.push(fs.rimraf(rmPath));\n    }\n\n    await B.all(deletePromises);\n  }\n\n  /**\n   * Retrieve paths to dirs where application data is stored. iOS 8+ stores app data in two places,\n   * and iOS 7.1 has only one directory\n   *\n   * @param {string} appFile - Application name minus \".app\".\n   * @param {string} appBundleId - Bundle identifier of the application.\n   * @param {boolean} scrub - The `Bundle` directory has the actual app in it. If we are just scrubbing,\n   *   we want this to stay. If we are cleaning we delete.\n   * @return {array<string>} Array of application data paths.\n   */\n  async getAppDirs (appFile, appBundleId, scrub = false) {\n    let dirs = [];\n    if (await this.getPlatformVersion() >= 8) {\n      let data = await this.getAppDir(appBundleId);\n      if (!data) return dirs; // eslint-disable-line curly\n\n      let bundle = !scrub ? await this.getAppDir(appBundleId, 'Bundle') : undefined;\n\n      for (let src of [data, bundle]) {\n        if (src) {\n          dirs.push(src);\n        }\n      }\n    } else {\n      let data = await this.getAppDir(appFile);\n      if (data) {\n        dirs.push(data);\n      }\n    }\n    return dirs;\n  }\n\n  /**\n   * Execute the Simulator in order to have the initial file structure created and shutdown it afterwards.\n   *\n   * @param {boolean} safari - Whether to execute mobile Safari after startup.\n   * @param {number} startupTimeout - How long to wait until Simulator booting is completed (in milliseconds).\n   */\n  async launchAndQuit (safari = false, startupTimeout = this.startupTimeout) {\n    log.debug('Attempting to launch and quit the simulator to create the directory structure');\n    await this.run({startupTimeout});\n\n    if (safari) {\n      log.debug('Spawning Safari browser in order to create the necessary file system items');\n      await launchApp(this.simctl, MOBILE_SAFARI_BUNDLE_ID);\n    }\n\n    // wait for the system to create the files we will manipulate\n    // need quite a high retry number, in order to accommodate iOS 7.1\n    // locally, 7.1 averages 8.5 retries (from 6 - 12)\n    //          8 averages 0.6 retries (from 0 - 2)\n    //          9 averages 14 retries\n    try {\n      await retryInterval(60, 250, async () => {\n        if (await this.isFresh()) {\n          throw new Error('Simulator files not fully created. Waiting a bit');\n        }\n      });\n    } catch (err) {\n      log.warn(`Timeout waiting for simulator files to be created. Continuing`);\n    }\n\n    // and quit\n    await this.shutdown();\n  }\n\n  /**\n   * Looks for launchd daemons corresponding to the sim udid and tries to stop them cleanly\n   * This prevents xcrun simctl erase from hanging.\n   */\n  async endSimulatorDaemon () {\n    log.debug(`Killing any simulator daemons for ${this.udid}`);\n\n    let launchctlCmd = `launchctl list | grep ${this.udid} | cut -f 3 | xargs -n 1 launchctl`;\n    try {\n      let stopCmd = `${launchctlCmd} stop`;\n      await exec('bash', ['-c', stopCmd]);\n    } catch (err) {\n      log.warn(`Could not stop simulator daemons: ${err.message}`);\n      log.debug('Carrying on anyway!');\n    }\n    try {\n      let removeCmd = `${launchctlCmd} remove`;\n      await exec('bash', ['-c', removeCmd]);\n    } catch (err) {\n      log.warn(`Could not remove simulator daemons: ${err.message}`);\n      log.debug('Carrying on anyway!');\n    }\n    try {\n      // Waits 10 sec for the simulator launchd services to stop.\n      await waitForCondition(async () => {\n        let {stdout} = await exec('bash', ['-c',\n          `ps -e  | grep ${this.udid} | grep launchd_sim | grep -v bash | grep -v grep | awk {'print$1'}`]);\n        return stdout.trim().length === 0;\n      }, {waitMs: 10000, intervalMs: 500});\n    } catch (err) {\n      log.warn(`Could not end simulator daemon for ${this.udid}: ${err.message}`);\n      log.debug('Carrying on anyway!');\n    }\n  }\n\n  /**\n   * @typedef {Object} ShutdownOptions\n   * @property {?number|string} timeout The number of milliseconds to wait until\n   * Simulator is shut down completely. No wait happens if the timeout value is not set\n   */\n\n  /**\n   * Shut down the current Simulator.\n   *\n   * @param {?ShutdownOptions} opts\n   * @throws {Error} If Simulator fails to transition into Shutdown state after\n   * the given timeout\n   */\n  async shutdown (opts = {}) {\n    if (await this.isShutdown()) {\n      return;\n    }\n\n    await retryInterval(5, 500, this.simctl.shutdownDevice.bind(this.simctl));\n    const waitMs = parseInt(opts.timeout, 10);\n    if (waitMs > 0) {\n      try {\n        await waitForCondition(async () => await this.isShutdown(), {\n          waitMs,\n          intervalMs: 100,\n        });\n      } catch (err) {\n        throw new Error(`Simulator is not in 'Shutdown' state after ${waitMs}ms`);\n      }\n    }\n  }\n\n  /**\n   * Delete the particular Simulator from devices list\n   */\n  async delete () {\n    await this.simctl.deleteDevice();\n  }\n\n  /**\n   * Update the particular preference file with the given key/value pairs.\n   *\n   * @param {string} plist - The preferences file to update.\n   * @param {object} updates - The key/value pairs to update.\n   */\n  async updateSettings (plist, updates) {\n    return await settings.updateSettings(this, plist, updates);\n  }\n\n  /**\n   * Authorize/de-authorize location settings for a particular application.\n   *\n   * @param {string} bundleId - The application ID to update.\n   * @param {boolean} authorized - Whether or not to authorize.\n   */\n  async updateLocationSettings (bundleId, authorized) {\n    return await settings.updateLocationSettings(this, bundleId, authorized);\n  }\n\n  /**\n   * Enable/Disable reduce motion.\n   *\n   * @param {boolean} reduceMotion - Whether or not to enable it.\n   */\n  async setReduceMotion (reduceMotion = true) {\n    if (await this.isFresh()) {\n      await this.launchAndQuit(false, STARTUP_TIMEOUT);\n    }\n\n    await settings.setReduceMotion(this, reduceMotion);\n  }\n\n  /**\n   * Sets UI appearance style.\n   * This function can only be called on a booted simulator.\n   *\n   * @since Xcode SDK 11.4\n   */\n  async setAppearance (/* value */) { // eslint-disable-line require-await\n    throw new Error(`Xcode SDK '${this.xcodeVersion}' is too old to set UI appearance`);\n  }\n\n  /**\n   * Gets the current UI appearance style\n   * This function can only be called on a booted simulator.\n   *\n   * @since Xcode SDK 11.4\n   */\n  async getAppearance () { // eslint-disable-line require-await\n    throw new Error(`Xcode SDK '${this.xcodeVersion}' is too old to get UI appearance`);\n  }\n\n  /**\n   * Update settings for Safari.\n   *\n   * @param {object} updates - The hash of key/value pairs to update for Safari.\n   */\n  async updateSafariSettings (updates) {\n    let updated = await settings.updateSafariUserSettings(this, updates);\n    return await settings.updateSettings(this, 'mobileSafari', updates) || updated;\n  }\n\n  /**\n   * Update global settings for Safari.\n   *\n   * @param {object} updates - The hash of key/value pairs to update for Safari.\n   */\n  async updateSafariGlobalSettings (updates) {\n    return await settings.updateSafariGlobalSettings(this, updates);\n  }\n\n  /**\n   * Update the locale for the Simulator.\n   *\n   * @param {string} language - The language for the simulator. E.g., `\"fr_US\"`.\n   * @param {string} locale - The locale to set for the simulator. E.g., `\"en\"`.\n   * @param {string} calendarFormat - The format of the calendar.\n   */\n  async updateLocale (language, locale, calendarFormat) {\n    return await settings.updateLocale(this, language, locale, calendarFormat);\n  }\n\n  /**\n   * Completely delete mobile Safari application from the current Simulator.\n   */\n  async deleteSafari () {\n    log.debug('Deleting Safari apps from simulator');\n\n    let dirs = [];\n\n    // get the data directory\n    dirs.push(await this.getAppDir(MOBILE_SAFARI_BUNDLE_ID));\n\n    let pv = await this.getPlatformVersion();\n    if (pv >= 8) {\n      // get the bundle directory\n      dirs.push(await this.getAppDir(MOBILE_SAFARI_BUNDLE_ID, 'Bundle'));\n    }\n\n    let deletePromises = [];\n    for (let dir of _.compact(dirs)) {\n      log.debug(`Deleting directory: '${dir}'`);\n      deletePromises.push(fs.rimraf(dir));\n    }\n    await B.all(deletePromises);\n  }\n\n  /**\n   * Clean up the directories for mobile Safari.\n   *\n   * @param {boolean} keepPrefs - Whether to keep Safari preferences from being deleted.\n   */\n  async cleanSafari (keepPrefs = true) {\n    log.debug('Cleaning mobile safari data files');\n    if (await this.isFresh()) {\n      log.info('Could not find Safari support directories to clean out old ' +\n               'data. Probably there is nothing to clean out');\n      return;\n    }\n\n    let libraryDir = path.resolve(this.getDir(), 'Library');\n    let safariRoot = await this.getAppDir(MOBILE_SAFARI_BUNDLE_ID);\n    if (!safariRoot) {\n      log.info('Could not find Safari support directories to clean out old ' +\n               'data. Probably there is nothing to clean out');\n      return;\n    }\n    let safariLibraryDir = path.resolve(safariRoot, 'Library');\n    let filesToDelete = [\n      `Caches/Snapshots/${MOBILE_SAFARI_BUNDLE_ID}`,\n      `Caches/${MOBILE_SAFARI_BUNDLE_ID}/*`,\n      'Caches/com.apple.WebAppCache/*',\n      'Caches/com.apple.WebKit.Networking/*',\n      'Caches/com.apple.WebKit.WebContent/*',\n      'Image Cache/*',\n      `WebKit/${MOBILE_SAFARI_BUNDLE_ID}/*`,\n      'WebKit/GeolocationSites.plist',\n      'WebKit/LocalStorage/*.*',\n      'Safari/*',\n      'Cookies/*.binarycookies',\n      'Caches/com.apple.UIStatusBar/*',\n      'Caches/com.apple.keyboards/images/*',\n      'Caches/com.apple.Safari.SafeBrowsing/*',\n      `../tmp/${MOBILE_SAFARI_BUNDLE_ID}/*`\n    ];\n    let deletePromises = [];\n\n    for (let file of filesToDelete) {\n      deletePromises.push(fs.rimraf(path.resolve(libraryDir, file)));\n      deletePromises.push(fs.rimraf(path.resolve(safariLibraryDir, file)));\n    }\n\n    if (!keepPrefs) {\n      deletePromises.push(fs.rimraf(path.resolve(safariLibraryDir, 'Preferences/*.plist')));\n    }\n\n    await B.all(deletePromises);\n  }\n\n  /**\n   * Uninstall the given application from the current Simulator.\n   *\n   * @param {string} bundleId - The buindle ID of the application to be removed.\n   */\n  async removeApp (bundleId) {\n    await this.simctl.removeApp(bundleId);\n  }\n\n  /**\n   * Move a built-in application to a new place (actually, rename it).\n   *\n   * @param {string} appName - The name of the app to be moved.\n   * @param {string} appPath - The current path to the application.\n   * @param {string} newAppPath - The new path to the application.\n   *   If some application already exists by this path then it's going to be removed.\n   */\n  async moveBuiltInApp (appName, appPath, newAppPath) {\n    await safeRimRaf(newAppPath);\n    await fs.copyFile(appPath, newAppPath);\n    log.debug(`Copied '${appName}' to '${newAppPath}'`);\n\n    await fs.rimraf(appPath);\n    log.debug(`Temporarily deleted original app at '${appPath}'`);\n\n    return [newAppPath, appPath];\n  }\n\n  /**\n   * Open the given URL in mobile Safari browser.\n   * The browser will be started automatically if it is not running.\n   *\n   * @param {string} url - The URL to be opened.\n   */\n  async openUrl (url) {\n    const SAFARI_BOOTED_INDICATOR = 'MobileSafari[';\n    const SAFARI_STARTUP_TIMEOUT = 15 * 1000;\n    const EXTRA_STARTUP_TIME = 3 * 1000;\n\n    if (await this.isRunning()) {\n      await retry(5000, this.simctl.openUrl.bind(this.simctl), url);\n      await this.tailLogsUntil(SAFARI_BOOTED_INDICATOR, SAFARI_STARTUP_TIMEOUT);\n      // So sorry, but the logs have nothing else for Safari starting.. just delay a little bit\n      log.debug(`Safari started, waiting ${EXTRA_STARTUP_TIME}ms for it to fully start`);\n      await B.delay(EXTRA_STARTUP_TIME);\n      log.debug('Done waiting for Safari');\n      return;\n    } else {\n      throw new Error('Tried to open a url, but the Simulator is not Booted');\n    }\n  }\n\n  /**\n   * Perform Simulator caches cleanup.\n   *\n   * @param {...string} folderNames - The names of Caches subfolders to be cleaned.\n   *   Non-accessible/non-existing subfolders will be skipped.\n   *   All existing subfolders under Caches will be deleted if this parameter is omitted.\n   * @returns {number} The count of cleaned cache items.\n   *   Zero is returned if no items were matched for cleanup (either not accessible or not directories).\n   */\n  async clearCaches (...folderNames) {\n    const cachesRoot = path.resolve(this.getDir(), 'Library', 'Caches');\n    if (!(await fs.hasAccess(cachesRoot))) {\n      log.debug(`Caches root at '${cachesRoot}' does not exist or is not accessible. Nothing to do there`);\n      return 0;\n    }\n\n    let itemsToRemove = folderNames.length ? folderNames : (await fs.readdir(cachesRoot));\n    itemsToRemove = itemsToRemove.map((x) => path.resolve(cachesRoot, x));\n    if (folderNames.length) {\n      itemsToRemove = await B.filter(itemsToRemove, (x) => fs.hasAccess(x));\n    }\n    itemsToRemove = await B.filter(itemsToRemove, async (x) => (await fs.stat(x)).isDirectory());\n    if (!itemsToRemove.length) {\n      log.debug(`No Simulator cache items for cleanup were matched in '${cachesRoot}'`);\n      return 0;\n    }\n\n    log.debug(`Matched ${util.pluralize('simulator cache item', itemsToRemove.length, true)} ` +\n      `for cleanup: ${itemsToRemove}`);\n    try {\n      await B.all(itemsToRemove, (x) => fs.rimraf(x));\n    } catch (e) {\n      log.warn(`Got an exception while cleaning Simulator caches: ${e.message}`);\n    }\n    return itemsToRemove.length;\n  }\n\n  /**\n   * Blocks until the given indicater string appears in Simulator logs.\n   *\n   * @param {string} bootedIndicator - The magic string, which appears in logs after Simulator booting is completed.\n   * @param {number} timeoutMs - The maximumm number of milliseconds to wait for the string indicator presence.\n   * @returns {Promise} A promise that resolves when the ios simulator logs output a line matching `bootedIndicator`\n   * times out after timeoutMs\n   */\n  async tailLogsUntil (bootedIndicator, timeoutMs) {\n    let simLog = path.resolve(this.getLogDir(), 'system.log');\n\n    // we need to make sure log file exists before we can tail it\n    await retryInterval(200, 200, async () => {\n      let exists = await fs.exists(simLog);\n      if (!exists) {\n        throw new Error(`Could not find Simulator log: '${simLog}'`);\n      }\n    });\n\n    log.info(`Simulator log at '${simLog}'`);\n    log.info(`Tailing simulator logs until we encounter the string \"${bootedIndicator}\"`);\n    log.info(`We will time out after ${timeoutMs}ms`);\n    try {\n      await tailUntil(simLog, bootedIndicator, timeoutMs);\n    } catch (err) {\n      log.debug('Simulator startup timed out. Continuing anyway.');\n    }\n  }\n\n  /**\n   * Enable Calendar access for the given application.\n   *\n   * @param {string} bundleID - Bundle ID of the application, for which the access should be granted.\n   */\n  async enableCalendarAccess (bundleID) {\n    await this.calendar.enableCalendarAccess(bundleID);\n  }\n\n  /**\n   * Disable Calendar access for the given application.\n   *\n   * @param {string} bundleID - Bundle ID of the application, for which the access should be denied.\n   */\n  async disableCalendarAccess (bundleID) {\n    await this.calendar.disableCalendarAccess(bundleID);\n  }\n\n  /**\n   * Check whether the given application has access to Calendar.\n   *\n   * @return {boolean} True if the given application has the access.\n   */\n  async hasCalendarAccess (bundleID) {\n    return await this.calendar.hasCalendarAccess(bundleID);\n  }\n\n  /**\n   * Activates Simulator window.\n   *\n   * @private\n   * @returns {?string} If the method returns a string then it should be a valid Apple Script which\n   * is appended before each UI client command is executed. Otherwise the method should activate the window\n   * itself and return nothing.\n   */\n  async _activateWindow () { // eslint-disable-line require-await\n    const pid = await this.getUIClientPid();\n    if (pid) {\n      try {\n        return await activateApp(pid);\n      } catch (e) {\n        log.debug(e.stderr || e.message);\n      }\n    }\n    return `\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set frontmost to false\n          set frontmost to true\n        end tell\n      end tell\n    `;\n  }\n\n  /**\n   * Execute given Apple Script inside a critical section, so other\n   * sessions cannot influence the UI client at the same time.\n   *\n   * @param {string} appleScript - The valid Apple Script snippet to be executed.\n   * @return {string} The stdout output produced by the script.\n   * @throws {Error} If osascript tool returns non-zero exit code.\n   */\n  async executeUIClientScript (appleScript) {\n    const windowActivationScript = await this._activateWindow();\n    const resultScript = `${windowActivationScript ? windowActivationScript + '\\n' : ''}${appleScript}`;\n    log.debug(`Executing UI Apple Script on Simulator with UDID ${this.udid}: ${resultScript}`);\n    return await UI_CLIENT_ACCESS_GUARD.acquire(this.simulatorApp, async () => {\n      try {\n        const {stdout} = await exec('osascript', ['-e', resultScript]);\n        return stdout;\n      } catch (err) {\n        log.errorAndThrow(`Could not complete operation. Make sure Simulator UI is running and the parent Appium application (e. g. Appium.app or Terminal.app) ` +\n                          `is present in System Preferences > Security & Privacy > Privacy > Accessibility list. If the operation is still unsuccessful then ` +\n                          `it is not supported by this Simulator. ` +\n                          `Original error: ${err.message}`);\n      }\n    });\n  }\n\n  /**\n   * Get the current state of Biometric Enrollment feature.\n   *\n   * @returns {boolean} Either true or false\n   * @throws {Error} If Enrollment state cannot be determined\n   */\n  async isBiometricEnrolled () {\n    const output = await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"Touch ID Enrolled\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          set isChecked to (value of attribute \"AXMenuItemMarkChar\" of dstMenuItem) is \"✓\"\n        end tell\n      end tell\n    `);\n    log.debug(`Touch ID enrolled state: ${output}`);\n    return _.isString(output) && output.trim() === 'true';\n  }\n\n  /**\n   * Enrolls biometric (TouchId, FaceId) feature testing in Simulator UI client.\n   *\n   * @param {boolean} isEnabled - Defines whether biometric state is enabled/disabled\n   * @throws {Error} If the enrolled state cannot be changed\n   */\n  async enrollBiometric (isEnabled = true) {\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"Touch ID Enrolled\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          set isChecked to (value of attribute \"AXMenuItemMarkChar\" of dstMenuItem) is \"✓\"\n          if ${isEnabled ? 'not ' : ''}isChecked then\n            click dstMenuItem\n          end if\n        end tell\n      end tell\n    `);\n  }\n\n  /**\n   * Sends a notification to match/not match the touch id.\n   *\n   * @param {?boolean} shouldMatch [true] - Set it to true or false in order to emulate\n   * matching/not matching the corresponding biometric\n   */\n  async sendBiometricMatch (shouldMatch = true) {\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"${shouldMatch ? 'Matching' : 'Non-matching'}\" of menu 1 of menu item \"Simulate Finger Touch\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          click dstMenuItem\n        end tell\n      end tell\n    `);\n  }\n\n  /**\n   * Execute a special Apple script, which clicks the particular button on Database alert.\n   *\n   * @param {boolean} increase - Click the button with 'Increase' title on the alert if this\n   *   parameter is true. The 'Cancel' button will be clicked otherwise.\n   */\n  async dismissDatabaseAlert (increase = true) {\n    let button = increase ? 'Increase' : 'Cancel';\n    log.debug(`Attempting to dismiss database alert with '${button}' button`);\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          click button \"${button}\" of window 1\n        end tell\n      end tell\n    `);\n  }\n\n  //region Keychains Interaction\n  /**\n   * Create the backup of keychains folder.\n   * The previously created backup will be automatically\n   * deleted if this method was called twice in a row without\n   * `restoreKeychains` being invoked.\n   *\n   * @returns {boolean} True if the backup operation was successfull.\n   */\n  async backupKeychains () {\n    if (!await fs.exists(this.keychainPath)) {\n      return false;\n    }\n\n    const backupPath = await tempDir.path({\n      prefix: `keychains_backup_${Math.floor((1 + Math.random()) * 0x10000).toString(16).substring(1)}`,\n      suffix: '.zip',\n    });\n    const zipArgs = [\n      '-r', backupPath,\n      `${this.keychainPath}${path.sep}`\n    ];\n    log.debug(`Creating keychains backup with 'zip ${zipArgs.join(' ')}' command`);\n    await exec('zip', zipArgs);\n    if (_.isString(this._keychainsBackupPath) && await fs.exists(this._keychainsBackupPath)) {\n      await fs.unlink(this._keychainsBackupPath);\n    }\n    this._keychainsBackupPath = backupPath;\n    return true;\n  }\n\n  /**\n   * Restore the previsouly created keychains backup.\n   *\n   * @param {?string|Array<string>} excludePatterns - The list\n   * of file name patterns to be excluded from restore. The format\n   * of each item should be the same as '-x' option format for\n   * 'unzip' utility. This can also be a comma-separated string,\n   * which is going be transformed into a list automatically,\n   * for example: '*.db*,blabla.sqlite'\n   * @returns {boolean} If the restore opration was successful.\n   * @throws {Error} If there is no keychains backup available for restore.\n   */\n  async restoreKeychains (excludePatterns = []) {\n    if (!_.isString(this._keychainsBackupPath) || !await fs.exists(this._keychainsBackupPath)) {\n      throw new Error(`The keychains backup archive does not exist. ` +\n                      `Are you sure it was created before?`);\n    }\n\n    if (_.isString(excludePatterns)) {\n      excludePatterns = excludePatterns.split(',').map((x) => x.trim());\n    }\n    const isServerRunning = await this.isRunning();\n    let plistPath;\n    if (isServerRunning) {\n      plistPath = path.resolve(await this.getLaunchDaemonsRoot(), 'com.apple.securityd.plist');\n      if (!await fs.exists(plistPath)) {\n        throw new Error(`Cannot clear keychains because '${plistPath}' does not exist`);\n      }\n      await this.simctl.spawnProcess(['launchctl', 'unload', plistPath]);\n    }\n    try {\n      await fs.rimraf(this.keychainPath);\n      await mkdirp(this.keychainPath);\n      const unzipArgs = [\n        '-o', this._keychainsBackupPath,\n        ...(_.flatMap(excludePatterns.map((x) => ['-x', x]))),\n        '-d', '/'\n      ];\n      log.debug(`Restoring keychains with 'unzip ${unzipArgs.join(' ')}' command`);\n      await exec('unzip', unzipArgs);\n      await fs.unlink(this._keychainsBackupPath);\n      this._keychainsBackupPath = null;\n    } finally {\n      if (isServerRunning && plistPath) {\n        await this.simctl.spawnProcess(['launchctl', 'load', plistPath]);\n      }\n    }\n    return true;\n  }\n\n  /**\n   * Clears Keychains for the particular simulator in runtime (there is no need to stop it).\n   *\n   * @throws {Error} If keychain cleanup has failed.\n   */\n  async clearKeychains () {\n    const plistPath = path.resolve(await this.getLaunchDaemonsRoot(), 'com.apple.securityd.plist');\n    if (!await fs.exists(plistPath)) {\n      throw new Error(`Cannot clear keychains because '${plistPath}' does not exist`);\n    }\n    await this.simctl.spawnProcess(['launchctl', 'unload', plistPath]);\n    try {\n      if (await fs.exists(this.keychainPath)) {\n        await fs.rimraf(this.keychainPath);\n        await mkdirp(this.keychainPath);\n      }\n    } finally {\n      await this.simctl.spawnProcess(['launchctl', 'load', plistPath]);\n    }\n  }\n\n  //endregion\n\n  /**\n   * @typedef {Object} ProcessInfo\n   * @property {number} pid The actual process identifier.\n   * Could be zero if the process is the system one.\n   * @property {?string} group The process group identifier.\n   * This could be `null` if the process is not a part of the\n   * particular group. For `normal` application processes the group\n   * name usually equals to `UIKitApplication`.\n   * @property {string} name The process name, for example\n   * `com.apple.Preferences`\n   */\n\n  /**\n   * Lists processes that are currently running on the given Simulator.\n   * The simulator must be in running state in order for this\n   * method to work properly.\n   *\n   * @return {Array<ProcessInfo>} The list of retrieved process\n   * information\n   * @throws {Error} if no process information could be retrieved.\n   */\n  async ps () {\n    const {stdout} = await this.simctl.spawnProcess([\n      'launchctl',\n      'print',\n      'system',\n    ]);\n\n    const servicesMatch = /^\\s*services\\s*=\\s*{([^}]+)/m.exec(stdout);\n    if (!servicesMatch) {\n      log.debug(stdout);\n      throw new Error(`The list of active processes cannot be retrieved`);\n    }\n    /*\n    Example match:\n        0     78 \tcom.apple.resourcegrabberd\n    82158      - \tcom.apple.assistant_service\n    82120      - \tcom.apple.nanoregistryd\n    82087      - \tcom.apple.notifyd\n    82264      - \tUIKitApplication:com.apple.Preferences[704b][rb-legacy]\n    */\n    const result = [];\n    const pattern = /^\\s*(\\d+)\\s+[\\d-]+\\s+([\\w\\-.]+:)?([\\w\\-.]+)/gm;\n    let match;\n    while ((match = pattern.exec(servicesMatch[1]))) {\n      result.push({\n        pid: parseInt(match[1], 10),\n        group: _.trimEnd(match[2], ':') || null,\n        name: match[3],\n      });\n    }\n    return result;\n  }\n\n  /**\n   * Sets the particular permission to the application bundle. See\n   * https://github.com/wix/AppleSimulatorUtils for more details on\n   * the available service names and statuses.\n   *\n   * @param {string} bundleId - Application bundle identifier.\n   * @param {string} permission - Service name to be set.\n   * @param {string} value - The desired status for the service.\n   * @throws {Error} If there was an error while changing permission.\n   */\n  async setPermission (bundleId, permission, value) {\n    await this.setPermissions(bundleId, {[permission]: value});\n  }\n\n  /**\n   * Sets the permissions for the particular application bundle.\n   *\n   * @param {string} bundleId - Application bundle identifier.\n   * @param {Object} permissionsMapping - A mapping where kays\n   * are service names and values are their corresponding status values.\n   * See https://github.com/wix/AppleSimulatorUtils\n   * for more details on available service names and statuses.\n   * @throws {Error} If there was an error while changing permissions.\n   */\n  async setPermissions (bundleId, permissionsMapping) {\n    log.debug(`Setting access for '${bundleId}': ` +\n      JSON.stringify(permissionsMapping, null, 2));\n    await this.permissions.setAccess(bundleId, permissionsMapping);\n  }\n\n  /**\n   * Retrieves current permission status for the given application bundle.\n   *\n   * @param {string} bundleId - Application bundle identifier.\n   * @param {string} serviceName - One of available service names.\n   * @throws {Error} If there was an error while retrieving permissions.\n   */\n  async getPermission (bundleId, serviceName) {\n    const result = await this.permissions.getAccess(bundleId, serviceName);\n    log.debug(`Got ${serviceName} access status for '${bundleId}': ${result}`);\n    return result;\n  }\n\n  /**\n   * Adds the given certificate into the Trusted Root Store on the simulator.\n   * The simulator must be shut down in order for this method to work properly.\n   *\n   * @param {string} payload the content of the PEM certificate\n   * @returns {boolean} `true` if the certificate has been successfully installed\n   * or `false` if it has already been there\n   */\n  async addCertificate (payload, /* opts = {} */) {\n    if (await hasSSLCert(payload, this.udid)) {\n      log.info(`SSL certificate '${_.truncate(payload, {length: 20})}' already installed`);\n      return false;\n    }\n    log.info(`Installing SSL root certificate '${_.truncate(payload, {length: 20})}'`);\n    await installSSLCert(payload, this.udid);\n    return true;\n  }\n\n  /**\n   * Simulates push notification delivery\n   *\n   * @since Xcode SDK 11.4\n   */\n  async pushNotification (/* payload */) { // eslint-disable-line require-await\n    throw new Error(`Xcode SDK '${this.xcodeVersion}' is too old to push notifications`);\n  }\n\n  async getLaunchDaemonsRoot () {\n    const devRoot = await getDeveloperRoot();\n    return path.resolve(devRoot,\n      'Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator.sdk/System/Library/LaunchDaemons');\n  }\n\n  static async _getDeviceStringPlatformVersion (platformVersion) {\n    let reqVersion = platformVersion;\n    if (!reqVersion) {\n      reqVersion = await xcode.getMaxIOSSDK();\n      log.warn(`No platform version set. Using max SDK version: ${reqVersion}`);\n      // this will be a number, and possibly an integer (e.g., if max iOS SDK is 9)\n      // so turn it into a string and add a .0 if necessary\n      if (!_.isString(reqVersion)) {\n        reqVersion = (reqVersion % 1) ? String(reqVersion) : `${reqVersion}.0`;\n      }\n    }\n    return reqVersion;\n  }\n\n  // change the format in subclasses, as necessary\n  static async _getDeviceStringVersionString (platformVersion) {\n    let reqVersion = await this._getDeviceStringPlatformVersion(platformVersion);\n\n    return `(${reqVersion} Simulator)`;\n  }\n\n  // change the format in subclasses, as necessary\n  static _getDeviceStringConfigFix () {\n    // some devices need to be updated\n    return {\n      'iPad Simulator (7.1 Simulator)': 'iPad 2 (7.1 Simulator)',\n      'iPad Simulator (8.0 Simulator)': 'iPad 2 (8.0 Simulator)',\n      'iPad Simulator (8.1 Simulator)': 'iPad 2 (8.1 Simulator)',\n      'iPad Simulator (8.2 Simulator)': 'iPad 2 (8.2 Simulator)',\n      'iPad Simulator (8.3 Simulator)': 'iPad 2 (8.3 Simulator)',\n      'iPad Simulator (8.4 Simulator)': 'iPad 2 (8.4 Simulator)',\n      'iPhone Simulator (7.1 Simulator)': 'iPhone 5s (7.1 Simulator)',\n      'iPhone Simulator (8.4 Simulator)': 'iPhone 6 (8.4 Simulator)',\n      'iPhone Simulator (8.3 Simulator)': 'iPhone 6 (8.3 Simulator)',\n      'iPhone Simulator (8.2 Simulator)': 'iPhone 6 (8.2 Simulator)',\n      'iPhone Simulator (8.1 Simulator)': 'iPhone 6 (8.1 Simulator)',\n      'iPhone Simulator (8.0 Simulator)': 'iPhone 6 (8.0 Simulator)'\n    };\n  }\n\n  /**\n   * Takes a set of options and finds the correct device string in order for Instruments to\n   * identify the correct simulator.\n   *\n   * @param {object} opts - The options available are:\n   *   - `deviceName` - a name for the device. If the given device name starts with `=`, the name, less the equals sign, is returned.\n   *   - `platformVersion` - the version of iOS to use. Defaults to the current Xcode's maximum SDK version.\n   *   - `forceIphone` - force the configuration of the device string to iPhone. Defaults to `false`.\n   *   - `forceIpad` - force the configuration of the device string to iPad. Defaults to `false`.\n   *   If both `forceIphone` and `forceIpad` are true, the device will be forced to iPhone.\n   *\n   * @return {string} The found device string.\n   */\n  static async getDeviceString (opts) {\n    opts = Object.assign({}, {\n      deviceName: null,\n      platformVersion: null,\n      forceIphone: false,\n      forceIpad: false\n    }, opts);\n    let logOpts = {\n      deviceName: opts.deviceName,\n      platformVersion: opts.platformVersion,\n      forceIphone: opts.forceIphone,\n      forceIpad: opts.forceIpad\n    };\n    log.debug(`Getting device string from options: ${JSON.stringify(logOpts)}`);\n\n    // short circuit if we already have a device name\n    if ((opts.deviceName || '')[0] === '=') {\n      return opts.deviceName.substring(1);\n    }\n\n    let isiPhone = !!opts.forceIphone || !opts.forceIpad;\n\n    if (opts.deviceName) {\n      let device = opts.deviceName.toLowerCase();\n      if (device.indexOf('iphone') !== -1) {\n        isiPhone = true;\n      } else if (device.indexOf('ipad') !== -1) {\n        isiPhone = false;\n      }\n    }\n\n    let iosDeviceString = opts.deviceName || (isiPhone ? 'iPhone Simulator' : 'iPad Simulator');\n\n    // if someone passes in just \"iPhone\", make that \"iPhone Simulator\" to\n    // conform to all the logic below\n    if (/^(iPhone|iPad)$/.test(iosDeviceString)) {\n      iosDeviceString += ' Simulator';\n    }\n\n    // we support deviceName: \"iPhone Simulator\", and also want to support\n    // \"iPhone XYZ Simulator\", but these strings aren't in the device list.\n    // So, if someone sent in \"iPhone XYZ Simulator\", strip off \" Simulator\"\n    // in order to allow the default \"iPhone XYZ\" match\n    if (/[^(iPhone|iPad)] Simulator/.test(iosDeviceString)) {\n      iosDeviceString = iosDeviceString.replace(' Simulator', '');\n    }\n    iosDeviceString += ` ${await this._getDeviceStringVersionString(opts.platformVersion)}`;\n\n    let CONFIG_FIX = this._getDeviceStringConfigFix();\n\n    let configFix = CONFIG_FIX;\n    if (configFix[iosDeviceString]) {\n      iosDeviceString = configFix[iosDeviceString];\n      log.debug(`Fixing device. Changed from '${opts.deviceName}' ` +\n                `to '${iosDeviceString}'`);\n    }\n\n    log.debug(`Final device string is '${iosDeviceString}'`);\n    return iosDeviceString;\n  }\n\n  /**\n   * @return {?string} The full path to the simulator's WebInspector Unix Domain Socket\n   *   or `null` if there is no socket.\n   */\n  async getWebInspectorSocket () { // eslint-disable-line require-await\n    // there is no WebInspector socket for this version of Xcode\n    return null;\n  }\n}\n\nfor (let [cmd, fn] of _.toPairs(extensions)) {\n  SimulatorXcode6.prototype[cmd] = fn;\n}\n\nexport default SimulatorXcode6;\nexport { SimulatorXcode6, BOOT_COMPLETED_EVENT, SPRINGBOARD_BUNDLE_ID };\n"],"file":"lib/simulator-xcode-6.js","sourceRoot":"../.."}
@@ -17,8 +17,6 @@ var _lodash = _interopRequireDefault(require("lodash"));
17
17
 
18
18
  var _logger = _interopRequireDefault(require("./logger"));
19
19
 
20
- var _asyncbox = require("asyncbox");
21
-
22
20
  var _teen_process = require("teen_process");
23
21
 
24
22
  var _geolocation = require("./geolocation");
@@ -29,8 +27,6 @@ var _utils = require("./utils");
29
27
 
30
28
  const STARTUP_TIMEOUT = 120 * 1000;
31
29
 
32
- const PROCESS_LAUNCH_OK_PATTERN = bundleId => new RegExp(`${bundleId.replace('.', '\\.')}:\\s+\\d+`);
33
-
34
30
  class SimulatorXcode8 extends _simulatorXcode2.default {
35
31
  constructor(udid, xcodeVersion) {
36
32
  super(udid, xcodeVersion);
@@ -103,30 +99,12 @@ class SimulatorXcode8 extends _simulatorXcode2.default {
103
99
  }
104
100
 
105
101
  const timer = new _support.timing.Timer().start();
106
- let lastError = null;
107
102
 
108
103
  try {
109
- await (0, _asyncbox.waitForCondition)(async () => {
110
- try {
111
- const stdout = await this.simctl.launchApp(_utils.MOBILE_SAFARI_BUNDLE_ID);
112
-
113
- if (PROCESS_LAUNCH_OK_PATTERN(_utils.MOBILE_SAFARI_BUNDLE_ID).test(stdout)) {
114
- await this.simctl.openUrl(url);
115
- return true;
116
- }
117
- } catch (err) {
118
- _logger.default.warn(`Failed to open '${url}' in Safari. Retrying...`);
119
-
120
- lastError = err.stderr || err.message;
121
- }
122
-
123
- return false;
124
- }, {
125
- waitMs: _utils.SAFARI_STARTUP_TIMEOUT,
126
- intervalMs: 500
127
- });
104
+ await (0, _utils.launchApp)(this.simctl, _utils.MOBILE_SAFARI_BUNDLE_ID, _utils.SAFARI_STARTUP_TIMEOUT);
105
+ await this.simctl.openUrl(url);
128
106
  } catch (err) {
129
- throw new Error(`Safari cannot open '${url}' after ${timer.getDuration().asSeconds.toFixed(3)}s ` + `because of: ${lastError || 'an unknown error'}`);
107
+ throw new Error(`Safari could not open '${url}' after ${timer.getDuration().asSeconds.toFixed(3)}s. ` + `Original error: ${err.stderr || err.message}`);
130
108
  }
131
109
 
132
110
  _logger.default.debug(`Safari successfully opened '${url}' in ${timer.getDuration().asSeconds.toFixed(3)}s`);
@@ -220,4 +198,4 @@ var _default = SimulatorXcode8;
220
198
  exports.default = _default;require('source-map-support').install();
221
199
 
222
200
 
223
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/simulator-xcode-8.js"],"names":["STARTUP_TIMEOUT","PROCESS_LAUNCH_OK_PATTERN","bundleId","RegExp","replace","SimulatorXcode8","SimulatorXcode7","constructor","udid","xcodeVersion","isFreshFiles","_idb","_locationMenu","idb","value","killUIClient","opts","pid","signal","getUIClientPid","log","debug","e","code","Error","message","isAppInstalled","appContainer","simctl","getAppContainer","endsWith","err","info","appInfo","includes","startupTimeout","waitForBoot","startBootMonitor","timeout","emit","BOOT_COMPLETED_EVENT","openUrl","url","isRunning","timer","timing","Timer","start","lastError","stdout","launchApp","MOBILE_SAFARI_BUNDLE_ID","test","warn","stderr","waitMs","SAFARI_STARTUP_TIMEOUT","intervalMs","getDuration","asSeconds","toFixed","cleanSafari","keepPrefs","terminateApp","ign","cleanCustomApp","appFile","appBundleId","scrub","shake","spawnProcess","isBiometricEnrolled","output","executeUIClientScript","_","isString","trim","enrollBiometric","isEnabled","sendBiometricMatch","shouldMatch","setGeolocation","latitude","longitude","locationSetters","setter"],"mappings":"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","sourcesContent":["import { BOOT_COMPLETED_EVENT } from './simulator-xcode-6';\nimport SimulatorXcode7 from './simulator-xcode-7';\nimport _ from 'lodash';\nimport log from './logger';\nimport { waitForCondition } from 'asyncbox';\nimport { exec } from 'teen_process';\nimport {\n  setLocationWithLyft,\n  setLocationWithIdb,\n  setLocationWithAppleScript } from './geolocation';\nimport { timing } from '@appium/support';\nimport { MOBILE_SAFARI_BUNDLE_ID, SAFARI_STARTUP_TIMEOUT } from './utils';\n\n\n// these sims are sloooooooow\nconst STARTUP_TIMEOUT = 120 * 1000;\nconst PROCESS_LAUNCH_OK_PATTERN = (bundleId) => new RegExp(`${bundleId.replace('.', '\\\\.')}:\\\\s+\\\\d+`);\n\nclass SimulatorXcode8 extends SimulatorXcode7 {\n  constructor (udid, xcodeVersion) {\n    super(udid, xcodeVersion);\n\n    // list of files to check for when seeing if a simulator is \"fresh\"\n    // (meaning it has never been booted).\n    // If these files are present, we assume it's been successfully booted\n    this.isFreshFiles = [\n      'Library/Cookies',\n      'Library/Preferences/.GlobalPreferences.plist',\n      'Library/Preferences/com.apple.springboard.plist',\n      'var/run/syslog.pid'\n    ];\n    this._idb = null;\n\n    // for setting the location using AppleScript, the top-level menu through which\n    // the 'Location' option is found\n    this._locationMenu = 'Debug';\n  }\n\n  /**\n   * IDB instance setter\n   *\n   * @param {IDB} value\n   */\n  set idb (value) {\n    this._idb = value;\n  }\n\n  /**\n   * @return {IDB} idb instance\n   */\n  get idb () {\n    return this._idb;\n  }\n\n  /**\n   * @typedef {Object} killOpts\n   * @property {?number|string} pid - Process id of the UI Simulator window\n   * @property {number|string} signal [2] - The signal number to send to the\n   * `kill` command\n   */\n\n  /**\n   * Kill the UI client if it is running.\n   *\n   * @param {?killOpts} opts\n   * @return {boolean} True if the UI client was successfully killed or false\n   *                   if it is not running.\n   * @throws {Error} If sending the signal to the client process fails\n   */\n  async killUIClient (opts = {}) {\n    let {\n      pid,\n      signal = 2,\n    } = opts;\n    pid = pid || await this.getUIClientPid();\n    if (!pid) {\n      return false;\n    }\n\n    log.debug(`Sending ${signal} kill signal to Simulator UI client with PID ${pid}`);\n    try {\n      await exec('kill', [`-${signal}`, pid]);\n      return true;\n    } catch (e) {\n      if (e.code === 1) {\n        return false;\n      }\n      throw new Error(`Cannot kill the Simulator UI client. Original error: ${e.message}`);\n    }\n  }\n\n  /**\n   * Verify whether the particular application is installed on Simulator.\n   * @override\n   *\n   * @param {string} bundleId - The bundle id of the application to be checked.\n   * @return {boolean} True if the given application is installed.\n   */\n  async isAppInstalled (bundleId) {\n    try {\n      const appContainer = await this.simctl.getAppContainer(bundleId);\n      return appContainer.endsWith('.app');\n    } catch (err) {\n      // get_app_container subcommand fails for system applications,\n      // so we try the hidden appinfo subcommand, which prints correct info for\n      // system/hidden apps\n      try {\n        const info = await this.simctl.appInfo(bundleId);\n        return info.includes('ApplicationType');\n      } catch (e) {\n        return false;\n      }\n    }\n  }\n\n  /**\n   * @return {number} The max number of milliseconds to wait until Simulator booting is completed.\n   */\n  get startupTimeout () {\n    return STARTUP_TIMEOUT;\n  }\n\n  /**\n   * Verify whether the Simulator booting is completed and/or wait for it\n   * until the timeout expires.\n   * @override\n   *\n   * @param {number} startupTimeout - the number of milliseconds to wait until booting is completed.\n   * @emits BOOT_COMPLETED_EVENT if the current Simulator is ready to accept simctl commands, like 'install'.\n   */\n  async waitForBoot (startupTimeout) {\n    await this.simctl.startBootMonitor({timeout: startupTimeout});\n    this.emit(BOOT_COMPLETED_EVENT);\n  }\n\n  /**\n   * Open the given URL in mobile Safari browser.\n   * The browser will be started automatically if it is not running.\n   * @override\n   *\n   * @param {string} url - The URL to be opened.\n   */\n  async openUrl (url) {\n    if (!await this.isRunning()) {\n      throw new Error(`Tried to open '${url}', but Simulator is not in Booted state`);\n    }\n    const timer = new timing.Timer().start();\n    let lastError = null;\n    try {\n      await waitForCondition(async () => {\n        try {\n          // This is to make sure Safari is already running\n          const stdout = await this.simctl.launchApp(MOBILE_SAFARI_BUNDLE_ID);\n          if (PROCESS_LAUNCH_OK_PATTERN(MOBILE_SAFARI_BUNDLE_ID).test(stdout)) {\n            await this.simctl.openUrl(url);\n            return true;\n          }\n        } catch (err) {\n          log.warn(`Failed to open '${url}' in Safari. Retrying...`);\n          lastError = err.stderr || err.message;\n        }\n        return false;\n      }, {waitMs: SAFARI_STARTUP_TIMEOUT, intervalMs: 500});\n    } catch (err) {\n      throw new Error(`Safari cannot open '${url}' after ${timer.getDuration().asSeconds.toFixed(3)}s ` +\n        `because of: ${lastError || 'an unknown error'}`);\n    }\n    log.debug(`Safari successfully opened '${url}' in ${timer.getDuration().asSeconds.toFixed(3)}s`);\n  }\n\n  /**\n   * Clean up the directories for mobile Safari.\n   * @override\n   *\n   * @param {boolean} keepPrefs - Whether to keep Safari preferences from being deleted.\n   */\n  async cleanSafari (keepPrefs = true) {\n    try {\n      if (await this.isRunning()) {\n        await this.simctl.terminateApp(MOBILE_SAFARI_BUNDLE_ID);\n      }\n    } catch (ign) {\n      // ignore error\n    }\n    await super.cleanSafari(keepPrefs);\n  }\n\n  /**\n   * Clean/scrub the particular application on Simulator.\n   * @override\n   *\n   * @param {string} appFile - Application name minus \".app\".\n   * @param {string} appBundleId - Bundle identifier of the application.\n   * @param {boolean} scrub - If `scrub` is false, we want to clean by deleting the app and all\n   *   files associated with it. If `scrub` is true, we just want to delete the preferences and\n   *   changed files.\n   */\n  async cleanCustomApp (appFile, appBundleId, scrub = false) {\n    try {\n      await this.simctl.terminateApp(appBundleId);\n    } catch (ign) {\n      // ignore error\n    }\n    await super.cleanCustomApp(appFile, appBundleId, scrub);\n  }\n\n  /**\n   * Perform Shake gesture on Simulator window.\n   */\n  async shake () {\n    log.info(`Performing shake gesture on ${this.udid} Simulator`);\n    await this.simctl.spawnProcess([\n      'notifyutil',\n      '-p', 'com.apple.UIKit.SimulatorShake'\n    ]);\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async isBiometricEnrolled () {\n    const output = await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"Toggle Enrolled State\" of menu 1 of menu item \"Touch ID\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          set isChecked to (value of attribute \"AXMenuItemMarkChar\" of dstMenuItem) is \"✓\"\n        end tell\n      end tell\n    `);\n    log.debug(`Touch ID enrolled state: ${output}`);\n    return _.isString(output) && output.trim() === 'true';\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async enrollBiometric (isEnabled = true) {\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"Toggle Enrolled State\" of menu 1 of menu item \"Touch ID\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          set isChecked to (value of attribute \"AXMenuItemMarkChar\" of dstMenuItem) is \"✓\"\n          if ${isEnabled ? 'not ' : ''}isChecked then\n            click dstMenuItem\n          end if\n        end tell\n      end tell\n    `);\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async sendBiometricMatch (shouldMatch = true) {\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"${shouldMatch ? 'Matching Touch' : 'Non-matching Touch'}\" of menu 1 of menu item \"Touch ID\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          click dstMenuItem\n        end tell\n      end tell\n    `);\n  }\n\n  /**\n   * Set custom geolocation parameters for the given Simulator using AppleScript.\n   *\n   * @param {string|number} latitude - The latitude value, which is going to be entered\n   *   into the corresponding edit field, for example '39,0006'.\n   * @param {string|number} longitude - The longitude value, which is going to be entered\n   *   into the corresponding edit field, for example '19,0068'.\n   * @returns {boolean} True if the given parameters have correct format and were successfully accepted.\n   * @throws {Error} If there was an error while setting the location\n   */\n  async setGeolocation (latitude, longitude) {\n    const locationSetters = [\n      async () => await setLocationWithLyft(this.udid, latitude, longitude),\n      async () => await setLocationWithIdb(this.idb, latitude, longitude),\n      async () => await setLocationWithAppleScript(this, latitude, longitude, this._locationMenu),\n    ];\n\n    let lastError;\n    for (const setter of locationSetters) {\n      try {\n        await setter();\n        return true;\n      } catch (e) {\n        log.info(e.message);\n        lastError = e;\n      }\n    }\n    throw lastError;\n  }\n}\n\nexport default SimulatorXcode8;\n"],"file":"lib/simulator-xcode-8.js","sourceRoot":"../.."}
201
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/simulator-xcode-8.js"],"names":["STARTUP_TIMEOUT","SimulatorXcode8","SimulatorXcode7","constructor","udid","xcodeVersion","isFreshFiles","_idb","_locationMenu","idb","value","killUIClient","opts","pid","signal","getUIClientPid","log","debug","e","code","Error","message","isAppInstalled","bundleId","appContainer","simctl","getAppContainer","endsWith","err","info","appInfo","includes","startupTimeout","waitForBoot","startBootMonitor","timeout","emit","BOOT_COMPLETED_EVENT","openUrl","url","isRunning","timer","timing","Timer","start","MOBILE_SAFARI_BUNDLE_ID","SAFARI_STARTUP_TIMEOUT","getDuration","asSeconds","toFixed","stderr","cleanSafari","keepPrefs","terminateApp","ign","cleanCustomApp","appFile","appBundleId","scrub","shake","spawnProcess","isBiometricEnrolled","output","executeUIClientScript","_","isString","trim","enrollBiometric","isEnabled","sendBiometricMatch","shouldMatch","setGeolocation","latitude","longitude","locationSetters","lastError","setter"],"mappings":"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","sourcesContent":["import { BOOT_COMPLETED_EVENT } from './simulator-xcode-6';\nimport SimulatorXcode7 from './simulator-xcode-7';\nimport _ from 'lodash';\nimport log from './logger';\nimport { exec } from 'teen_process';\nimport {\n  setLocationWithLyft,\n  setLocationWithIdb,\n  setLocationWithAppleScript } from './geolocation';\nimport { timing } from '@appium/support';\nimport { MOBILE_SAFARI_BUNDLE_ID, SAFARI_STARTUP_TIMEOUT, launchApp } from './utils';\n\n\n// these sims are sloooooooow\nconst STARTUP_TIMEOUT = 120 * 1000;\n\nclass SimulatorXcode8 extends SimulatorXcode7 {\n  constructor (udid, xcodeVersion) {\n    super(udid, xcodeVersion);\n\n    // list of files to check for when seeing if a simulator is \"fresh\"\n    // (meaning it has never been booted).\n    // If these files are present, we assume it's been successfully booted\n    this.isFreshFiles = [\n      'Library/Cookies',\n      'Library/Preferences/.GlobalPreferences.plist',\n      'Library/Preferences/com.apple.springboard.plist',\n      'var/run/syslog.pid'\n    ];\n    this._idb = null;\n\n    // for setting the location using AppleScript, the top-level menu through which\n    // the 'Location' option is found\n    this._locationMenu = 'Debug';\n  }\n\n  /**\n   * IDB instance setter\n   *\n   * @param {IDB} value\n   */\n  set idb (value) {\n    this._idb = value;\n  }\n\n  /**\n   * @return {IDB} idb instance\n   */\n  get idb () {\n    return this._idb;\n  }\n\n  /**\n   * @typedef {Object} killOpts\n   * @property {?number|string} pid - Process id of the UI Simulator window\n   * @property {number|string} signal [2] - The signal number to send to the\n   * `kill` command\n   */\n\n  /**\n   * Kill the UI client if it is running.\n   *\n   * @param {?killOpts} opts\n   * @return {boolean} True if the UI client was successfully killed or false\n   *                   if it is not running.\n   * @throws {Error} If sending the signal to the client process fails\n   */\n  async killUIClient (opts = {}) {\n    let {\n      pid,\n      signal = 2,\n    } = opts;\n    pid = pid || await this.getUIClientPid();\n    if (!pid) {\n      return false;\n    }\n\n    log.debug(`Sending ${signal} kill signal to Simulator UI client with PID ${pid}`);\n    try {\n      await exec('kill', [`-${signal}`, pid]);\n      return true;\n    } catch (e) {\n      if (e.code === 1) {\n        return false;\n      }\n      throw new Error(`Cannot kill the Simulator UI client. Original error: ${e.message}`);\n    }\n  }\n\n  /**\n   * Verify whether the particular application is installed on Simulator.\n   * @override\n   *\n   * @param {string} bundleId - The bundle id of the application to be checked.\n   * @return {boolean} True if the given application is installed.\n   */\n  async isAppInstalled (bundleId) {\n    try {\n      const appContainer = await this.simctl.getAppContainer(bundleId);\n      return appContainer.endsWith('.app');\n    } catch (err) {\n      // get_app_container subcommand fails for system applications,\n      // so we try the hidden appinfo subcommand, which prints correct info for\n      // system/hidden apps\n      try {\n        const info = await this.simctl.appInfo(bundleId);\n        return info.includes('ApplicationType');\n      } catch (e) {\n        return false;\n      }\n    }\n  }\n\n  /**\n   * @return {number} The max number of milliseconds to wait until Simulator booting is completed.\n   */\n  get startupTimeout () {\n    return STARTUP_TIMEOUT;\n  }\n\n  /**\n   * Verify whether the Simulator booting is completed and/or wait for it\n   * until the timeout expires.\n   * @override\n   *\n   * @param {number} startupTimeout - the number of milliseconds to wait until booting is completed.\n   * @emits BOOT_COMPLETED_EVENT if the current Simulator is ready to accept simctl commands, like 'install'.\n   */\n  async waitForBoot (startupTimeout) {\n    await this.simctl.startBootMonitor({timeout: startupTimeout});\n    this.emit(BOOT_COMPLETED_EVENT);\n  }\n\n  /**\n   * Open the given URL in mobile Safari browser.\n   * The browser will be started automatically if it is not running.\n   * @override\n   *\n   * @param {string} url - The URL to be opened.\n   */\n  async openUrl (url) {\n    if (!await this.isRunning()) {\n      throw new Error(`Tried to open '${url}', but Simulator is not in Booted state`);\n    }\n    const timer = new timing.Timer().start();\n    try {\n      await launchApp(this.simctl, MOBILE_SAFARI_BUNDLE_ID, SAFARI_STARTUP_TIMEOUT);\n      await this.simctl.openUrl(url);\n    } catch (err) {\n      throw new Error(`Safari could not open '${url}' after ${timer.getDuration().asSeconds.toFixed(3)}s. ` +\n        `Original error: ${err.stderr || err.message}`);\n    }\n    log.debug(`Safari successfully opened '${url}' in ${timer.getDuration().asSeconds.toFixed(3)}s`);\n  }\n\n  /**\n   * Clean up the directories for mobile Safari.\n   * @override\n   *\n   * @param {boolean} keepPrefs - Whether to keep Safari preferences from being deleted.\n   */\n  async cleanSafari (keepPrefs = true) {\n    try {\n      if (await this.isRunning()) {\n        await this.simctl.terminateApp(MOBILE_SAFARI_BUNDLE_ID);\n      }\n    } catch (ign) {\n      // ignore error\n    }\n    await super.cleanSafari(keepPrefs);\n  }\n\n  /**\n   * Clean/scrub the particular application on Simulator.\n   * @override\n   *\n   * @param {string} appFile - Application name minus \".app\".\n   * @param {string} appBundleId - Bundle identifier of the application.\n   * @param {boolean} scrub - If `scrub` is false, we want to clean by deleting the app and all\n   *   files associated with it. If `scrub` is true, we just want to delete the preferences and\n   *   changed files.\n   */\n  async cleanCustomApp (appFile, appBundleId, scrub = false) {\n    try {\n      await this.simctl.terminateApp(appBundleId);\n    } catch (ign) {\n      // ignore error\n    }\n    await super.cleanCustomApp(appFile, appBundleId, scrub);\n  }\n\n  /**\n   * Perform Shake gesture on Simulator window.\n   */\n  async shake () {\n    log.info(`Performing shake gesture on ${this.udid} Simulator`);\n    await this.simctl.spawnProcess([\n      'notifyutil',\n      '-p', 'com.apple.UIKit.SimulatorShake'\n    ]);\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async isBiometricEnrolled () {\n    const output = await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"Toggle Enrolled State\" of menu 1 of menu item \"Touch ID\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          set isChecked to (value of attribute \"AXMenuItemMarkChar\" of dstMenuItem) is \"✓\"\n        end tell\n      end tell\n    `);\n    log.debug(`Touch ID enrolled state: ${output}`);\n    return _.isString(output) && output.trim() === 'true';\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async enrollBiometric (isEnabled = true) {\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"Toggle Enrolled State\" of menu 1 of menu item \"Touch ID\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          set isChecked to (value of attribute \"AXMenuItemMarkChar\" of dstMenuItem) is \"✓\"\n          if ${isEnabled ? 'not ' : ''}isChecked then\n            click dstMenuItem\n          end if\n        end tell\n      end tell\n    `);\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async sendBiometricMatch (shouldMatch = true) {\n    await this.executeUIClientScript(`\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set dstMenuItem to menu item \"${shouldMatch ? 'Matching Touch' : 'Non-matching Touch'}\" of menu 1 of menu item \"Touch ID\" of menu 1 of menu bar item \"Hardware\" of menu bar 1\n          click dstMenuItem\n        end tell\n      end tell\n    `);\n  }\n\n  /**\n   * Set custom geolocation parameters for the given Simulator using AppleScript.\n   *\n   * @param {string|number} latitude - The latitude value, which is going to be entered\n   *   into the corresponding edit field, for example '39,0006'.\n   * @param {string|number} longitude - The longitude value, which is going to be entered\n   *   into the corresponding edit field, for example '19,0068'.\n   * @returns {boolean} True if the given parameters have correct format and were successfully accepted.\n   * @throws {Error} If there was an error while setting the location\n   */\n  async setGeolocation (latitude, longitude) {\n    const locationSetters = [\n      async () => await setLocationWithLyft(this.udid, latitude, longitude),\n      async () => await setLocationWithIdb(this.idb, latitude, longitude),\n      async () => await setLocationWithAppleScript(this, latitude, longitude, this._locationMenu),\n    ];\n\n    let lastError;\n    for (const setter of locationSetters) {\n      try {\n        await setter();\n        return true;\n      } catch (e) {\n        log.info(e.message);\n        lastError = e;\n      }\n    }\n    throw lastError;\n  }\n}\n\nexport default SimulatorXcode8;\n"],"file":"lib/simulator-xcode-8.js","sourceRoot":"../.."}
@@ -14,6 +14,7 @@ exports.getSimulatorInfo = getSimulatorInfo;
14
14
  exports.hasSSLCert = hasSSLCert;
15
15
  exports.installSSLCert = installSSLCert;
16
16
  exports.killAllSimulators = killAllSimulators;
17
+ exports.launchApp = launchApp;
17
18
  exports.safeRimRaf = safeRimRaf;
18
19
  exports.simExists = simExists;
19
20
  exports.toBiometricDomainComponent = toBiometricDomainComponent;
@@ -47,6 +48,8 @@ exports.SAFARI_STARTUP_TIMEOUT = SAFARI_STARTUP_TIMEOUT;
47
48
  const MOBILE_SAFARI_BUNDLE_ID = 'com.apple.mobilesafari';
48
49
  exports.MOBILE_SAFARI_BUNDLE_ID = MOBILE_SAFARI_BUNDLE_ID;
49
50
 
51
+ const PROCESS_LAUNCH_OK_PATTERN = bundleId => new RegExp(`${bundleId.replace('.', '\\.')}:\\s+\\d+`);
52
+
50
53
  const APP_ACTIVATION_SCRIPT = pid => `#!/usr/bin/python
51
54
 
52
55
  from AppKit import NSApplicationActivateIgnoringOtherApps, NSApplicationActivateAllWindows
@@ -352,7 +355,35 @@ async function activateApp(pid) {
352
355
  } finally {
353
356
  await _support.fs.rimraf(tmpScript);
354
357
  }
358
+ }
359
+
360
+ async function launchApp(simctl, bundleId, timeoutMs = 25000) {
361
+ let lastError;
362
+
363
+ try {
364
+ await (0, _asyncbox.waitForCondition)(async () => {
365
+ try {
366
+ const stdout = await simctl.launchApp(MOBILE_SAFARI_BUNDLE_ID, 1);
367
+ return PROCESS_LAUNCH_OK_PATTERN(MOBILE_SAFARI_BUNDLE_ID).test(stdout);
368
+ } catch (err) {
369
+ lastError = err.stderr || err.message;
370
+ }
371
+
372
+ return false;
373
+ }, {
374
+ waitMs: timeoutMs,
375
+ intervalMs: 500
376
+ });
377
+ } catch (e) {
378
+ let msg = `'${bundleId}' did not run after ${timeoutMs}ms timeout.`;
379
+
380
+ if (lastError) {
381
+ msg += ` Original error: ${lastError}`;
382
+ }
383
+
384
+ throw new Error(msg);
385
+ }
355
386
  }require('source-map-support').install();
356
387
 
357
388
 
358
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/utils.js"],"names":["DEFAULT_SIM_SHUTDOWN_TIMEOUT","SAFARI_STARTUP_TIMEOUT","MOBILE_SAFARI_BUNDLE_ID","APP_ACTIVATION_SCRIPT","pid","BIOMETRICS","touchId","faceId","toBiometricDomainComponent","name","Error","JSON","stringify","_","keys","pkill","appName","forceKill","args","push","err","isUndefined","code","log","error","message","killAllSimulators","timeout","debug","xcodeVersion","major","ign","pids","stdout","trim","split","e","isEmpty","warn","join","map","remainingDevices","allSimsAreDown","devices","Simctl","getDevices","flatten","values","every","sim","state","toLowerCase","done","sdk","udid","waitMs","intervalMs","length","device","endAllSimulatorDaemons","servicePattern","launchCtlCommand","stopCmd","removeCmd","getSimulatorInfo","opts","devicesSetPath","toPairs","pair","reduce","a","b","concat","find","simExists","safeRimRaf","delPath","tryNum","fs","rimraf","indexOf","installSSLCert","pemText","which","errorAndThrow","tempFileName","path","resolve","tempDir","openDir","pathToKeychain","Simulator","getDir","writeFile","stat","certificate","Certificate","add","unlink","uninstallSSLCert","__dirname","remove","hasSSLCert","has","execSQLiteQuery","db","query","queryParams","replace","queryTokens","formattedQuery","param","forEach","i","stderr","getDeveloperRoot","activateApp","tmpScript","prefix","util","uuidV4","substring","suffix"],"mappings":"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","sourcesContent":["import log from './logger';\nimport _ from 'lodash';\nimport { exec } from 'teen_process';\nimport { waitForCondition } from 'asyncbox';\nimport { getVersion } from 'appium-xcode';\nimport Simctl from 'node-simctl';\nimport { fs, tempDir, util } from '@appium/support';\nimport { Certificate } from './certificate';\nimport path from 'path';\nimport Simulator from './simulator-xcode-6';\n\n\nconst DEFAULT_SIM_SHUTDOWN_TIMEOUT = 30000;\nconst SAFARI_STARTUP_TIMEOUT = 25 * 1000;\nconst MOBILE_SAFARI_BUNDLE_ID = 'com.apple.mobilesafari';\nconst APP_ACTIVATION_SCRIPT = (pid) => `#!/usr/bin/python\n\nfrom AppKit import NSApplicationActivateIgnoringOtherApps, NSApplicationActivateAllWindows\nfrom Cocoa import NSRunningApplication\n\napp = NSRunningApplication.runningApplicationWithProcessIdentifier_(${pid})\nif not app:\n    raise ValueError('App with PID ${pid} is not running')\nif not app.activateWithOptions_(NSApplicationActivateAllWindows | NSApplicationActivateIgnoringOtherApps):\n    raise ValueError('App with PID ${pid} cannot be activated')\n`;\n\n\nconst BIOMETRICS = {\n  touchId: 'fingerTouch',\n  faceId: 'pearl',\n};\n\nfunction toBiometricDomainComponent (name) {\n  if (!BIOMETRICS[name]) {\n    throw new Error(`'${name}' is not a valid biometric. Use one of: ${JSON.stringify(_.keys(BIOMETRICS))}`);\n  }\n  return BIOMETRICS[name];\n}\n\n// pgrep/pkill exit codes:\n// 0       One or more processes were matched.\n// 1       No processes were matched.\n// 2       Invalid options were specified on the command line.\n// 3       An internal error occurred.\n\nasync function pkill (appName, forceKill = false) {\n  let args = forceKill ? ['-9'] : [];\n  args.push('-x', appName);\n  try {\n    await exec('pkill', args);\n    return 0;\n  } catch (err) {\n    if (!_.isUndefined(err.code)) {\n      throw new Error(`Cannot forcefully terminate ${appName}. pkill error code: ${err.code}`);\n    }\n    log.error(`Received unexpected error while trying to kill ${appName}: ${err.message}`);\n    throw err;\n  }\n}\n\nasync function killAllSimulators (timeout = DEFAULT_SIM_SHUTDOWN_TIMEOUT) {\n  log.debug('Killing all iOS Simulators');\n  const xcodeVersion = await getVersion(true);\n  const appName = xcodeVersion.major >= 7 ? 'Simulator' : 'iOS Simulator';\n\n  // later versions are slower to close\n  timeout = timeout * (xcodeVersion.major >= 8 ? 2 : 1);\n\n  try {\n    await exec('xcrun', ['simctl', 'shutdown', xcodeVersion.major > 8 ? 'all' : 'booted'], {timeout});\n  } catch (ign) {}\n\n  const pids = [];\n  try {\n    const {stdout} = await exec('pgrep', ['-f', `${appName}.app/Contents/MacOS/`]);\n    if (stdout.trim()) {\n      pids.push(...(stdout.trim().split(/\\s+/)));\n    }\n  } catch (e) {\n    if (e.code === 1) {\n      log.debug(`${appName} is not running. Continuing...`);\n      return;\n    }\n    if (_.isEmpty(pids)) {\n      log.warn(`pgrep error ${e.code} while detecting whether ${appName} is running. Trying to kill anyway.`);\n    }\n  }\n  if (!_.isEmpty(pids)) {\n    log.debug(`Killing processes: ${pids.join(', ')}`);\n    try {\n      await exec('kill', ['-9', ...(pids.map((pid) => `${pid}`))]);\n    } catch (ign) {}\n  }\n\n  log.debug(`Using pkill to kill application: ${appName}`);\n  try {\n    await pkill(appName, true);\n  } catch (ign) {}\n\n  // wait for all the devices to be shutdown before Continuing\n  // but only print out the failed ones when they are actually fully failed\n  let remainingDevices = [];\n  async function allSimsAreDown () {\n    remainingDevices = [];\n    let devices = await new Simctl().getDevices();\n    devices = _.flatten(_.values(devices));\n    return _.every(devices, (sim) => {\n      let state = sim.state.toLowerCase();\n      let done = state === 'shutdown' ||\n                 state === 'unavailable' ||\n                 state === 'disconnected';\n      if (!done) {\n        remainingDevices.push(`${sim.name} (${sim.sdk}, udid: ${sim.udid}) is still in state '${state}'`);\n      }\n      return done;\n    });\n  }\n  try {\n    await waitForCondition(allSimsAreDown, {\n      waitMs: timeout,\n      intervalMs: 200\n    });\n  } catch (err) {\n    if (remainingDevices.length > 0) {\n      log.warn(`The following devices are still not in the correct state after ${timeout} ms:`);\n      for (let device of remainingDevices) {\n        log.warn(`    ${device}`);\n      }\n    }\n    throw err;\n  }\n}\n\nasync function endAllSimulatorDaemons () {\n  log.debug('Ending all simulator daemons');\n  for (let servicePattern of ['com.apple.iphonesimulator', 'com.apple.CoreSimulator']) {\n    log.debug(`Killing any other ${servicePattern} daemons`);\n    let launchCtlCommand = `launchctl list | grep ${servicePattern} | cut -f 3 | xargs -n 1 launchctl`;\n    try {\n      let stopCmd = `${launchCtlCommand} stop`;\n      await exec('bash', ['-c', stopCmd]);\n    } catch (err) {\n      log.warn(`Could not stop ${servicePattern} daemons, carrying on anyway!`);\n    }\n    try {\n      let removeCmd = `${launchCtlCommand} remove`;\n      await exec('bash', ['-c', removeCmd]);\n    } catch (err) {\n      log.warn(`Could not remove ${servicePattern} daemons, carrying on anyway!`);\n    }\n  }\n  // waiting until the simulator service has died.\n  try {\n    await waitForCondition(async () => {\n      let {stdout} = await exec('bash', ['-c',\n        `ps -e  | grep launchd_sim | grep -v bash | grep -v grep | awk {'print$1'}`]);\n      return stdout.trim().length === 0;\n    }, {waitMs: 5000, intervalMs: 500});\n  } catch (err) {\n    log.warn(`Could not end all simulator daemons, carrying on!`);\n  }\n  log.debug('Finishing ending all simulator daemons');\n}\n\nasync function getSimulatorInfo (udid, opts = {}) {\n  const {\n    devicesSetPath\n  } = opts;\n  // see the README for github.com/appium/node-simctl for example output of getDevices()\n  let devices = await new Simctl({\n    devicesSetPath\n  }).getDevices();\n\n  devices = _.toPairs(devices)\n    .map((pair) => pair[1])\n    .reduce((a, b) => a.concat(b), []);\n  return _.find(devices, (sim) => sim.udid === udid);\n}\n\nasync function simExists (udid) {\n  return !!(await getSimulatorInfo(udid));\n}\n\nasync function safeRimRaf (delPath, tryNum = 0) {\n  try {\n    await fs.rimraf(delPath);\n  } catch (err) {\n    if (tryNum < 20) {\n      if (err.message.indexOf('ENOTEMPTY') !== -1) {\n        log.debug(`Path '${delPath}' was not empty during delete; retrying`);\n        return await safeRimRaf(delPath, tryNum + 1);\n      } else if (err.message.indexOf('ENOENT') !== -1) {\n        log.debug(`Path '${delPath}' did not exist when we tried to delete, ignoring`);\n        return await safeRimRaf(delPath, tryNum + 1);\n      }\n    }\n  }\n}\n\n/**\n * Install an SSL certificate to a device with given udid\n * @param {string} pemText SSL pem text\n * @param {string} udid Identifier of the Simulator\n */\nasync function installSSLCert (pemText, udid) {\n  // Check that openssl is installed on the path\n  try {\n    await fs.which('openssl');\n  } catch (e) {\n    log.debug(`customSSLCert requires openssl to be available on path`);\n    log.errorAndThrow(`Command 'openssl' not found`);\n  }\n\n  // Check that sqlite3 is installed on the path\n  try {\n    await fs.which('sqlite3');\n  } catch (e) {\n    log.debug(`customSSLCert requires sqlite3 to be available on path`);\n    log.errorAndThrow(`Command 'sqlite3' not found`);\n  }\n\n  // Create a temporary file to store PEM text\n  // (a temp file is necessary to run `openssl` shell commands, can't be done in memory)\n  let tempFileName = path.resolve(await tempDir.openDir(), 'temp-ssl-cert.pem');\n  let pathToKeychain = new Simulator(udid).getDir();\n  await fs.writeFile(tempFileName, pemText);\n  try {\n    await fs.stat(pathToKeychain);\n  } catch (e) {\n    log.debug(`Could not install SSL certificate. No simulator with udid '${udid}'`);\n    log.errorAndThrow(e);\n  }\n\n  // Do the certificate installation\n  let certificate = new Certificate(tempFileName);\n  log.debug(`Installing certificate to ${pathToKeychain}`);\n  await certificate.add(pathToKeychain);\n\n  // Remove the temporary file\n  await fs.unlink(tempFileName);\n\n  return certificate;\n}\n\nasync function uninstallSSLCert (pemText, udid) {\n  try {\n    let tempFileName = path.resolve(__dirname, 'temp-ssl-cert.pem');\n    let pathToKeychain = path.resolve(new Simulator(udid).getDir());\n    await fs.writeFile(tempFileName, pemText);\n    let certificate = new Certificate(tempFileName);\n    await certificate.remove(pathToKeychain);\n    await fs.unlink(tempFileName);\n    return certificate;\n  } catch (e) {\n    log.debug(`Could not uninstall SSL certificate. No simulator with udid '${udid}'`);\n    log.errorAndThrow(e);\n  }\n}\n\n/**\n * Check if the Simulator already has this SSL certificate\n * @param {string} pemText PEM text of SSL cert\n * @param {string} udid Identifier of the Simulator\n */\nasync function hasSSLCert (pemText, udid) {\n  const tempFileName = path.resolve(await tempDir.openDir(), 'temp-ssl-cert.pem');\n  const pathToKeychain = new Simulator(udid).getDir();\n  await fs.writeFile(tempFileName, pemText);\n  const certificate = new Certificate(tempFileName);\n  return certificate.has(pathToKeychain);\n}\n\n/**\n * Runs a command line sqlite3 query\n *\n * @param {string} db - Full path to sqlite database\n * @param {string} query - The actual query string\n * @param {...string} queryParams - The list of query parameters\n * @returns {string} sqlite command stdout\n */\nasync function execSQLiteQuery (db, query, ...queryParams) {\n  query = query.replace(/\\n+/g, ' ');\n  let queryTokens = query.split('?');\n  let formattedQuery = [];\n  queryParams\n    .map((param) => `${param}`)\n    .forEach((param, i) => {\n      formattedQuery.push(queryTokens[i]);\n      formattedQuery.push(param.replace(/'/g, \"''\"));\n    });\n  formattedQuery.push(queryTokens[queryTokens.length - 1]);\n\n  log.debug(`Executing SQL query \"${formattedQuery.join('')}\" on '${db}'`);\n  try {\n    return (await exec('sqlite3', ['-line', db, formattedQuery.join('')])).stdout;\n  } catch (err) {\n    throw new Error(`Cannot execute SQLite query \"${formattedQuery.join('')}\" to '${db}'. ` +\n      `Original error: ${err.stderr}`);\n  }\n}\n\nasync function getDeveloperRoot () {\n  const {stdout} = await exec('xcode-select', ['-p']);\n  return stdout.trim();\n}\n\n/**\n * Activates the app having the given process identifier.\n * See https://developer.apple.com/documentation/appkit/nsrunningapplication/1528725-activatewithoptions?language=objc\n * for more details.\n *\n * @param {number|string} pid App process identifier\n * @throws {Error} If the given PID is not running or there was a failure\n * while activating the app\n */\nasync function activateApp (pid) {\n  const tmpScript = await tempDir.path({\n    prefix: `activate_sim_${util.uuidV4().substring(0, 8)}`,\n    suffix: '.py',\n  });\n  await fs.writeFile(tmpScript, APP_ACTIVATION_SCRIPT(pid), 'utf8');\n  try {\n    await exec('/usr/bin/python', [tmpScript]);\n  } finally {\n    await fs.rimraf(tmpScript);\n  }\n}\n\nexport {\n  killAllSimulators,\n  endAllSimulatorDaemons,\n  safeRimRaf,\n  simExists,\n  getSimulatorInfo,\n  installSSLCert,\n  uninstallSSLCert,\n  hasSSLCert,\n  execSQLiteQuery,\n  toBiometricDomainComponent,\n  getDeveloperRoot,\n  activateApp,\n  SAFARI_STARTUP_TIMEOUT,\n  MOBILE_SAFARI_BUNDLE_ID,\n};\n"],"file":"lib/utils.js","sourceRoot":"../.."}
389
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/utils.js"],"names":["DEFAULT_SIM_SHUTDOWN_TIMEOUT","SAFARI_STARTUP_TIMEOUT","MOBILE_SAFARI_BUNDLE_ID","PROCESS_LAUNCH_OK_PATTERN","bundleId","RegExp","replace","APP_ACTIVATION_SCRIPT","pid","BIOMETRICS","touchId","faceId","toBiometricDomainComponent","name","Error","JSON","stringify","_","keys","pkill","appName","forceKill","args","push","err","isUndefined","code","log","error","message","killAllSimulators","timeout","debug","xcodeVersion","major","ign","pids","stdout","trim","split","e","isEmpty","warn","join","map","remainingDevices","allSimsAreDown","devices","Simctl","getDevices","flatten","values","every","sim","state","toLowerCase","done","sdk","udid","waitMs","intervalMs","length","device","endAllSimulatorDaemons","servicePattern","launchCtlCommand","stopCmd","removeCmd","getSimulatorInfo","opts","devicesSetPath","toPairs","pair","reduce","a","b","concat","find","simExists","safeRimRaf","delPath","tryNum","fs","rimraf","indexOf","installSSLCert","pemText","which","errorAndThrow","tempFileName","path","resolve","tempDir","openDir","pathToKeychain","Simulator","getDir","writeFile","stat","certificate","Certificate","add","unlink","uninstallSSLCert","__dirname","remove","hasSSLCert","has","execSQLiteQuery","db","query","queryParams","queryTokens","formattedQuery","param","forEach","i","stderr","getDeveloperRoot","activateApp","tmpScript","prefix","util","uuidV4","substring","suffix","launchApp","simctl","timeoutMs","lastError","test","msg"],"mappings":"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","sourcesContent":["import log from './logger';\nimport _ from 'lodash';\nimport { exec } from 'teen_process';\nimport { waitForCondition } from 'asyncbox';\nimport { getVersion } from 'appium-xcode';\nimport Simctl from 'node-simctl';\nimport { fs, tempDir, util } from '@appium/support';\nimport { Certificate } from './certificate';\nimport path from 'path';\nimport Simulator from './simulator-xcode-6';\n\n\nconst DEFAULT_SIM_SHUTDOWN_TIMEOUT = 30000;\nconst SAFARI_STARTUP_TIMEOUT = 25 * 1000;\nconst MOBILE_SAFARI_BUNDLE_ID = 'com.apple.mobilesafari';\nconst PROCESS_LAUNCH_OK_PATTERN = (bundleId) => new RegExp(`${bundleId.replace('.', '\\\\.')}:\\\\s+\\\\d+`);\nconst APP_ACTIVATION_SCRIPT = (pid) => `#!/usr/bin/python\n\nfrom AppKit import NSApplicationActivateIgnoringOtherApps, NSApplicationActivateAllWindows\nfrom Cocoa import NSRunningApplication\n\napp = NSRunningApplication.runningApplicationWithProcessIdentifier_(${pid})\nif not app:\n    raise ValueError('App with PID ${pid} is not running')\nif not app.activateWithOptions_(NSApplicationActivateAllWindows | NSApplicationActivateIgnoringOtherApps):\n    raise ValueError('App with PID ${pid} cannot be activated')\n`;\n\n\nconst BIOMETRICS = {\n  touchId: 'fingerTouch',\n  faceId: 'pearl',\n};\n\nfunction toBiometricDomainComponent (name) {\n  if (!BIOMETRICS[name]) {\n    throw new Error(`'${name}' is not a valid biometric. Use one of: ${JSON.stringify(_.keys(BIOMETRICS))}`);\n  }\n  return BIOMETRICS[name];\n}\n\n// pgrep/pkill exit codes:\n// 0       One or more processes were matched.\n// 1       No processes were matched.\n// 2       Invalid options were specified on the command line.\n// 3       An internal error occurred.\n\nasync function pkill (appName, forceKill = false) {\n  let args = forceKill ? ['-9'] : [];\n  args.push('-x', appName);\n  try {\n    await exec('pkill', args);\n    return 0;\n  } catch (err) {\n    if (!_.isUndefined(err.code)) {\n      throw new Error(`Cannot forcefully terminate ${appName}. pkill error code: ${err.code}`);\n    }\n    log.error(`Received unexpected error while trying to kill ${appName}: ${err.message}`);\n    throw err;\n  }\n}\n\nasync function killAllSimulators (timeout = DEFAULT_SIM_SHUTDOWN_TIMEOUT) {\n  log.debug('Killing all iOS Simulators');\n  const xcodeVersion = await getVersion(true);\n  const appName = xcodeVersion.major >= 7 ? 'Simulator' : 'iOS Simulator';\n\n  // later versions are slower to close\n  timeout = timeout * (xcodeVersion.major >= 8 ? 2 : 1);\n\n  try {\n    await exec('xcrun', ['simctl', 'shutdown', xcodeVersion.major > 8 ? 'all' : 'booted'], {timeout});\n  } catch (ign) {}\n\n  const pids = [];\n  try {\n    const {stdout} = await exec('pgrep', ['-f', `${appName}.app/Contents/MacOS/`]);\n    if (stdout.trim()) {\n      pids.push(...(stdout.trim().split(/\\s+/)));\n    }\n  } catch (e) {\n    if (e.code === 1) {\n      log.debug(`${appName} is not running. Continuing...`);\n      return;\n    }\n    if (_.isEmpty(pids)) {\n      log.warn(`pgrep error ${e.code} while detecting whether ${appName} is running. Trying to kill anyway.`);\n    }\n  }\n  if (!_.isEmpty(pids)) {\n    log.debug(`Killing processes: ${pids.join(', ')}`);\n    try {\n      await exec('kill', ['-9', ...(pids.map((pid) => `${pid}`))]);\n    } catch (ign) {}\n  }\n\n  log.debug(`Using pkill to kill application: ${appName}`);\n  try {\n    await pkill(appName, true);\n  } catch (ign) {}\n\n  // wait for all the devices to be shutdown before Continuing\n  // but only print out the failed ones when they are actually fully failed\n  let remainingDevices = [];\n  async function allSimsAreDown () {\n    remainingDevices = [];\n    let devices = await new Simctl().getDevices();\n    devices = _.flatten(_.values(devices));\n    return _.every(devices, (sim) => {\n      let state = sim.state.toLowerCase();\n      let done = state === 'shutdown' ||\n                 state === 'unavailable' ||\n                 state === 'disconnected';\n      if (!done) {\n        remainingDevices.push(`${sim.name} (${sim.sdk}, udid: ${sim.udid}) is still in state '${state}'`);\n      }\n      return done;\n    });\n  }\n  try {\n    await waitForCondition(allSimsAreDown, {\n      waitMs: timeout,\n      intervalMs: 200\n    });\n  } catch (err) {\n    if (remainingDevices.length > 0) {\n      log.warn(`The following devices are still not in the correct state after ${timeout} ms:`);\n      for (let device of remainingDevices) {\n        log.warn(`    ${device}`);\n      }\n    }\n    throw err;\n  }\n}\n\nasync function endAllSimulatorDaemons () {\n  log.debug('Ending all simulator daemons');\n  for (let servicePattern of ['com.apple.iphonesimulator', 'com.apple.CoreSimulator']) {\n    log.debug(`Killing any other ${servicePattern} daemons`);\n    let launchCtlCommand = `launchctl list | grep ${servicePattern} | cut -f 3 | xargs -n 1 launchctl`;\n    try {\n      let stopCmd = `${launchCtlCommand} stop`;\n      await exec('bash', ['-c', stopCmd]);\n    } catch (err) {\n      log.warn(`Could not stop ${servicePattern} daemons, carrying on anyway!`);\n    }\n    try {\n      let removeCmd = `${launchCtlCommand} remove`;\n      await exec('bash', ['-c', removeCmd]);\n    } catch (err) {\n      log.warn(`Could not remove ${servicePattern} daemons, carrying on anyway!`);\n    }\n  }\n  // waiting until the simulator service has died.\n  try {\n    await waitForCondition(async () => {\n      let {stdout} = await exec('bash', ['-c',\n        `ps -e  | grep launchd_sim | grep -v bash | grep -v grep | awk {'print$1'}`]);\n      return stdout.trim().length === 0;\n    }, {waitMs: 5000, intervalMs: 500});\n  } catch (err) {\n    log.warn(`Could not end all simulator daemons, carrying on!`);\n  }\n  log.debug('Finishing ending all simulator daemons');\n}\n\nasync function getSimulatorInfo (udid, opts = {}) {\n  const {\n    devicesSetPath\n  } = opts;\n  // see the README for github.com/appium/node-simctl for example output of getDevices()\n  let devices = await new Simctl({\n    devicesSetPath\n  }).getDevices();\n\n  devices = _.toPairs(devices)\n    .map((pair) => pair[1])\n    .reduce((a, b) => a.concat(b), []);\n  return _.find(devices, (sim) => sim.udid === udid);\n}\n\nasync function simExists (udid) {\n  return !!(await getSimulatorInfo(udid));\n}\n\nasync function safeRimRaf (delPath, tryNum = 0) {\n  try {\n    await fs.rimraf(delPath);\n  } catch (err) {\n    if (tryNum < 20) {\n      if (err.message.indexOf('ENOTEMPTY') !== -1) {\n        log.debug(`Path '${delPath}' was not empty during delete; retrying`);\n        return await safeRimRaf(delPath, tryNum + 1);\n      } else if (err.message.indexOf('ENOENT') !== -1) {\n        log.debug(`Path '${delPath}' did not exist when we tried to delete, ignoring`);\n        return await safeRimRaf(delPath, tryNum + 1);\n      }\n    }\n  }\n}\n\n/**\n * Install an SSL certificate to a device with given udid\n * @param {string} pemText SSL pem text\n * @param {string} udid Identifier of the Simulator\n */\nasync function installSSLCert (pemText, udid) {\n  // Check that openssl is installed on the path\n  try {\n    await fs.which('openssl');\n  } catch (e) {\n    log.debug(`customSSLCert requires openssl to be available on path`);\n    log.errorAndThrow(`Command 'openssl' not found`);\n  }\n\n  // Check that sqlite3 is installed on the path\n  try {\n    await fs.which('sqlite3');\n  } catch (e) {\n    log.debug(`customSSLCert requires sqlite3 to be available on path`);\n    log.errorAndThrow(`Command 'sqlite3' not found`);\n  }\n\n  // Create a temporary file to store PEM text\n  // (a temp file is necessary to run `openssl` shell commands, can't be done in memory)\n  let tempFileName = path.resolve(await tempDir.openDir(), 'temp-ssl-cert.pem');\n  let pathToKeychain = new Simulator(udid).getDir();\n  await fs.writeFile(tempFileName, pemText);\n  try {\n    await fs.stat(pathToKeychain);\n  } catch (e) {\n    log.debug(`Could not install SSL certificate. No simulator with udid '${udid}'`);\n    log.errorAndThrow(e);\n  }\n\n  // Do the certificate installation\n  let certificate = new Certificate(tempFileName);\n  log.debug(`Installing certificate to ${pathToKeychain}`);\n  await certificate.add(pathToKeychain);\n\n  // Remove the temporary file\n  await fs.unlink(tempFileName);\n\n  return certificate;\n}\n\nasync function uninstallSSLCert (pemText, udid) {\n  try {\n    let tempFileName = path.resolve(__dirname, 'temp-ssl-cert.pem');\n    let pathToKeychain = path.resolve(new Simulator(udid).getDir());\n    await fs.writeFile(tempFileName, pemText);\n    let certificate = new Certificate(tempFileName);\n    await certificate.remove(pathToKeychain);\n    await fs.unlink(tempFileName);\n    return certificate;\n  } catch (e) {\n    log.debug(`Could not uninstall SSL certificate. No simulator with udid '${udid}'`);\n    log.errorAndThrow(e);\n  }\n}\n\n/**\n * Check if the Simulator already has this SSL certificate\n * @param {string} pemText PEM text of SSL cert\n * @param {string} udid Identifier of the Simulator\n */\nasync function hasSSLCert (pemText, udid) {\n  const tempFileName = path.resolve(await tempDir.openDir(), 'temp-ssl-cert.pem');\n  const pathToKeychain = new Simulator(udid).getDir();\n  await fs.writeFile(tempFileName, pemText);\n  const certificate = new Certificate(tempFileName);\n  return certificate.has(pathToKeychain);\n}\n\n/**\n * Runs a command line sqlite3 query\n *\n * @param {string} db - Full path to sqlite database\n * @param {string} query - The actual query string\n * @param {...string} queryParams - The list of query parameters\n * @returns {string} sqlite command stdout\n */\nasync function execSQLiteQuery (db, query, ...queryParams) {\n  query = query.replace(/\\n+/g, ' ');\n  let queryTokens = query.split('?');\n  let formattedQuery = [];\n  queryParams\n    .map((param) => `${param}`)\n    .forEach((param, i) => {\n      formattedQuery.push(queryTokens[i]);\n      formattedQuery.push(param.replace(/'/g, \"''\"));\n    });\n  formattedQuery.push(queryTokens[queryTokens.length - 1]);\n\n  log.debug(`Executing SQL query \"${formattedQuery.join('')}\" on '${db}'`);\n  try {\n    return (await exec('sqlite3', ['-line', db, formattedQuery.join('')])).stdout;\n  } catch (err) {\n    throw new Error(`Cannot execute SQLite query \"${formattedQuery.join('')}\" to '${db}'. ` +\n      `Original error: ${err.stderr}`);\n  }\n}\n\nasync function getDeveloperRoot () {\n  const {stdout} = await exec('xcode-select', ['-p']);\n  return stdout.trim();\n}\n\n/**\n * Activates the app having the given process identifier.\n * See https://developer.apple.com/documentation/appkit/nsrunningapplication/1528725-activatewithoptions?language=objc\n * for more details.\n *\n * @param {number|string} pid App process identifier\n * @throws {Error} If the given PID is not running or there was a failure\n * while activating the app\n */\nasync function activateApp (pid) {\n  const tmpScript = await tempDir.path({\n    prefix: `activate_sim_${util.uuidV4().substring(0, 8)}`,\n    suffix: '.py',\n  });\n  await fs.writeFile(tmpScript, APP_ACTIVATION_SCRIPT(pid), 'utf8');\n  try {\n    await exec('/usr/bin/python', [tmpScript]);\n  } finally {\n    await fs.rimraf(tmpScript);\n  }\n}\n\n/**\n * Executes an app and verifies if it was launched properly\n *\n * @param {Simctl} simctl Simctl instance\n * @param {string} bundleId bundle identifier of the destination app\n * @param {number} timeoutMs [25000] The maximum time to wait until\n * the app is running\n * @throws {Error} If the app is still not running after the given timeout\n */\nasync function launchApp (simctl, bundleId, timeoutMs = 25000) {\n  let lastError;\n  try {\n    await waitForCondition(async () => {\n      try {\n        const stdout = await simctl.launchApp(MOBILE_SAFARI_BUNDLE_ID, 1);\n        return PROCESS_LAUNCH_OK_PATTERN(MOBILE_SAFARI_BUNDLE_ID).test(stdout);\n      } catch (err) {\n        lastError = err.stderr || err.message;\n      }\n      return false;\n    }, {waitMs: timeoutMs, intervalMs: 500});\n  } catch (e) {\n    let msg = `'${bundleId}' did not run after ${timeoutMs}ms timeout.`;\n    if (lastError) {\n      msg += ` Original error: ${lastError}`;\n    }\n    throw new Error(msg);\n  }\n}\n\nexport {\n  killAllSimulators,\n  endAllSimulatorDaemons,\n  safeRimRaf,\n  simExists,\n  getSimulatorInfo,\n  installSSLCert,\n  uninstallSSLCert,\n  hasSSLCert,\n  execSQLiteQuery,\n  toBiometricDomainComponent,\n  getDeveloperRoot,\n  activateApp,\n  SAFARI_STARTUP_TIMEOUT,\n  MOBILE_SAFARI_BUNDLE_ID,\n  launchApp,\n};\n"],"file":"lib/utils.js","sourceRoot":"../.."}
@@ -7,6 +7,7 @@ import _ from 'lodash';
7
7
  import AsyncLock from 'async-lock';
8
8
  import {
9
9
  safeRimRaf, getDeveloperRoot, installSSLCert, hasSSLCert, activateApp,
10
+ MOBILE_SAFARI_BUNDLE_ID, launchApp
10
11
  } from './utils.js';
11
12
  import { asyncmap, retryInterval, waitForCondition, retry } from 'asyncbox';
12
13
  import * as settings from './settings';
@@ -637,13 +638,12 @@ class SimulatorXcode6 extends EventEmitter {
637
638
  * @param {number} startupTimeout - How long to wait until Simulator booting is completed (in milliseconds).
638
639
  */
639
640
  async launchAndQuit (safari = false, startupTimeout = this.startupTimeout) {
640
- log.debug('Attempting to launch and quit the simulator, to create directory structure');
641
- log.debug(`Will launch with Safari? ${safari}`);
642
-
641
+ log.debug('Attempting to launch and quit the simulator to create the directory structure');
643
642
  await this.run({startupTimeout});
644
643
 
645
644
  if (safari) {
646
- await this.openUrl('http://www.appium.io');
645
+ log.debug('Spawning Safari browser in order to create the necessary file system items');
646
+ await launchApp(this.simctl, MOBILE_SAFARI_BUNDLE_ID);
647
647
  }
648
648
 
649
649
  // wait for the system to create the files we will manipulate
@@ -831,12 +831,12 @@ class SimulatorXcode6 extends EventEmitter {
831
831
  let dirs = [];
832
832
 
833
833
  // get the data directory
834
- dirs.push(await this.getAppDir('com.apple.mobilesafari'));
834
+ dirs.push(await this.getAppDir(MOBILE_SAFARI_BUNDLE_ID));
835
835
 
836
836
  let pv = await this.getPlatformVersion();
837
837
  if (pv >= 8) {
838
838
  // get the bundle directory
839
- dirs.push(await this.getAppDir('com.apple.mobilesafari', 'Bundle'));
839
+ dirs.push(await this.getAppDir(MOBILE_SAFARI_BUNDLE_ID, 'Bundle'));
840
840
  }
841
841
 
842
842
  let deletePromises = [];
@@ -861,7 +861,7 @@ class SimulatorXcode6 extends EventEmitter {
861
861
  }
862
862
 
863
863
  let libraryDir = path.resolve(this.getDir(), 'Library');
864
- let safariRoot = await this.getAppDir('com.apple.mobilesafari');
864
+ let safariRoot = await this.getAppDir(MOBILE_SAFARI_BUNDLE_ID);
865
865
  if (!safariRoot) {
866
866
  log.info('Could not find Safari support directories to clean out old ' +
867
867
  'data. Probably there is nothing to clean out');
@@ -869,13 +869,13 @@ class SimulatorXcode6 extends EventEmitter {
869
869
  }
870
870
  let safariLibraryDir = path.resolve(safariRoot, 'Library');
871
871
  let filesToDelete = [
872
- 'Caches/Snapshots/com.apple.mobilesafari',
873
- 'Caches/com.apple.mobilesafari/*',
872
+ `Caches/Snapshots/${MOBILE_SAFARI_BUNDLE_ID}`,
873
+ `Caches/${MOBILE_SAFARI_BUNDLE_ID}/*`,
874
874
  'Caches/com.apple.WebAppCache/*',
875
875
  'Caches/com.apple.WebKit.Networking/*',
876
876
  'Caches/com.apple.WebKit.WebContent/*',
877
877
  'Image Cache/*',
878
- 'WebKit/com.apple.mobilesafari/*',
878
+ `WebKit/${MOBILE_SAFARI_BUNDLE_ID}/*`,
879
879
  'WebKit/GeolocationSites.plist',
880
880
  'WebKit/LocalStorage/*.*',
881
881
  'Safari/*',
@@ -883,7 +883,7 @@ class SimulatorXcode6 extends EventEmitter {
883
883
  'Caches/com.apple.UIStatusBar/*',
884
884
  'Caches/com.apple.keyboards/images/*',
885
885
  'Caches/com.apple.Safari.SafeBrowsing/*',
886
- '../tmp/com.apple.mobilesafari/*'
886
+ `../tmp/${MOBILE_SAFARI_BUNDLE_ID}/*`
887
887
  ];
888
888
  let deletePromises = [];
889
889
 
@@ -2,19 +2,17 @@ import { BOOT_COMPLETED_EVENT } from './simulator-xcode-6';
2
2
  import SimulatorXcode7 from './simulator-xcode-7';
3
3
  import _ from 'lodash';
4
4
  import log from './logger';
5
- import { waitForCondition } from 'asyncbox';
6
5
  import { exec } from 'teen_process';
7
6
  import {
8
7
  setLocationWithLyft,
9
8
  setLocationWithIdb,
10
9
  setLocationWithAppleScript } from './geolocation';
11
10
  import { timing } from '@appium/support';
12
- import { MOBILE_SAFARI_BUNDLE_ID, SAFARI_STARTUP_TIMEOUT } from './utils';
11
+ import { MOBILE_SAFARI_BUNDLE_ID, SAFARI_STARTUP_TIMEOUT, launchApp } from './utils';
13
12
 
14
13
 
15
14
  // these sims are sloooooooow
16
15
  const STARTUP_TIMEOUT = 120 * 1000;
17
- const PROCESS_LAUNCH_OK_PATTERN = (bundleId) => new RegExp(`${bundleId.replace('.', '\\.')}:\\s+\\d+`);
18
16
 
19
17
  class SimulatorXcode8 extends SimulatorXcode7 {
20
18
  constructor (udid, xcodeVersion) {
@@ -145,25 +143,12 @@ class SimulatorXcode8 extends SimulatorXcode7 {
145
143
  throw new Error(`Tried to open '${url}', but Simulator is not in Booted state`);
146
144
  }
147
145
  const timer = new timing.Timer().start();
148
- let lastError = null;
149
146
  try {
150
- await waitForCondition(async () => {
151
- try {
152
- // This is to make sure Safari is already running
153
- const stdout = await this.simctl.launchApp(MOBILE_SAFARI_BUNDLE_ID);
154
- if (PROCESS_LAUNCH_OK_PATTERN(MOBILE_SAFARI_BUNDLE_ID).test(stdout)) {
155
- await this.simctl.openUrl(url);
156
- return true;
157
- }
158
- } catch (err) {
159
- log.warn(`Failed to open '${url}' in Safari. Retrying...`);
160
- lastError = err.stderr || err.message;
161
- }
162
- return false;
163
- }, {waitMs: SAFARI_STARTUP_TIMEOUT, intervalMs: 500});
147
+ await launchApp(this.simctl, MOBILE_SAFARI_BUNDLE_ID, SAFARI_STARTUP_TIMEOUT);
148
+ await this.simctl.openUrl(url);
164
149
  } catch (err) {
165
- throw new Error(`Safari cannot open '${url}' after ${timer.getDuration().asSeconds.toFixed(3)}s ` +
166
- `because of: ${lastError || 'an unknown error'}`);
150
+ throw new Error(`Safari could not open '${url}' after ${timer.getDuration().asSeconds.toFixed(3)}s. ` +
151
+ `Original error: ${err.stderr || err.message}`);
167
152
  }
168
153
  log.debug(`Safari successfully opened '${url}' in ${timer.getDuration().asSeconds.toFixed(3)}s`);
169
154
  }
package/lib/utils.js CHANGED
@@ -13,6 +13,7 @@ import Simulator from './simulator-xcode-6';
13
13
  const DEFAULT_SIM_SHUTDOWN_TIMEOUT = 30000;
14
14
  const SAFARI_STARTUP_TIMEOUT = 25 * 1000;
15
15
  const MOBILE_SAFARI_BUNDLE_ID = 'com.apple.mobilesafari';
16
+ const PROCESS_LAUNCH_OK_PATTERN = (bundleId) => new RegExp(`${bundleId.replace('.', '\\.')}:\\s+\\d+`);
16
17
  const APP_ACTIVATION_SCRIPT = (pid) => `#!/usr/bin/python
17
18
 
18
19
  from AppKit import NSApplicationActivateIgnoringOtherApps, NSApplicationActivateAllWindows
@@ -327,6 +328,36 @@ async function activateApp (pid) {
327
328
  }
328
329
  }
329
330
 
331
+ /**
332
+ * Executes an app and verifies if it was launched properly
333
+ *
334
+ * @param {Simctl} simctl Simctl instance
335
+ * @param {string} bundleId bundle identifier of the destination app
336
+ * @param {number} timeoutMs [25000] The maximum time to wait until
337
+ * the app is running
338
+ * @throws {Error} If the app is still not running after the given timeout
339
+ */
340
+ async function launchApp (simctl, bundleId, timeoutMs = 25000) {
341
+ let lastError;
342
+ try {
343
+ await waitForCondition(async () => {
344
+ try {
345
+ const stdout = await simctl.launchApp(MOBILE_SAFARI_BUNDLE_ID, 1);
346
+ return PROCESS_LAUNCH_OK_PATTERN(MOBILE_SAFARI_BUNDLE_ID).test(stdout);
347
+ } catch (err) {
348
+ lastError = err.stderr || err.message;
349
+ }
350
+ return false;
351
+ }, {waitMs: timeoutMs, intervalMs: 500});
352
+ } catch (e) {
353
+ let msg = `'${bundleId}' did not run after ${timeoutMs}ms timeout.`;
354
+ if (lastError) {
355
+ msg += ` Original error: ${lastError}`;
356
+ }
357
+ throw new Error(msg);
358
+ }
359
+ }
360
+
330
361
  export {
331
362
  killAllSimulators,
332
363
  endAllSimulatorDaemons,
@@ -342,4 +373,5 @@ export {
342
373
  activateApp,
343
374
  SAFARI_STARTUP_TIMEOUT,
344
375
  MOBILE_SAFARI_BUNDLE_ID,
376
+ launchApp,
345
377
  };
package/package.json CHANGED
@@ -4,7 +4,7 @@
4
4
  "keywords": [
5
5
  "appium"
6
6
  ],
7
- "version": "4.0.1",
7
+ "version": "4.0.2",
8
8
  "author": "appium",
9
9
  "license": "Apache-2.0",
10
10
  "repository": {
@@ -63,9 +63,9 @@
63
63
  "devDependencies": {
64
64
  "@appium/gulp-plugins": "^6.0.0",
65
65
  "@appium/eslint-config-appium": "^5.0.0",
66
+ "@colors/colors": "^1.5.0",
66
67
  "chai": "^4.1.0",
67
68
  "chai-as-promised": "^7.1.1",
68
- "colors": "^1.1.2",
69
69
  "fs-extra": "^10.0.0",
70
70
  "gulp": "^4.0.0",
71
71
  "ios-test-app": "^3.0.0",