appium-ios-simulator 3.28.1 → 3.28.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.
- package/build/lib/simulator-xcode-9.js +22 -17
- package/lib/simulator-xcode-9.js +28 -19
- package/package.json +2 -2
|
@@ -29,6 +29,8 @@ var _defaultsUtils = require("./defaults-utils");
|
|
|
29
29
|
|
|
30
30
|
var _bluebird = _interopRequireDefault(require("bluebird"));
|
|
31
31
|
|
|
32
|
+
var _events = require("events");
|
|
33
|
+
|
|
32
34
|
const SIMULATOR_SHUTDOWN_TIMEOUT = 15 * 1000;
|
|
33
35
|
const startupLock = new _asyncLock.default();
|
|
34
36
|
const preferencesPlistGuard = new _asyncLock.default();
|
|
@@ -160,25 +162,28 @@ class SimulatorXcode9 extends _simulatorXcode.default {
|
|
|
160
162
|
}
|
|
161
163
|
|
|
162
164
|
async boot() {
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
_logger.default.info(`Booting Simulator '${this.udid}'`);
|
|
165
|
+
const bootEventsEmitter = new _events.EventEmitter();
|
|
166
|
+
await this.simctl.startBootMonitor({
|
|
167
|
+
onError: err => bootEventsEmitter.emit('failure', err),
|
|
168
|
+
onFinished: () => bootEventsEmitter.emit('finish'),
|
|
169
|
+
shouldPreboot: true
|
|
170
|
+
});
|
|
170
171
|
|
|
171
172
|
try {
|
|
172
|
-
await
|
|
173
|
-
|
|
174
|
-
|
|
173
|
+
await new _bluebird.default((resolve, reject) => {
|
|
174
|
+
setTimeout(resolve, 3000);
|
|
175
|
+
bootEventsEmitter.once('failure', err => {
|
|
176
|
+
if (_lodash.default.includes(err === null || err === void 0 ? void 0 : err.message, 'state: Booted')) {
|
|
177
|
+
resolve();
|
|
178
|
+
} else {
|
|
179
|
+
reject(err);
|
|
180
|
+
}
|
|
181
|
+
});
|
|
182
|
+
bootEventsEmitter.once('finish', resolve);
|
|
183
|
+
});
|
|
184
|
+
} finally {
|
|
185
|
+
bootEventsEmitter.removeAllListeners();
|
|
175
186
|
}
|
|
176
|
-
|
|
177
|
-
await (0, _asyncbox.retryInterval)(5, 1000, async () => {
|
|
178
|
-
if (!(await this.isRunning())) {
|
|
179
|
-
throw new Error(`Simulator '${this.udid}' is not running after being booted. ` + `Check the Appium log for more details.`);
|
|
180
|
-
}
|
|
181
|
-
});
|
|
182
187
|
}
|
|
183
188
|
|
|
184
189
|
verifyDevicePreferences(prefs = {}) {
|
|
@@ -451,4 +456,4 @@ var _default = SimulatorXcode9;
|
|
|
451
456
|
exports.default = _default;require('source-map-support').install();
|
|
452
457
|
|
|
453
458
|
|
|
454
|
-
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/simulator-xcode-9.js"],"names":["SIMULATOR_SHUTDOWN_TIMEOUT","startupLock","AsyncLock","preferencesPlistGuard","ENROLLMENT_NOTIFICATION_RECEIVER","SimulatorXcode9","SimulatorXcode8","constructor","udid","xcodeVersion","run","opts","_","cloneDeep","defaultsDeep","devicePreferences","isHeadless","startupTimeout","scaleFactor","SimulatorWindowLastScale","parseFloat","commonPreferences","RotateWindowWhenSignaledByGuest","isBoolean","connectHardwareKeyboard","isNil","ConnectHardwareKeyboard","tracePointer","ShowSingleTouches","ShowPinches","ShowPinchPivotPoint","HighlightEdgeGestures","lowerCase","pasteboardAutomaticSync","PasteboardAutomaticSync","log","info","updatePreferences","timer","timing","Timer","start","shouldWaitForBoot","acquire","uiClientBundleId","isServerRunning","isRunning","uiClientPid","getUIClientPid","killUIClient","pid","isShutdown","waitMs","intervalMs","e","Error","boot","shutdown","timeout","launchWindow","waitForBoot","getDuration","asSeconds","toFixed","isUiClientRunning","startUIClient","simctl","bootDevice","warn","stderr","message","verifyDevicePreferences","prefs","isEmpty","isUndefined","isNumber","errorAndThrow","SimulatorWindowCenter","verificationPattern","isString","test","SimulatorWindowOrientation","acceptableValues","indexOf","SimulatorWindowRotationAngle","devicePrefs","commonPrefs","debug","JSON","stringify","homeFolderPath","process","env","HOME","plistPath","path","resolve","name","defaults","NSUserDefaults","prefsToUpdate","clone","existingDevicePrefs","udidKey","toUpperCase","fs","exists","currentPlistContent","asJson","isPlainObject","DevicePreferences","Object","assign","update","clean","eraseDevice","_activateWindow","selfName","selfSdk","bootedDevicesCount","sdk","deviceArr","toPairs","getDevices","state","isBiometricEnrolled","stdout","spawnProcess","match","RegExp","escapeRegExp","exec","enrollBiometric","isEnabled","sendBiometricMatch","shouldMatch","biometricName","domainComponent","domain","getLaunchDaemonsRoot","devRoot","configureLocalization","language","locale","keyboard","globalPrefs","keyboardId","layout","hardware","AppleKeyboards","AppleLanguages","calendar","localeId","AppleLocale","argChunks","B","all","map","args","KeyboardsCurrentAndNext","KeyboardLastUsed","KeyboardLastUsedForLanguage"],"mappings":"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","sourcesContent":["import SimulatorXcode8 from './simulator-xcode-8';\nimport _ from 'lodash';\nimport path from 'path';\nimport { fs, timing } from 'appium-support';\nimport AsyncLock from 'async-lock';\nimport log from './logger';\nimport { waitForCondition, retryInterval } from 'asyncbox';\nimport { toBiometricDomainComponent, getDeveloperRoot } from './utils.js';\nimport { NSUserDefaults, generateDefaultsCommandArgs } from './defaults-utils';\nimport B from 'bluebird';\n\nconst SIMULATOR_SHUTDOWN_TIMEOUT = 15 * 1000;\nconst startupLock = new AsyncLock();\nconst preferencesPlistGuard = new AsyncLock();\nconst ENROLLMENT_NOTIFICATION_RECEIVER = 'com.apple.BiometricKit.enrollmentChanged';\n\nclass SimulatorXcode9 extends SimulatorXcode8 {\n  constructor (udid, xcodeVersion) {\n    super(udid, xcodeVersion);\n  }\n\n  /**\n   * @typedef {Object} DevicePreferences\n   * @property {?number} SimulatorExternalDisplay - TBD. Example value: 2.114\n   * @property {?string} ChromeTint - TBD. Example value: ''\n   * @property {?number} SimulatorWindowLastScale - Scale value for the particular Simulator window.\n   *                                                1.0 means 100% scale.\n   * @property {?string} SimulatorWindowOrientation - Simulator window orientation. Possible values are:\n   *                                                  'Portrait', 'LandscapeLeft', 'PortraitUpsideDown' and 'LandscapeRight'.\n   * @property {?number} SimulatorWindowRotationAngle - Window rotation angle. This value is expected to be in sync\n   *                                                    with _SimulatorWindowOrientation_. The corresponding values are:\n   *                                                    0, 90, 180 and 270.\n   * @property {?string} SimulatorWindowCenter - The coordinates of Simulator's window center in pixels,\n   *                                             for example '{-1294.5, 775.5}'.\n   * @property {?boolean} ConnectHardwareKeyboard - Equals to 1 if hardware keyboard should be connected.\n   *                                                Otherwise 0.\n   */\n\n  /**\n   * @typedef {Object} CommonPreferences\n   * @property {boolean} ConnectHardwareKeyboard - Whether to connect hardware keyboard\n   */\n\n  /**\n   * @typedef {Object} RunOptions\n   * @property {string} scaleFactor: Any positive float value. 1.0 means 1:1 scale.\n   * 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   * @property {boolean} connectHardwareKeyboard: whether to connect the hardware keyboard to the\n   * Simulator UI client. Equals to `false` by default.\n   * @property {number} startupTimeout: number of milliseconds to wait until Simulator booting\n   * process is completed. The default timeout will be used if not set explicitly.\n   * @property {boolean} isHeadless: whether to start the Simulator in headless mode (with UI\n   * client invisible). `false` by default.\n   * @property {?boolean} tracePointer [false] - Whether to highlight touches on Simulator\n   * screen. This is helpful while debugging automated tests or while observing the automation\n   * recordings.\n   * @property {string} pasteboardAutomaticSync ['off'] - Whether to disable pasteboard sync with the\n   * Simulator UI client or respect the system wide preference. 'on', 'off', or 'system' is available.\n   * The sync increases launching simulator process time, but it allows system to sync pasteboard\n   * with simulators. Follows system-wide preference if the value is 'system'.\n   * Defaults to 'off'.\n   * @property {DevicePreferences} devicePreferences: preferences of the newly created Simulator\n   * device\n   */\n\n  /**\n   * Executes given Simulator with options. The Simulator will not be restarted if\n   * it is already running and the current UI state matches to `isHeadless` option.\n   * @override\n   *\n   * @param {RunOptions} opts - One or more of available Simulator options\n   */\n  async run (opts = {}) {\n    opts = _.cloneDeep(opts);\n    _.defaultsDeep(opts, {\n      devicePreferences: {},\n      isHeadless: false,\n      startupTimeout: this.startupTimeout,\n    });\n\n    if (opts.scaleFactor) {\n      opts.devicePreferences.SimulatorWindowLastScale = parseFloat(opts.scaleFactor);\n    }\n    // This option is necessary to make the Simulator window follow\n    // the actual XCUIDevice orientation\n    const commonPreferences = {\n      RotateWindowWhenSignaledByGuest: true\n    };\n    if (_.isBoolean(opts.connectHardwareKeyboard) || _.isNil(opts.connectHardwareKeyboard)) {\n      opts.devicePreferences.ConnectHardwareKeyboard = opts.connectHardwareKeyboard ?? false;\n      commonPreferences.ConnectHardwareKeyboard = opts.connectHardwareKeyboard ?? false;\n    }\n    if (_.isBoolean(opts.tracePointer)) {\n      commonPreferences.ShowSingleTouches = opts.tracePointer;\n      commonPreferences.ShowPinches = opts.tracePointer;\n      commonPreferences.ShowPinchPivotPoint = opts.tracePointer;\n      commonPreferences.HighlightEdgeGestures = opts.tracePointer;\n    }\n    switch (_.lowerCase(opts.pasteboardAutomaticSync)) {\n      case 'on':\n        commonPreferences.PasteboardAutomaticSync = true;\n        break;\n      case 'off':\n        // Improve launching simulator performance\n        // https://github.com/WebKit/webkit/blob/master/Tools/Scripts/webkitpy/xcode/simulated_device.py#L413\n        commonPreferences.PasteboardAutomaticSync = false;\n        break;\n      case 'system':\n        // Do not add -PasteboardAutomaticSync\n        break;\n      default:\n        log.info(`['on', 'off' or 'system'] are available as the pasteboard automatic sync option. Defaulting to 'off'`);\n        commonPreferences.PasteboardAutomaticSync = false;\n    }\n    await this.updatePreferences(opts.devicePreferences, commonPreferences);\n\n    const timer = new timing.Timer().start();\n    const shouldWaitForBoot = await startupLock.acquire(this.uiClientBundleId, async () => {\n      const isServerRunning = await this.isRunning();\n      const uiClientPid = await this.getUIClientPid();\n      if (opts.isHeadless) {\n        if (isServerRunning && !uiClientPid) {\n          log.info(`Simulator with UDID '${this.udid}' is already booted in headless mode.`);\n          return false;\n        }\n        if (await this.killUIClient({pid: uiClientPid})) {\n          log.info(`Detected the Simulator UI client was running and killed it. Verifying the current Simulator state`);\n        }\n        try {\n          // Stopping the UI client kills all running servers for some early XCode versions. This is a known bug\n          await waitForCondition(async () => await this.isShutdown(), {\n            waitMs: 5000,\n            intervalMs: 100,\n          });\n        } catch (e) {\n          if (!await this.isRunning()) {\n            throw new Error(`Simulator with UDID '${this.udid}' cannot be transitioned to headless mode`);\n          }\n          return false;\n        }\n        log.info(`Booting Simulator with UDID '${this.udid}' in headless mode. ` +\n          `All UI-related capabilities are going to be ignored`);\n        await this.boot();\n      } else {\n        if (isServerRunning && uiClientPid) {\n          log.info(`Both Simulator with UDID '${this.udid}' and the UI client are currently running`);\n          return false;\n        }\n        if (isServerRunning) {\n          log.info(`Simulator '${this.udid}' is booted while its UI is not visible. ` +\n            `Trying to restart it with the Simulator window visible`);\n          await this.shutdown({timeout: SIMULATOR_SHUTDOWN_TIMEOUT});\n        }\n        await this.launchWindow(uiClientPid, opts);\n      }\n      return true;\n    });\n\n    if (shouldWaitForBoot) {\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\n  /***\n   * Boots simulator and opens simulators UI Client if not already opened.\n   *\n   * @param {boolean} isUiClientRunning - process id of simulator UI client.\n   * @param {RunOptions} opts - arguments to start simulator UI client with.\n   */\n  async launchWindow (isUiClientRunning, opts = {}) {\n    await this.boot();\n    if (!isUiClientRunning) {\n      await this.startUIClient(opts);\n    }\n  }\n\n  /**\n   * Boots simulator if not already booted.\n   * Does nothing if it is already running.\n   *\n   * @throws {Error} If Simulator is still not running after being booted\n   */\n  async boot () {\n    if (await this.isRunning()) {\n      log.info(`Simulator '${this.udid}' is already running`);\n      return;\n    }\n\n    log.info(`Booting Simulator '${this.udid}'`);\n    try {\n      await this.simctl.bootDevice();\n    } catch (e) {\n      log.warn(e.stderr || e.message);\n    }\n\n    await retryInterval(5, 1000, async () => {\n      if (!await this.isRunning()) {\n        throw new Error(`Simulator '${this.udid}' is not running after being booted. ` +\n          `Check the Appium log for more details.`);\n      }\n    });\n  }\n\n  /**\n   * Perform verification of device preferences correctness.\n   *\n   * @param {DevicePreferences} prefs [{}] - The preferences to be verified\n   * @throws {Error} If any of the given preference values does not match the expected\n   * format.\n   */\n  verifyDevicePreferences (prefs = {}) {\n    if (_.isEmpty(prefs)) {\n      return;\n    }\n\n    if (!_.isUndefined(prefs.SimulatorWindowLastScale)) {\n      if (!_.isNumber(prefs.SimulatorWindowLastScale) || prefs.SimulatorWindowLastScale <= 0) {\n        log.errorAndThrow(`SimulatorWindowLastScale is expected to be a positive float value. ` +\n          `'${prefs.SimulatorWindowLastScale}' is assigned instead.`);\n      }\n    }\n\n    if (!_.isUndefined(prefs.SimulatorWindowCenter)) {\n      // https://regex101.com/r/2ZXOij/2\n      const verificationPattern = /{-?\\d+(\\.\\d+)?,-?\\d+(\\.\\d+)?}/;\n      if (!_.isString(prefs.SimulatorWindowCenter) || !verificationPattern.test(prefs.SimulatorWindowCenter)) {\n        log.errorAndThrow(`SimulatorWindowCenter is expected to match \"{floatXPosition,floatYPosition}\" format (without spaces). ` +\n          `'${prefs.SimulatorWindowCenter}' is assigned instead.`);\n      }\n    }\n\n    if (!_.isUndefined(prefs.SimulatorWindowOrientation)) {\n      const acceptableValues = ['Portrait', 'LandscapeLeft', 'PortraitUpsideDown', 'LandscapeRight'];\n      if (acceptableValues.indexOf(prefs.SimulatorWindowOrientation) === -1) {\n        log.errorAndThrow(`SimulatorWindowOrientation is expected to be one of ${acceptableValues}. ` +\n          `'${prefs.SimulatorWindowOrientation}' is assigned instead.`);\n      }\n    }\n\n    if (!_.isUndefined(prefs.SimulatorWindowRotationAngle)) {\n      if (!_.isNumber(prefs.SimulatorWindowRotationAngle)) {\n        log.errorAndThrow(`SimulatorWindowRotationAngle is expected to be a valid number. ` +\n          `'${prefs.SimulatorWindowRotationAngle}' is assigned instead.`);\n      }\n    }\n  }\n\n  /**\n   * Update the common iOS Simulator preferences file with new values.\n   * It is necessary to restart the corresponding Simulator before\n   * these changes are applied.\n   *\n   * @param {DevicePreferences} devicePrefs [{}] - The mapping, which represents new device preference values\n   *                                               for the given Simulator.\n   * @param {CommonPreferences} commonPrefs [{}] - The mapping, which represents new common preference values\n   *                                               for all Simulators.\n   * @return {boolean} True if the preferences were successfully updated.\n   */\n  async updatePreferences (devicePrefs = {}, commonPrefs = {}) {\n    if (!_.isEmpty(devicePrefs)) {\n      log.debug(`Setting preferences of ${this.udid} Simulator to ${JSON.stringify(devicePrefs)}`);\n    }\n    if (!_.isEmpty(commonPrefs)) {\n      log.debug(`Setting common Simulator preferences to ${JSON.stringify(commonPrefs)}`);\n    }\n    const homeFolderPath = process.env.HOME;\n    if (!homeFolderPath) {\n      log.warn(`Cannot get the path to HOME folder from the process environment. ` +\n        `Ignoring Simulator preferences update.`);\n      return false;\n    }\n    this.verifyDevicePreferences(devicePrefs);\n    const plistPath = path.resolve(homeFolderPath, 'Library', 'Preferences', 'com.apple.iphonesimulator.plist');\n    return await preferencesPlistGuard.acquire(SimulatorXcode9.name, async () => {\n      const defaults = new NSUserDefaults(plistPath);\n      const prefsToUpdate = _.clone(commonPrefs);\n      try {\n        if (!_.isEmpty(devicePrefs)) {\n          let existingDevicePrefs;\n          const udidKey = this.udid.toUpperCase();\n          if (await fs.exists(plistPath)) {\n            const currentPlistContent = await defaults.asJson();\n            if (_.isPlainObject(currentPlistContent.DevicePreferences)\n                && _.isPlainObject(currentPlistContent.DevicePreferences[udidKey])) {\n              existingDevicePrefs = currentPlistContent.DevicePreferences[udidKey];\n            }\n          }\n          Object.assign(prefsToUpdate, {\n            DevicePreferences: {\n              [udidKey]: Object.assign({}, existingDevicePrefs || {}, devicePrefs)\n            }\n          });\n        }\n        await defaults.update(prefsToUpdate);\n        log.debug(`Updated ${this.udid} Simulator preferences at '${plistPath}' with ` +\n          JSON.stringify(prefsToUpdate));\n        return true;\n      } catch (e) {\n        log.warn(`Cannot update ${this.udid} Simulator preferences at '${plistPath}'. ` +\n          `Try to delete the file manually in order to reset it. Original error: ${e.message}`);\n        return false;\n      }\n    });\n  }\n\n  /**\n   * Reset the current Simulator to the clean state.\n   * @override\n   */\n  async clean () {\n    log.info(`Cleaning simulator ${this.udid}`);\n    await this.simctl.eraseDevice(10000);\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   * @private\n   */\n  async _activateWindow () {\n    let selfName;\n    let selfSdk;\n    let bootedDevicesCount = 0;\n    for (const [sdk, deviceArr] of _.toPairs(await this.simctl.getDevices())) {\n      for (const {state, udid, name} of deviceArr) {\n        if (state === 'Booted') {\n          bootedDevicesCount++;\n        }\n        if (!selfName && udid === this.udid) {\n          selfSdk = sdk;\n          selfName = name;\n        }\n      }\n    }\n    if (bootedDevicesCount < 2) {\n      return await super._activateWindow();\n    }\n\n    // There are potentially more that one Simulator window\n    return `\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set frontmost to false\n          set frontmost to true\n          click (menu item 1 where (its name contains \"${selfName} \" and its name contains \"${selfSdk}\")) of menu 1 of menu bar item \"Window\" of menu bar 1\n        end tell\n      end tell\n    `;\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async isBiometricEnrolled () {\n    const {stdout} = await this.simctl.spawnProcess([\n      'notifyutil',\n      '-g', ENROLLMENT_NOTIFICATION_RECEIVER\n    ]);\n    const match = (new RegExp(`${_.escapeRegExp(ENROLLMENT_NOTIFICATION_RECEIVER)}\\\\s+([01])`))\n      .exec(stdout);\n    if (!match) {\n      throw new Error(`Cannot parse biometric enrollment state from '${stdout}'`);\n    }\n    log.info(`Current biometric enrolled state for ${this.udid} Simulator: ${match[1]}`);\n    return match[1] === '1';\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async enrollBiometric (isEnabled = true) {\n    log.debug(`Setting biometric enrolled state for ${this.udid} Simulator to '${isEnabled ? 'enabled' : 'disabled'}'`);\n    await this.simctl.spawnProcess([\n      'notifyutil',\n      '-s', ENROLLMENT_NOTIFICATION_RECEIVER, isEnabled ? '1' : '0'\n    ]);\n    await this.simctl.spawnProcess([\n      'notifyutil',\n      '-p', ENROLLMENT_NOTIFICATION_RECEIVER\n    ]);\n    if (await this.isBiometricEnrolled() !== isEnabled) {\n      throw new Error(`Cannot set biometric enrolled state for ${this.udid} Simulator to '${isEnabled ? 'enabled' : 'disabled'}'`);\n    }\n  }\n\n  /**\n   * Sends a notification to match/not match the particular biometric.\n   * @override\n   *\n   * @param {?boolean} shouldMatch [true] - Set it to true or false in order to emulate\n   * matching/not matching the corresponding biometric\n   * @param {?string} biometricName [touchId] - Either touchId or faceId (faceId is only available since iOS 11)\n   */\n  async sendBiometricMatch (shouldMatch = true, biometricName = 'touchId') {\n    const domainComponent = toBiometricDomainComponent(biometricName);\n    const domain = `com.apple.BiometricKit_Sim.${domainComponent}.${shouldMatch ? '' : 'no'}match`;\n    await this.simctl.spawnProcess([\n      'notifyutil',\n      '-p', domain\n    ]);\n    log.info(`Sent notification ${domain} to ${shouldMatch ? 'match' : 'not match'} ${biometricName} biometric ` +\n      `for ${this.udid} Simulator`);\n  }\n\n  /**\n   * @override\n   */\n  async getLaunchDaemonsRoot () {\n    const devRoot = await getDeveloperRoot();\n    return path.resolve(devRoot,\n      'Platforms/iPhoneOS.platform/Developer/Library/CoreSimulator/Profiles/Runtimes/iOS.simruntime/Contents/Resources/RuntimeRoot/System/Library/LaunchDaemons');\n  }\n\n  /**\n   * @typedef {Object} KeyboardOptions\n   * @property {!string} name The name of the keyboard locale, for example `en_US` or `de_CH`\n   * @property {!string} layout The keyboard layout, for example `QUERTY` or `Ukrainian`\n   * @property {?string} hardware Could either be `Automatic` or `null`\n   */\n\n  /**\n   * @typedef {Object} LanguageOptions\n   * @property {!string} name The name of the language, for example `de` or `zh-Hant-CN`\n   */\n\n  /**\n   * @typedef {Object} LocaleOptions\n   * @property {!string} name The name of the system locale, for example `de_CH` or `zh_CN`\n   * @property {?string} calendar Optional calendar format, for example `gregorian` or `persian`\n   */\n\n  /**\n   * @typedef {Object} LocalizationOptions\n   * @property {?KeyboardOptions} keyboard\n   * @property {?LanguageOptions} language\n   * @property {?LocaleOptions} locale\n   */\n\n  /**\n   * Change localization settings on the currently booted simulator\n   *\n   * @param {?LocalizationOptions} opts\n   * @throws {Error} If there was a failure while setting the preferences\n   * @returns {boolean} `true` if any of settings has been successfully changed\n   */\n  async configureLocalization (opts = {}) {\n    if (_.isEmpty(opts)) {\n      return false;\n    }\n\n    const { language, locale, keyboard } = opts;\n    const globalPrefs = {};\n    let keyboardId = null;\n    if (_.isPlainObject(keyboard)) {\n      const { name, layout, hardware } = keyboard;\n      if (!name) {\n        throw new Error(`The 'keyboard' field must have a valid name set`);\n      }\n      if (!layout) {\n        throw new Error(`The 'keyboard' field must have a valid layout set`);\n      }\n      keyboardId = `${name}@sw=${layout}`;\n      if (hardware) {\n        keyboardId += `;@hw=${hardware}`;\n      }\n      globalPrefs.AppleKeyboards = [keyboardId];\n    }\n    if (_.isPlainObject(language)) {\n      const { name } = language;\n      if (!name) {\n        throw new Error(`The 'language' field must have a valid name set`);\n      }\n      globalPrefs.AppleLanguages = [name];\n    }\n    if (_.isPlainObject(locale)) {\n      const { name, calendar } = locale;\n      if (!name) {\n        throw new Error(`The 'locale' field must have a valid name set`);\n      }\n      let localeId = name;\n      if (calendar) {\n        localeId += `@calendar=${calendar}`;\n      }\n      globalPrefs.AppleLocale = localeId;\n    }\n    if (_.isEmpty(globalPrefs)) {\n      return false;\n    }\n\n    const argChunks = generateDefaultsCommandArgs(globalPrefs, true);\n    await B.all(argChunks.map((args) => this.simctl.spawnProcess([\n      'defaults', 'write', '.GlobalPreferences.plist', ...args\n    ])));\n\n    if (keyboardId) {\n      const argChunks = generateDefaultsCommandArgs({\n        KeyboardsCurrentAndNext: [keyboardId],\n        KeyboardLastUsed: keyboardId,\n        KeyboardLastUsedForLanguage: { [keyboard.name]: keyboardId }\n      }, true);\n      await B.all(argChunks.map((args) => this.simctl.spawnProcess([\n        'defaults', 'write', 'com.apple.Preferences', ...args\n      ])));\n    }\n\n    return true;\n  }\n\n}\n\nexport default SimulatorXcode9;\n"],"file":"lib/simulator-xcode-9.js","sourceRoot":"../.."}
|
|
459
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/simulator-xcode-9.js"],"names":["SIMULATOR_SHUTDOWN_TIMEOUT","startupLock","AsyncLock","preferencesPlistGuard","ENROLLMENT_NOTIFICATION_RECEIVER","SimulatorXcode9","SimulatorXcode8","constructor","udid","xcodeVersion","run","opts","_","cloneDeep","defaultsDeep","devicePreferences","isHeadless","startupTimeout","scaleFactor","SimulatorWindowLastScale","parseFloat","commonPreferences","RotateWindowWhenSignaledByGuest","isBoolean","connectHardwareKeyboard","isNil","ConnectHardwareKeyboard","tracePointer","ShowSingleTouches","ShowPinches","ShowPinchPivotPoint","HighlightEdgeGestures","lowerCase","pasteboardAutomaticSync","PasteboardAutomaticSync","log","info","updatePreferences","timer","timing","Timer","start","shouldWaitForBoot","acquire","uiClientBundleId","isServerRunning","isRunning","uiClientPid","getUIClientPid","killUIClient","pid","isShutdown","waitMs","intervalMs","e","Error","boot","shutdown","timeout","launchWindow","waitForBoot","getDuration","asSeconds","toFixed","isUiClientRunning","startUIClient","bootEventsEmitter","EventEmitter","simctl","startBootMonitor","onError","err","emit","onFinished","shouldPreboot","B","resolve","reject","setTimeout","once","includes","message","removeAllListeners","verifyDevicePreferences","prefs","isEmpty","isUndefined","isNumber","errorAndThrow","SimulatorWindowCenter","verificationPattern","isString","test","SimulatorWindowOrientation","acceptableValues","indexOf","SimulatorWindowRotationAngle","devicePrefs","commonPrefs","debug","JSON","stringify","homeFolderPath","process","env","HOME","warn","plistPath","path","name","defaults","NSUserDefaults","prefsToUpdate","clone","existingDevicePrefs","udidKey","toUpperCase","fs","exists","currentPlistContent","asJson","isPlainObject","DevicePreferences","Object","assign","update","clean","eraseDevice","_activateWindow","selfName","selfSdk","bootedDevicesCount","sdk","deviceArr","toPairs","getDevices","state","isBiometricEnrolled","stdout","spawnProcess","match","RegExp","escapeRegExp","exec","enrollBiometric","isEnabled","sendBiometricMatch","shouldMatch","biometricName","domainComponent","domain","getLaunchDaemonsRoot","devRoot","configureLocalization","language","locale","keyboard","globalPrefs","keyboardId","layout","hardware","AppleKeyboards","AppleLanguages","calendar","localeId","AppleLocale","argChunks","all","map","args","KeyboardsCurrentAndNext","KeyboardLastUsed","KeyboardLastUsedForLanguage"],"mappings":"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","sourcesContent":["import SimulatorXcode8 from './simulator-xcode-8';\nimport _ from 'lodash';\nimport path from 'path';\nimport { fs, timing } from 'appium-support';\nimport AsyncLock from 'async-lock';\nimport log from './logger';\nimport { waitForCondition } from 'asyncbox';\nimport { toBiometricDomainComponent, getDeveloperRoot } from './utils.js';\nimport { NSUserDefaults, generateDefaultsCommandArgs } from './defaults-utils';\nimport B from 'bluebird';\nimport { EventEmitter } from 'events';\n\nconst SIMULATOR_SHUTDOWN_TIMEOUT = 15 * 1000;\nconst startupLock = new AsyncLock();\nconst preferencesPlistGuard = new AsyncLock();\nconst ENROLLMENT_NOTIFICATION_RECEIVER = 'com.apple.BiometricKit.enrollmentChanged';\n\nclass SimulatorXcode9 extends SimulatorXcode8 {\n  constructor (udid, xcodeVersion) {\n    super(udid, xcodeVersion);\n  }\n\n  /**\n   * @typedef {Object} DevicePreferences\n   * @property {?number} SimulatorExternalDisplay - TBD. Example value: 2.114\n   * @property {?string} ChromeTint - TBD. Example value: ''\n   * @property {?number} SimulatorWindowLastScale - Scale value for the particular Simulator window.\n   *                                                1.0 means 100% scale.\n   * @property {?string} SimulatorWindowOrientation - Simulator window orientation. Possible values are:\n   *                                                  'Portrait', 'LandscapeLeft', 'PortraitUpsideDown' and 'LandscapeRight'.\n   * @property {?number} SimulatorWindowRotationAngle - Window rotation angle. This value is expected to be in sync\n   *                                                    with _SimulatorWindowOrientation_. The corresponding values are:\n   *                                                    0, 90, 180 and 270.\n   * @property {?string} SimulatorWindowCenter - The coordinates of Simulator's window center in pixels,\n   *                                             for example '{-1294.5, 775.5}'.\n   * @property {?boolean} ConnectHardwareKeyboard - Equals to 1 if hardware keyboard should be connected.\n   *                                                Otherwise 0.\n   */\n\n  /**\n   * @typedef {Object} CommonPreferences\n   * @property {boolean} ConnectHardwareKeyboard - Whether to connect hardware keyboard\n   */\n\n  /**\n   * @typedef {Object} RunOptions\n   * @property {string} scaleFactor: Any positive float value. 1.0 means 1:1 scale.\n   * 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   * @property {boolean} connectHardwareKeyboard: whether to connect the hardware keyboard to the\n   * Simulator UI client. Equals to `false` by default.\n   * @property {number} startupTimeout: number of milliseconds to wait until Simulator booting\n   * process is completed. The default timeout will be used if not set explicitly.\n   * @property {boolean} isHeadless: whether to start the Simulator in headless mode (with UI\n   * client invisible). `false` by default.\n   * @property {?boolean} tracePointer [false] - Whether to highlight touches on Simulator\n   * screen. This is helpful while debugging automated tests or while observing the automation\n   * recordings.\n   * @property {string} pasteboardAutomaticSync ['off'] - Whether to disable pasteboard sync with the\n   * Simulator UI client or respect the system wide preference. 'on', 'off', or 'system' is available.\n   * The sync increases launching simulator process time, but it allows system to sync pasteboard\n   * with simulators. Follows system-wide preference if the value is 'system'.\n   * Defaults to 'off'.\n   * @property {DevicePreferences} devicePreferences: preferences of the newly created Simulator\n   * device\n   */\n\n  /**\n   * Executes given Simulator with options. The Simulator will not be restarted if\n   * it is already running and the current UI state matches to `isHeadless` option.\n   * @override\n   *\n   * @param {RunOptions} opts - One or more of available Simulator options\n   */\n  async run (opts = {}) {\n    opts = _.cloneDeep(opts);\n    _.defaultsDeep(opts, {\n      devicePreferences: {},\n      isHeadless: false,\n      startupTimeout: this.startupTimeout,\n    });\n\n    if (opts.scaleFactor) {\n      opts.devicePreferences.SimulatorWindowLastScale = parseFloat(opts.scaleFactor);\n    }\n    // This option is necessary to make the Simulator window follow\n    // the actual XCUIDevice orientation\n    const commonPreferences = {\n      RotateWindowWhenSignaledByGuest: true\n    };\n    if (_.isBoolean(opts.connectHardwareKeyboard) || _.isNil(opts.connectHardwareKeyboard)) {\n      opts.devicePreferences.ConnectHardwareKeyboard = opts.connectHardwareKeyboard ?? false;\n      commonPreferences.ConnectHardwareKeyboard = opts.connectHardwareKeyboard ?? false;\n    }\n    if (_.isBoolean(opts.tracePointer)) {\n      commonPreferences.ShowSingleTouches = opts.tracePointer;\n      commonPreferences.ShowPinches = opts.tracePointer;\n      commonPreferences.ShowPinchPivotPoint = opts.tracePointer;\n      commonPreferences.HighlightEdgeGestures = opts.tracePointer;\n    }\n    switch (_.lowerCase(opts.pasteboardAutomaticSync)) {\n      case 'on':\n        commonPreferences.PasteboardAutomaticSync = true;\n        break;\n      case 'off':\n        // Improve launching simulator performance\n        // https://github.com/WebKit/webkit/blob/master/Tools/Scripts/webkitpy/xcode/simulated_device.py#L413\n        commonPreferences.PasteboardAutomaticSync = false;\n        break;\n      case 'system':\n        // Do not add -PasteboardAutomaticSync\n        break;\n      default:\n        log.info(`['on', 'off' or 'system'] are available as the pasteboard automatic sync option. Defaulting to 'off'`);\n        commonPreferences.PasteboardAutomaticSync = false;\n    }\n    await this.updatePreferences(opts.devicePreferences, commonPreferences);\n\n    const timer = new timing.Timer().start();\n    const shouldWaitForBoot = await startupLock.acquire(this.uiClientBundleId, async () => {\n      const isServerRunning = await this.isRunning();\n      const uiClientPid = await this.getUIClientPid();\n      if (opts.isHeadless) {\n        if (isServerRunning && !uiClientPid) {\n          log.info(`Simulator with UDID '${this.udid}' is already booted in headless mode.`);\n          return false;\n        }\n        if (await this.killUIClient({pid: uiClientPid})) {\n          log.info(`Detected the Simulator UI client was running and killed it. Verifying the current Simulator state`);\n        }\n        try {\n          // Stopping the UI client kills all running servers for some early XCode versions. This is a known bug\n          await waitForCondition(async () => await this.isShutdown(), {\n            waitMs: 5000,\n            intervalMs: 100,\n          });\n        } catch (e) {\n          if (!await this.isRunning()) {\n            throw new Error(`Simulator with UDID '${this.udid}' cannot be transitioned to headless mode`);\n          }\n          return false;\n        }\n        log.info(`Booting Simulator with UDID '${this.udid}' in headless mode. ` +\n          `All UI-related capabilities are going to be ignored`);\n        await this.boot();\n      } else {\n        if (isServerRunning && uiClientPid) {\n          log.info(`Both Simulator with UDID '${this.udid}' and the UI client are currently running`);\n          return false;\n        }\n        if (isServerRunning) {\n          log.info(`Simulator '${this.udid}' is booted while its UI is not visible. ` +\n            `Trying to restart it with the Simulator window visible`);\n          await this.shutdown({timeout: SIMULATOR_SHUTDOWN_TIMEOUT});\n        }\n        await this.launchWindow(uiClientPid, opts);\n      }\n      return true;\n    });\n\n    if (shouldWaitForBoot) {\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\n  /***\n   * Boots simulator and opens simulators UI Client if not already opened.\n   *\n   * @param {boolean} isUiClientRunning - process id of simulator UI client.\n   * @param {RunOptions} opts - arguments to start simulator UI client with.\n   */\n  async launchWindow (isUiClientRunning, opts = {}) {\n    await this.boot();\n    if (!isUiClientRunning) {\n      await this.startUIClient(opts);\n    }\n  }\n\n  /**\n   * Boots Simulator if not already booted.\n   * Does nothing if it is already running.\n   * This API does NOT wait until Simulator is fully booted.\n   *\n   * @throws {Error} If there was a failure while booting the Simulator.\n   */\n  async boot () {\n    const bootEventsEmitter = new EventEmitter();\n    await this.simctl.startBootMonitor({\n      onError: (err) => bootEventsEmitter.emit('failure', err),\n      onFinished: () => bootEventsEmitter.emit('finish'),\n      shouldPreboot: true,\n    });\n    try {\n      await new B((resolve, reject) => {\n        // Historically this call was always asynchronous,\n        // e.g. it was not waiting until Simulator is fully booted.\n        // So we preserve that behavior, and if no errors are received for a while\n        // then we assume the Simulator booting is still in progress.\n        setTimeout(resolve, 3000);\n        bootEventsEmitter.once('failure', (err) => {\n          if (_.includes(err?.message, 'state: Booted')) {\n            resolve();\n          } else {\n            reject(err);\n          }\n        });\n        bootEventsEmitter.once('finish', resolve);\n      });\n    } finally {\n      bootEventsEmitter.removeAllListeners();\n    }\n  }\n\n  /**\n   * Perform verification of device preferences correctness.\n   *\n   * @param {DevicePreferences} prefs [{}] - The preferences to be verified\n   * @throws {Error} If any of the given preference values does not match the expected\n   * format.\n   */\n  verifyDevicePreferences (prefs = {}) {\n    if (_.isEmpty(prefs)) {\n      return;\n    }\n\n    if (!_.isUndefined(prefs.SimulatorWindowLastScale)) {\n      if (!_.isNumber(prefs.SimulatorWindowLastScale) || prefs.SimulatorWindowLastScale <= 0) {\n        log.errorAndThrow(`SimulatorWindowLastScale is expected to be a positive float value. ` +\n          `'${prefs.SimulatorWindowLastScale}' is assigned instead.`);\n      }\n    }\n\n    if (!_.isUndefined(prefs.SimulatorWindowCenter)) {\n      // https://regex101.com/r/2ZXOij/2\n      const verificationPattern = /{-?\\d+(\\.\\d+)?,-?\\d+(\\.\\d+)?}/;\n      if (!_.isString(prefs.SimulatorWindowCenter) || !verificationPattern.test(prefs.SimulatorWindowCenter)) {\n        log.errorAndThrow(`SimulatorWindowCenter is expected to match \"{floatXPosition,floatYPosition}\" format (without spaces). ` +\n          `'${prefs.SimulatorWindowCenter}' is assigned instead.`);\n      }\n    }\n\n    if (!_.isUndefined(prefs.SimulatorWindowOrientation)) {\n      const acceptableValues = ['Portrait', 'LandscapeLeft', 'PortraitUpsideDown', 'LandscapeRight'];\n      if (acceptableValues.indexOf(prefs.SimulatorWindowOrientation) === -1) {\n        log.errorAndThrow(`SimulatorWindowOrientation is expected to be one of ${acceptableValues}. ` +\n          `'${prefs.SimulatorWindowOrientation}' is assigned instead.`);\n      }\n    }\n\n    if (!_.isUndefined(prefs.SimulatorWindowRotationAngle)) {\n      if (!_.isNumber(prefs.SimulatorWindowRotationAngle)) {\n        log.errorAndThrow(`SimulatorWindowRotationAngle is expected to be a valid number. ` +\n          `'${prefs.SimulatorWindowRotationAngle}' is assigned instead.`);\n      }\n    }\n  }\n\n  /**\n   * Update the common iOS Simulator preferences file with new values.\n   * It is necessary to restart the corresponding Simulator before\n   * these changes are applied.\n   *\n   * @param {DevicePreferences} devicePrefs [{}] - The mapping, which represents new device preference values\n   *                                               for the given Simulator.\n   * @param {CommonPreferences} commonPrefs [{}] - The mapping, which represents new common preference values\n   *                                               for all Simulators.\n   * @return {boolean} True if the preferences were successfully updated.\n   */\n  async updatePreferences (devicePrefs = {}, commonPrefs = {}) {\n    if (!_.isEmpty(devicePrefs)) {\n      log.debug(`Setting preferences of ${this.udid} Simulator to ${JSON.stringify(devicePrefs)}`);\n    }\n    if (!_.isEmpty(commonPrefs)) {\n      log.debug(`Setting common Simulator preferences to ${JSON.stringify(commonPrefs)}`);\n    }\n    const homeFolderPath = process.env.HOME;\n    if (!homeFolderPath) {\n      log.warn(`Cannot get the path to HOME folder from the process environment. ` +\n        `Ignoring Simulator preferences update.`);\n      return false;\n    }\n    this.verifyDevicePreferences(devicePrefs);\n    const plistPath = path.resolve(homeFolderPath, 'Library', 'Preferences', 'com.apple.iphonesimulator.plist');\n    return await preferencesPlistGuard.acquire(SimulatorXcode9.name, async () => {\n      const defaults = new NSUserDefaults(plistPath);\n      const prefsToUpdate = _.clone(commonPrefs);\n      try {\n        if (!_.isEmpty(devicePrefs)) {\n          let existingDevicePrefs;\n          const udidKey = this.udid.toUpperCase();\n          if (await fs.exists(plistPath)) {\n            const currentPlistContent = await defaults.asJson();\n            if (_.isPlainObject(currentPlistContent.DevicePreferences)\n                && _.isPlainObject(currentPlistContent.DevicePreferences[udidKey])) {\n              existingDevicePrefs = currentPlistContent.DevicePreferences[udidKey];\n            }\n          }\n          Object.assign(prefsToUpdate, {\n            DevicePreferences: {\n              [udidKey]: Object.assign({}, existingDevicePrefs || {}, devicePrefs)\n            }\n          });\n        }\n        await defaults.update(prefsToUpdate);\n        log.debug(`Updated ${this.udid} Simulator preferences at '${plistPath}' with ` +\n          JSON.stringify(prefsToUpdate));\n        return true;\n      } catch (e) {\n        log.warn(`Cannot update ${this.udid} Simulator preferences at '${plistPath}'. ` +\n          `Try to delete the file manually in order to reset it. Original error: ${e.message}`);\n        return false;\n      }\n    });\n  }\n\n  /**\n   * Reset the current Simulator to the clean state.\n   * @override\n   */\n  async clean () {\n    log.info(`Cleaning simulator ${this.udid}`);\n    await this.simctl.eraseDevice(10000);\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   * @private\n   */\n  async _activateWindow () {\n    let selfName;\n    let selfSdk;\n    let bootedDevicesCount = 0;\n    for (const [sdk, deviceArr] of _.toPairs(await this.simctl.getDevices())) {\n      for (const {state, udid, name} of deviceArr) {\n        if (state === 'Booted') {\n          bootedDevicesCount++;\n        }\n        if (!selfName && udid === this.udid) {\n          selfSdk = sdk;\n          selfName = name;\n        }\n      }\n    }\n    if (bootedDevicesCount < 2) {\n      return await super._activateWindow();\n    }\n\n    // There are potentially more that one Simulator window\n    return `\n      tell application \"System Events\"\n        tell process \"Simulator\"\n          set frontmost to false\n          set frontmost to true\n          click (menu item 1 where (its name contains \"${selfName} \" and its name contains \"${selfSdk}\")) of menu 1 of menu bar item \"Window\" of menu bar 1\n        end tell\n      end tell\n    `;\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async isBiometricEnrolled () {\n    const {stdout} = await this.simctl.spawnProcess([\n      'notifyutil',\n      '-g', ENROLLMENT_NOTIFICATION_RECEIVER\n    ]);\n    const match = (new RegExp(`${_.escapeRegExp(ENROLLMENT_NOTIFICATION_RECEIVER)}\\\\s+([01])`))\n      .exec(stdout);\n    if (!match) {\n      throw new Error(`Cannot parse biometric enrollment state from '${stdout}'`);\n    }\n    log.info(`Current biometric enrolled state for ${this.udid} Simulator: ${match[1]}`);\n    return match[1] === '1';\n  }\n\n  /**\n   * @inheritdoc\n   * @override\n   */\n  async enrollBiometric (isEnabled = true) {\n    log.debug(`Setting biometric enrolled state for ${this.udid} Simulator to '${isEnabled ? 'enabled' : 'disabled'}'`);\n    await this.simctl.spawnProcess([\n      'notifyutil',\n      '-s', ENROLLMENT_NOTIFICATION_RECEIVER, isEnabled ? '1' : '0'\n    ]);\n    await this.simctl.spawnProcess([\n      'notifyutil',\n      '-p', ENROLLMENT_NOTIFICATION_RECEIVER\n    ]);\n    if (await this.isBiometricEnrolled() !== isEnabled) {\n      throw new Error(`Cannot set biometric enrolled state for ${this.udid} Simulator to '${isEnabled ? 'enabled' : 'disabled'}'`);\n    }\n  }\n\n  /**\n   * Sends a notification to match/not match the particular biometric.\n   * @override\n   *\n   * @param {?boolean} shouldMatch [true] - Set it to true or false in order to emulate\n   * matching/not matching the corresponding biometric\n   * @param {?string} biometricName [touchId] - Either touchId or faceId (faceId is only available since iOS 11)\n   */\n  async sendBiometricMatch (shouldMatch = true, biometricName = 'touchId') {\n    const domainComponent = toBiometricDomainComponent(biometricName);\n    const domain = `com.apple.BiometricKit_Sim.${domainComponent}.${shouldMatch ? '' : 'no'}match`;\n    await this.simctl.spawnProcess([\n      'notifyutil',\n      '-p', domain\n    ]);\n    log.info(`Sent notification ${domain} to ${shouldMatch ? 'match' : 'not match'} ${biometricName} biometric ` +\n      `for ${this.udid} Simulator`);\n  }\n\n  /**\n   * @override\n   */\n  async getLaunchDaemonsRoot () {\n    const devRoot = await getDeveloperRoot();\n    return path.resolve(devRoot,\n      'Platforms/iPhoneOS.platform/Developer/Library/CoreSimulator/Profiles/Runtimes/iOS.simruntime/Contents/Resources/RuntimeRoot/System/Library/LaunchDaemons');\n  }\n\n  /**\n   * @typedef {Object} KeyboardOptions\n   * @property {!string} name The name of the keyboard locale, for example `en_US` or `de_CH`\n   * @property {!string} layout The keyboard layout, for example `QUERTY` or `Ukrainian`\n   * @property {?string} hardware Could either be `Automatic` or `null`\n   */\n\n  /**\n   * @typedef {Object} LanguageOptions\n   * @property {!string} name The name of the language, for example `de` or `zh-Hant-CN`\n   */\n\n  /**\n   * @typedef {Object} LocaleOptions\n   * @property {!string} name The name of the system locale, for example `de_CH` or `zh_CN`\n   * @property {?string} calendar Optional calendar format, for example `gregorian` or `persian`\n   */\n\n  /**\n   * @typedef {Object} LocalizationOptions\n   * @property {?KeyboardOptions} keyboard\n   * @property {?LanguageOptions} language\n   * @property {?LocaleOptions} locale\n   */\n\n  /**\n   * Change localization settings on the currently booted simulator\n   *\n   * @param {?LocalizationOptions} opts\n   * @throws {Error} If there was a failure while setting the preferences\n   * @returns {boolean} `true` if any of settings has been successfully changed\n   */\n  async configureLocalization (opts = {}) {\n    if (_.isEmpty(opts)) {\n      return false;\n    }\n\n    const { language, locale, keyboard } = opts;\n    const globalPrefs = {};\n    let keyboardId = null;\n    if (_.isPlainObject(keyboard)) {\n      const { name, layout, hardware } = keyboard;\n      if (!name) {\n        throw new Error(`The 'keyboard' field must have a valid name set`);\n      }\n      if (!layout) {\n        throw new Error(`The 'keyboard' field must have a valid layout set`);\n      }\n      keyboardId = `${name}@sw=${layout}`;\n      if (hardware) {\n        keyboardId += `;@hw=${hardware}`;\n      }\n      globalPrefs.AppleKeyboards = [keyboardId];\n    }\n    if (_.isPlainObject(language)) {\n      const { name } = language;\n      if (!name) {\n        throw new Error(`The 'language' field must have a valid name set`);\n      }\n      globalPrefs.AppleLanguages = [name];\n    }\n    if (_.isPlainObject(locale)) {\n      const { name, calendar } = locale;\n      if (!name) {\n        throw new Error(`The 'locale' field must have a valid name set`);\n      }\n      let localeId = name;\n      if (calendar) {\n        localeId += `@calendar=${calendar}`;\n      }\n      globalPrefs.AppleLocale = localeId;\n    }\n    if (_.isEmpty(globalPrefs)) {\n      return false;\n    }\n\n    const argChunks = generateDefaultsCommandArgs(globalPrefs, true);\n    await B.all(argChunks.map((args) => this.simctl.spawnProcess([\n      'defaults', 'write', '.GlobalPreferences.plist', ...args\n    ])));\n\n    if (keyboardId) {\n      const argChunks = generateDefaultsCommandArgs({\n        KeyboardsCurrentAndNext: [keyboardId],\n        KeyboardLastUsed: keyboardId,\n        KeyboardLastUsedForLanguage: { [keyboard.name]: keyboardId }\n      }, true);\n      await B.all(argChunks.map((args) => this.simctl.spawnProcess([\n        'defaults', 'write', 'com.apple.Preferences', ...args\n      ])));\n    }\n\n    return true;\n  }\n\n}\n\nexport default SimulatorXcode9;\n"],"file":"lib/simulator-xcode-9.js","sourceRoot":"../.."}
|
package/lib/simulator-xcode-9.js
CHANGED
|
@@ -4,10 +4,11 @@ import path from 'path';
|
|
|
4
4
|
import { fs, timing } from 'appium-support';
|
|
5
5
|
import AsyncLock from 'async-lock';
|
|
6
6
|
import log from './logger';
|
|
7
|
-
import { waitForCondition
|
|
7
|
+
import { waitForCondition } from 'asyncbox';
|
|
8
8
|
import { toBiometricDomainComponent, getDeveloperRoot } from './utils.js';
|
|
9
9
|
import { NSUserDefaults, generateDefaultsCommandArgs } from './defaults-utils';
|
|
10
10
|
import B from 'bluebird';
|
|
11
|
+
import { EventEmitter } from 'events';
|
|
11
12
|
|
|
12
13
|
const SIMULATOR_SHUTDOWN_TIMEOUT = 15 * 1000;
|
|
13
14
|
const startupLock = new AsyncLock();
|
|
@@ -177,30 +178,38 @@ class SimulatorXcode9 extends SimulatorXcode8 {
|
|
|
177
178
|
}
|
|
178
179
|
|
|
179
180
|
/**
|
|
180
|
-
* Boots
|
|
181
|
+
* Boots Simulator if not already booted.
|
|
181
182
|
* Does nothing if it is already running.
|
|
183
|
+
* This API does NOT wait until Simulator is fully booted.
|
|
182
184
|
*
|
|
183
|
-
* @throws {Error} If
|
|
185
|
+
* @throws {Error} If there was a failure while booting the Simulator.
|
|
184
186
|
*/
|
|
185
187
|
async boot () {
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
188
|
+
const bootEventsEmitter = new EventEmitter();
|
|
189
|
+
await this.simctl.startBootMonitor({
|
|
190
|
+
onError: (err) => bootEventsEmitter.emit('failure', err),
|
|
191
|
+
onFinished: () => bootEventsEmitter.emit('finish'),
|
|
192
|
+
shouldPreboot: true,
|
|
193
|
+
});
|
|
192
194
|
try {
|
|
193
|
-
await
|
|
194
|
-
|
|
195
|
-
|
|
195
|
+
await new B((resolve, reject) => {
|
|
196
|
+
// Historically this call was always asynchronous,
|
|
197
|
+
// e.g. it was not waiting until Simulator is fully booted.
|
|
198
|
+
// So we preserve that behavior, and if no errors are received for a while
|
|
199
|
+
// then we assume the Simulator booting is still in progress.
|
|
200
|
+
setTimeout(resolve, 3000);
|
|
201
|
+
bootEventsEmitter.once('failure', (err) => {
|
|
202
|
+
if (_.includes(err?.message, 'state: Booted')) {
|
|
203
|
+
resolve();
|
|
204
|
+
} else {
|
|
205
|
+
reject(err);
|
|
206
|
+
}
|
|
207
|
+
});
|
|
208
|
+
bootEventsEmitter.once('finish', resolve);
|
|
209
|
+
});
|
|
210
|
+
} finally {
|
|
211
|
+
bootEventsEmitter.removeAllListeners();
|
|
196
212
|
}
|
|
197
|
-
|
|
198
|
-
await retryInterval(5, 1000, async () => {
|
|
199
|
-
if (!await this.isRunning()) {
|
|
200
|
-
throw new Error(`Simulator '${this.udid}' is not running after being booted. ` +
|
|
201
|
-
`Check the Appium log for more details.`);
|
|
202
|
-
}
|
|
203
|
-
});
|
|
204
213
|
}
|
|
205
214
|
|
|
206
215
|
/**
|
package/package.json
CHANGED
|
@@ -4,7 +4,7 @@
|
|
|
4
4
|
"keywords": [
|
|
5
5
|
"appium"
|
|
6
6
|
],
|
|
7
|
-
"version": "3.28.
|
|
7
|
+
"version": "3.28.2",
|
|
8
8
|
"author": "appium",
|
|
9
9
|
"license": "Apache-2.0",
|
|
10
10
|
"repository": {
|
|
@@ -36,7 +36,7 @@
|
|
|
36
36
|
"asyncbox": "^2.3.1",
|
|
37
37
|
"bluebird": "^3.5.1",
|
|
38
38
|
"lodash": "^4.2.1",
|
|
39
|
-
"node-simctl": "^6.
|
|
39
|
+
"node-simctl": "^6.6.0",
|
|
40
40
|
"semver": "^7.0.0",
|
|
41
41
|
"source-map-support": "^0.5.3",
|
|
42
42
|
"teen_process": "^1.3.0",
|