appium-xcode
Advanced tools
Comparing version 2.0.2 to 2.0.3
@@ -41,2 +41,6 @@ 'use strict'; | ||
var _plist = require('plist'); | ||
var _plist2 = _interopRequireWildcard(_plist); | ||
var exec = _support2['default'].core.exec; | ||
@@ -320,4 +324,2 @@ var util = _support2['default'].util; | ||
// TODO remove this function. should really just be using 'getVersion()' everywhere. | ||
// This was added for tech-debt backwards compatibility | ||
function getMaxIOSSDKWithoutRetry() { | ||
@@ -377,2 +379,46 @@ var version, cmd, _ref4, _ref42, stdout, sdkVersion, match; | ||
function getConnectedDevices() { | ||
var _ref5, _ref52, stdout, plistContent, devicesFound, entriesToSearch, currentEntry, deviceInfo; | ||
return _regeneratorRuntime.async(function getConnectedDevices$(context$1$0) { | ||
while (1) switch (context$1$0.prev = context$1$0.next) { | ||
case 0: | ||
context$1$0.next = 2; | ||
return exec('/usr/sbin/system_profiler -xml SPUSBDataType', { maxBuffer: 524288, timeout: XCODE_SELECT_TIMEOUT }); | ||
case 2: | ||
_ref5 = context$1$0.sent; | ||
_ref52 = _slicedToArray(_ref5, 1); | ||
stdout = _ref52[0]; | ||
plistContent = _plist2['default'].parse(stdout); | ||
devicesFound = []; | ||
entriesToSearch = [plistContent[0]]; | ||
while (entriesToSearch.length > 0) { | ||
currentEntry = entriesToSearch.pop(); | ||
if (currentEntry instanceof Array) { | ||
entriesToSearch = entriesToSearch.concat(currentEntry); | ||
} else if (currentEntry._name && currentEntry._name.substring(0, 4) === 'iPad' || currentEntry._name && currentEntry._name.substring(0, 6) === 'iPhone') { | ||
deviceInfo = { | ||
name: currentEntry._name, | ||
udid: currentEntry.serial_num, | ||
productId: currentEntry.product_id, | ||
deviceVersion: currentEntry.bcd_device | ||
}; | ||
devicesFound.push(deviceInfo); | ||
} else if (currentEntry._items) { | ||
entriesToSearch = entriesToSearch.concat(currentEntry._items); | ||
} | ||
} | ||
return context$1$0.abrupt('return', devicesFound); | ||
case 10: | ||
case 'end': | ||
return context$1$0.stop(); | ||
} | ||
}, null, this); | ||
} | ||
function clearInternalCache() { | ||
@@ -396,2 +442,3 @@ | ||
exports.getMaxIOSSDKWithoutRetry = getMaxIOSSDKWithoutRetry; | ||
exports.getConnectedDevices = getConnectedDevices; | ||
exports.clearInternalCache = clearInternalCache; | ||
@@ -417,2 +464,2 @@ // Xcode < 5.x | ||
// rather than waste time getting the iOSSDKVersion, just get both paths and see which one exists | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["lib/xcode.js"],"names":[],"mappings":"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","file":"lib/xcode.js","sourcesContent":["import npmlog from 'npmlog';\nimport support from 'appium-support';\nimport fs from 'fs';\nimport denodeify from 'denodeify';\nimport path from 'path';\nimport { retry } from 'asyncbox';\nimport _ from 'lodash';\n\nconst exec = support.core.exec;\nconst util = support.util;\nconst fileExists = support.util.fileExists;\nconst escapeSpace = support.util.escapeSpace;\nconst readSymlink = denodeify(fs.readlink);\nconst env = process.env;\n\nconst XCODE_SELECT_TIMEOUT = 3000;\nconst XCODE_SUBDIR = \"/Contents/Developer\";\nconst DEFAULT_NUMBER_OF_RETRIES = 3;\n\nconst log = process.env.GLOBAL_NPMLOG ? global.log : npmlog;\n\n\nfunction hasExpectedSubDir (path) {\n  return path.substring(path.length - XCODE_SUBDIR.length) === XCODE_SUBDIR;\n}\n\nasync function getPathFromSymlink (failMessage) {\n  // Node's invocation of xcode-select sometimes flakes and returns an empty string.\n  // Not clear why. As a workaround, Appium can reliably deduce the version in use by checking\n  // the locations xcode-select uses to store the selected version's path. This should be 100%\n  // reliable so long as the link locations remain the same. However, since we're relying on\n  // hardcoded paths, this approach will break the next time Apple changes the symlink location.\n  log.warn(`Finding XcodePath by symlink because ${failMessage}`);\n\n  const symlinkPath = \"/var/db/xcode_select_link\";\n  const legacySymlinkPath = \"/usr/share/xcode-select/xcode_dir_link\"; //  Xcode < 5.x\n  let xcodePath = null;\n\n  // xcode-select allows users to override its settings with the DEVELOPER_DIR env var,\n  // so check that first\n  if (util.hasContent(env.DEVELOPER_DIR)) {\n    const customPath = hasExpectedSubDir(env.DEVELOPER_DIR) ?\n                                         env.DEVELOPER_DIR  :\n                                         env.DEVELOPER_DIR + XCODE_SUBDIR;\n\n    if (await fileExists(customPath)) {\n      xcodePath = customPath;\n    } else {\n      let mesg = `Could not find path to Xcode, environment variable ` +\n                 `DEVELOPER_DIR set to: ${env.DEVELOPER_DIR} ` +\n                 `but no Xcode found`;\n      log.warn(mesg);\n      throw new Error(mesg);\n    }\n  } else if (await fileExists(symlinkPath)) {\n    xcodePath = readSymlink(symlinkPath);\n  } else if (await fileExists(legacySymlinkPath)) {\n    xcodePath = readSymlink(legacySymlinkPath);\n  }\n\n  if (xcodePath) {\n    return xcodePath.replace(new RegExp(\"/$\"), \"\").trim();\n  }\n\n  // We should only get here is we failed to capture xcode-select's stdout and our\n  // other checks failed. Either Apple has moved the symlink to a new location or the user\n  // is not using the default install. 99.999% chance it's the latter, so issue a warning\n  // should we ever hit the edge case.\n  let msg = `Could not find path to Xcode by symlinks located in ${symlinkPath}, or ${legacySymlinkPath}`;\n  log.warn(msg);\n  throw new Error(msg);\n\n}\n\nasync function getPathFromXcodeSelect () {\n\n  let [stdout] = await exec('xcode-select --print-path', {maxBuffer: 524288, timeout: XCODE_SELECT_TIMEOUT});\n\n  // trim and remove trailing slash\n  const xcodeFolderPath = stdout.replace(new RegExp(\"/$\"), \"\").trim();\n\n  if (!util.hasContent(xcodeFolderPath)) {\n    throw new Error(\"xcode-select returned an empty string\");\n  }\n\n  if (await fileExists(xcodeFolderPath)) {\n    return xcodeFolderPath;\n  } else {\n    const msg = `xcode-select could not find xcode. Path: ${xcodeFolderPath} does not exist.`;\n    log.error(msg);\n    throw new Error(msg);\n  }\n}\n\nconst getPath = _.memoize(function () {\n\n  // first we try using xcode-select to find the path\n  // then we try using the symlinks that Apple has by default\n\n  return getPathFromXcodeSelect().catch(getPathFromSymlink);\n});\n\n\n\nasync function getVersionWithoutRetry () {\n\n  let xcodePath = await getPath();\n\n  // we want to read the CFBundleShortVersionString from Xcode's plist.\n  // It should be in /[root]/XCode.app/Contents/\n  const plistPath = xcodePath.replace(XCODE_SUBDIR, \"/Contents/Info.plist\");\n\n  if (!await fileExists(plistPath)) {\n    throw new Error(`Could not get Xcode version. ${plistPath} does not exist on disk.`);\n  }\n\n  const cmd = `/usr/libexec/PlistBuddy -c 'Print CFBundleShortVersionString' ${escapeSpace(plistPath)}`;\n  let [stdout] = await exec(cmd, {maxBuffer: 524288, timeout: XCODE_SELECT_TIMEOUT});\n\n  let versionPattern = /\\d\\.\\d\\.*\\d*/;\n  // need to use string#match here; previous code used regexp#exec, which does not return null\n  let match = stdout.match(versionPattern);\n  if (match === null || !util.hasContent(match[0])) {\n    throw new Error(`Could not parse Xcode version. xcodebuild output was: ${stdout}`);\n  }\n\n  return match[0];\n}\n\n\nconst getVersion = _.memoize(\n  function (retries = DEFAULT_NUMBER_OF_RETRIES) {\n    return retry(retries, getVersionWithoutRetry);\n  }\n);\n\nasync function getAutomationTraceTemplatePathWithoutRetry () {\n\n  const xcodePath = await getPath();\n\n  // for ios 8 and up, the file extension for AutiomationInstrument changed.\n  // rather than waste time getting the iOSSDKVersion, just get both paths and see which one exists\n  const extensions = ['xrplugin', 'bundle'];\n  const pathPrefix = path.resolve(xcodePath, \"../Applications/Instruments.app/Contents/PlugIns\");\n  const pathSuffix = \"Contents/Resources/Automation.tracetemplate\";\n  let automationTraceTemplatePaths = [\n    path.resolve(pathPrefix, \"AutomationInstrument.\" + extensions[0], pathSuffix),\n    path.resolve(pathPrefix, \"AutomationInstrument.\" + extensions[1], pathSuffix)\n  ];\n\n  if (await fileExists(automationTraceTemplatePaths[0])) {\n    return automationTraceTemplatePaths[0];\n  }\n\n  if (await fileExists(automationTraceTemplatePaths[1])) {\n    return automationTraceTemplatePaths[1];\n  }\n\n  const msg = \"Could not find Automation.tracetemplate in any of the following\" +\n              `locations ${automationTraceTemplatePaths.toString()}`;\n  log.error(msg);\n  throw new Error(msg);\n\n}\n\nconst getAutomationTraceTemplatePath = _.memoize(\n  function (retries = DEFAULT_NUMBER_OF_RETRIES) {\n    return retry(retries, getAutomationTraceTemplatePathWithoutRetry);\n  }\n);\n\n// TODO remove this function. should really just be using 'getVersion()' everywhere.\n// This was added for tech-debt backwards compatibility\nasync function getMaxIOSSDKWithoutRetry () {\n\n  const version = await getVersion();\n\n  if (version[0] === '4') {\n    return '6.1';\n  }\n\n  const cmd = `xcrun --sdk iphonesimulator --show-sdk-version`;\n  const [stdout] = await exec(cmd, {maxBuffer: 524288, timeout: XCODE_SELECT_TIMEOUT});\n\n  const sdkVersion = stdout.trim();\n  const match = /\\d.\\d/.exec(stdout);\n\n  if (!match) {\n    throw new Error(`xcrun returned a non-numeric iOS SDK version: ${sdkVersion}`);\n  }\n\n  return sdkVersion;\n}\n\nconst getMaxIOSSDK = _.memoize(\n  function (retries = DEFAULT_NUMBER_OF_RETRIES) {\n    return retry(retries, getMaxIOSSDKWithoutRetry);\n  }\n);\n\nfunction clearInternalCache () {\n\n  // memoized functions\n  const memoized = [getPath, getVersion, getAutomationTraceTemplatePath,\n                    getMaxIOSSDK];\n\n  memoized.forEach((f) => {\n    if (f.cache) {\n      f.cache = new _.memoize.Cache();\n    }\n  });\n\n\n}\n\nexport { getPath, getVersion, getAutomationTraceTemplatePath, getMaxIOSSDK,\n         getAutomationTraceTemplatePathWithoutRetry, getMaxIOSSDKWithoutRetry,\n         clearInternalCache };\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["lib/xcode.js"],"names":[],"mappings":"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","file":"lib/xcode.js","sourcesContent":["import npmlog from 'npmlog';\nimport support from 'appium-support';\nimport fs from 'fs';\nimport denodeify from 'denodeify';\nimport path from 'path';\nimport { retry } from 'asyncbox';\nimport _ from 'lodash';\nimport plist from 'plist';\n\nconst exec = support.core.exec;\nconst util = support.util;\nconst fileExists = support.util.fileExists;\nconst escapeSpace = support.util.escapeSpace;\nconst readSymlink = denodeify(fs.readlink);\nconst env = process.env;\n\nconst XCODE_SELECT_TIMEOUT = 3000;\nconst XCODE_SUBDIR = \"/Contents/Developer\";\nconst DEFAULT_NUMBER_OF_RETRIES = 3;\n\nconst log = process.env.GLOBAL_NPMLOG ? global.log : npmlog;\n\n\nfunction hasExpectedSubDir (path) {\n  return path.substring(path.length - XCODE_SUBDIR.length) === XCODE_SUBDIR;\n}\n\nasync function getPathFromSymlink (failMessage) {\n  // Node's invocation of xcode-select sometimes flakes and returns an empty string.\n  // Not clear why. As a workaround, Appium can reliably deduce the version in use by checking\n  // the locations xcode-select uses to store the selected version's path. This should be 100%\n  // reliable so long as the link locations remain the same. However, since we're relying on\n  // hardcoded paths, this approach will break the next time Apple changes the symlink location.\n  log.warn(`Finding XcodePath by symlink because ${failMessage}`);\n\n  const symlinkPath = \"/var/db/xcode_select_link\";\n  const legacySymlinkPath = \"/usr/share/xcode-select/xcode_dir_link\"; //  Xcode < 5.x\n  let xcodePath = null;\n\n  // xcode-select allows users to override its settings with the DEVELOPER_DIR env var,\n  // so check that first\n  if (util.hasContent(env.DEVELOPER_DIR)) {\n    const customPath = hasExpectedSubDir(env.DEVELOPER_DIR) ?\n                                         env.DEVELOPER_DIR  :\n                                         env.DEVELOPER_DIR + XCODE_SUBDIR;\n\n    if (await fileExists(customPath)) {\n      xcodePath = customPath;\n    } else {\n      let mesg = `Could not find path to Xcode, environment variable ` +\n                 `DEVELOPER_DIR set to: ${env.DEVELOPER_DIR} ` +\n                 `but no Xcode found`;\n      log.warn(mesg);\n      throw new Error(mesg);\n    }\n  } else if (await fileExists(symlinkPath)) {\n    xcodePath = readSymlink(symlinkPath);\n  } else if (await fileExists(legacySymlinkPath)) {\n    xcodePath = readSymlink(legacySymlinkPath);\n  }\n\n  if (xcodePath) {\n    return xcodePath.replace(new RegExp(\"/$\"), \"\").trim();\n  }\n\n  // We should only get here is we failed to capture xcode-select's stdout and our\n  // other checks failed. Either Apple has moved the symlink to a new location or the user\n  // is not using the default install. 99.999% chance it's the latter, so issue a warning\n  // should we ever hit the edge case.\n  let msg = `Could not find path to Xcode by symlinks located in ${symlinkPath}, or ${legacySymlinkPath}`;\n  log.warn(msg);\n  throw new Error(msg);\n\n}\n\nasync function getPathFromXcodeSelect () {\n\n  let [stdout] = await exec('xcode-select --print-path', {maxBuffer: 524288, timeout: XCODE_SELECT_TIMEOUT});\n\n  // trim and remove trailing slash\n  const xcodeFolderPath = stdout.replace(new RegExp(\"/$\"), \"\").trim();\n\n  if (!util.hasContent(xcodeFolderPath)) {\n    throw new Error(\"xcode-select returned an empty string\");\n  }\n\n  if (await fileExists(xcodeFolderPath)) {\n    return xcodeFolderPath;\n  } else {\n    const msg = `xcode-select could not find xcode. Path: ${xcodeFolderPath} does not exist.`;\n    log.error(msg);\n    throw new Error(msg);\n  }\n}\n\nconst getPath = _.memoize(function () {\n\n  // first we try using xcode-select to find the path\n  // then we try using the symlinks that Apple has by default\n\n  return getPathFromXcodeSelect().catch(getPathFromSymlink);\n});\n\n\n\nasync function getVersionWithoutRetry () {\n\n  let xcodePath = await getPath();\n\n  // we want to read the CFBundleShortVersionString from Xcode's plist.\n  // It should be in /[root]/XCode.app/Contents/\n  const plistPath = xcodePath.replace(XCODE_SUBDIR, \"/Contents/Info.plist\");\n\n  if (!await fileExists(plistPath)) {\n    throw new Error(`Could not get Xcode version. ${plistPath} does not exist on disk.`);\n  }\n\n  const cmd = `/usr/libexec/PlistBuddy -c 'Print CFBundleShortVersionString' ${escapeSpace(plistPath)}`;\n  let [stdout] = await exec(cmd, {maxBuffer: 524288, timeout: XCODE_SELECT_TIMEOUT});\n\n  let versionPattern = /\\d\\.\\d\\.*\\d*/;\n  // need to use string#match here; previous code used regexp#exec, which does not return null\n  let match = stdout.match(versionPattern);\n  if (match === null || !util.hasContent(match[0])) {\n    throw new Error(`Could not parse Xcode version. xcodebuild output was: ${stdout}`);\n  }\n\n  return match[0];\n}\n\n\nconst getVersion = _.memoize(\n  function (retries = DEFAULT_NUMBER_OF_RETRIES) {\n    return retry(retries, getVersionWithoutRetry);\n  }\n);\n\nasync function getAutomationTraceTemplatePathWithoutRetry () {\n\n  const xcodePath = await getPath();\n\n  // for ios 8 and up, the file extension for AutiomationInstrument changed.\n  // rather than waste time getting the iOSSDKVersion, just get both paths and see which one exists\n  const extensions = ['xrplugin', 'bundle'];\n  const pathPrefix = path.resolve(xcodePath, \"../Applications/Instruments.app/Contents/PlugIns\");\n  const pathSuffix = \"Contents/Resources/Automation.tracetemplate\";\n  let automationTraceTemplatePaths = [\n    path.resolve(pathPrefix, \"AutomationInstrument.\" + extensions[0], pathSuffix),\n    path.resolve(pathPrefix, \"AutomationInstrument.\" + extensions[1], pathSuffix)\n  ];\n\n  if (await fileExists(automationTraceTemplatePaths[0])) {\n    return automationTraceTemplatePaths[0];\n  }\n\n  if (await fileExists(automationTraceTemplatePaths[1])) {\n    return automationTraceTemplatePaths[1];\n  }\n\n  const msg = \"Could not find Automation.tracetemplate in any of the following\" +\n              `locations ${automationTraceTemplatePaths.toString()}`;\n  log.error(msg);\n  throw new Error(msg);\n\n}\n\nconst getAutomationTraceTemplatePath = _.memoize(\n  function (retries = DEFAULT_NUMBER_OF_RETRIES) {\n    return retry(retries, getAutomationTraceTemplatePathWithoutRetry);\n  }\n);\n\nasync function getMaxIOSSDKWithoutRetry () {\n\n  const version = await getVersion();\n\n  if (version[0] === '4') {\n    return '6.1';\n  }\n\n  const cmd = `xcrun --sdk iphonesimulator --show-sdk-version`;\n  const [stdout] = await exec(cmd, {maxBuffer: 524288, timeout: XCODE_SELECT_TIMEOUT});\n\n  const sdkVersion = stdout.trim();\n  const match = /\\d.\\d/.exec(stdout);\n\n  if (!match) {\n    throw new Error(`xcrun returned a non-numeric iOS SDK version: ${sdkVersion}`);\n  }\n\n  return sdkVersion;\n}\n\nconst getMaxIOSSDK = _.memoize(\n  function (retries = DEFAULT_NUMBER_OF_RETRIES) {\n    return retry(retries, getMaxIOSSDKWithoutRetry);\n  }\n);\n\nasync function getConnectedDevices () {\n\n  let [stdout] = await exec('/usr/sbin/system_profiler -xml SPUSBDataType',\n    {maxBuffer: 524288, timeout: XCODE_SELECT_TIMEOUT});\n  let plistContent = plist.parse(stdout);\n\n  let devicesFound = [];\n  let entriesToSearch = [plistContent[0]];\n  while (entriesToSearch.length > 0) {\n    let currentEntry = entriesToSearch.pop();\n    if (currentEntry instanceof Array) {\n      entriesToSearch = entriesToSearch.concat(currentEntry);\n    } else if ((currentEntry._name &&\n                currentEntry._name.substring(0, 4) === \"iPad\") ||\n               (currentEntry._name &&\n                currentEntry._name.substring(0, 6) === \"iPhone\")) {\n      let deviceInfo = {\n        name: currentEntry._name,\n        udid: currentEntry.serial_num,\n        productId: currentEntry.product_id,\n        deviceVersion: currentEntry.bcd_device\n      };\n      devicesFound.push(deviceInfo);\n    } else if (currentEntry._items) {\n      entriesToSearch = entriesToSearch.concat(currentEntry._items);\n    }\n  }\n  return devicesFound;\n}\n\nfunction clearInternalCache () {\n\n  // memoized functions\n  const memoized = [getPath, getVersion, getAutomationTraceTemplatePath,\n                    getMaxIOSSDK];\n\n  memoized.forEach((f) => {\n    if (f.cache) {\n      f.cache = new _.memoize.Cache();\n    }\n  });\n\n\n}\n\nexport { getPath, getVersion, getAutomationTraceTemplatePath, getMaxIOSSDK,\n         getAutomationTraceTemplatePathWithoutRetry, getMaxIOSSDKWithoutRetry,\n         getConnectedDevices, clearInternalCache };\n"]} |
@@ -206,3 +206,24 @@ 'use strict'; | ||
}); | ||
it('should get a list of iOS devices', function callee$1$6() { | ||
var devices; | ||
return _regeneratorRuntime.async(function callee$1$6$(context$2$0) { | ||
while (1) switch (context$2$0.prev = context$2$0.next) { | ||
case 0: | ||
context$2$0.next = 2; | ||
return xcode.getConnectedDevices(); | ||
case 2: | ||
devices = context$2$0.sent; | ||
should.exist(devices); | ||
(typeof devices).should.equal('object'); | ||
case 5: | ||
case 'end': | ||
return context$2$0.stop(); | ||
} | ||
}, null, _this); | ||
}); | ||
}); | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInRlc3QveGNvZGUtc3BlY3MuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7c0JBRXVCLGNBQWM7O0lBQXpCLEtBQUs7O29CQUNBLE1BQU07Ozs7OEJBQ0ksa0JBQWtCOzs7O1FBQ3RDLFdBQVc7O2tCQUNILElBQUk7Ozs7eUJBQ0csV0FBVzs7Ozt1QkFDbkIsUUFBUTs7OztBQVJ0QixPQUFPLENBQUMsb0JBQW9CLENBQUMsQ0FBQyxPQUFPLEVBQUUsQ0FBQzs7QUFVeEMsSUFBSSxNQUFNLEdBQUcsa0JBQUssTUFBTSxFQUFFLENBQUM7QUFDM0Isa0JBQUssR0FBRyw2QkFBZ0IsQ0FBQzs7QUFFekIsSUFBSSxVQUFVLEdBQUcsdUJBQVUsZ0JBQUcsSUFBSSxDQUFDLENBQUM7O0FBRXBDLFFBQVEsQ0FBQyxtQkFBbUIsRUFBRSxZQUFNOztBQUVsQyxJQUFFLENBQUMseUNBQXlDLEVBQUU7UUFFeEMsSUFBSTs7Ozs7aUJBQVMsS0FBSyxDQUFDLE9BQU8sRUFBRTs7O0FBQTVCLGNBQUk7O0FBQ1IsZ0JBQU0sQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7O2lCQUNiLFVBQVUsQ0FBQyxJQUFJLENBQUM7Ozs7Ozs7R0FFdkIsQ0FBQyxDQUFDOztBQUVILElBQUUsQ0FBQyxpQ0FBaUMsRUFBRTtRQUVoQyxPQUFPOzs7OztpQkFBUyxLQUFLLENBQUMsVUFBVSxFQUFFOzs7QUFBbEMsaUJBQU87O0FBQ1gsZ0JBQU0sQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7QUFDdEIsOEJBQUUsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLE1BQU0sQ0FBQyxFQUFFLFFBQUssQ0FBQztBQUNuQyx3QkFBYyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQyxNQUFNLENBQUMsRUFBRSxRQUFLLENBQUM7Ozs7Ozs7R0FDN0MsQ0FBQyxDQUFDOztBQUVILElBQUUsQ0FBQyxnRUFBZ0UsRUFBRTtRQUsvRCxNQUFNLEVBQ04sSUFBSSxFQUNKLEtBQUssRUFPTCxPQUFPOzs7OztpQkFaTCxLQUFLLENBQUMsT0FBTyxFQUFFOzs7O2lCQUNmLEtBQUssQ0FBQyxVQUFVLEVBQUU7OztBQUVwQixnQkFBTSxHQUFHLElBQUksSUFBSSxFQUFFOztpQkFDTixLQUFLLENBQUMsT0FBTyxFQUFFOzs7QUFBNUIsY0FBSTtBQUNKLGVBQUssR0FBRyxJQUFJLElBQUksRUFBRTs7QUFFdEIsZ0JBQU0sQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7O2lCQUNiLFVBQVUsQ0FBQyxJQUFJLENBQUM7OztBQUN0QixXQUFDLEtBQUssR0FBQyxNQUFNLENBQUEsQ0FBRSxNQUFNLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7O0FBRXBDLGdCQUFNLEdBQUcsSUFBSSxJQUFJLEVBQUUsQ0FBQzs7aUJBQ0EsS0FBSyxDQUFDLFVBQVUsRUFBRTs7O0FBQWxDLGlCQUFPOztBQUNYLGVBQUssR0FBRyxJQUFJLElBQUksRUFBRSxDQUFDOztBQUVuQixnQkFBTSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztBQUN0Qiw4QkFBRSxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsTUFBTSxDQUFDLEVBQUUsUUFBSyxDQUFDO0FBQ25DLHdCQUFjLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDLE1BQU0sQ0FBQyxFQUFFLFFBQUssQ0FBQztBQUM1QyxXQUFDLEtBQUssR0FBQyxNQUFNLENBQUEsQ0FBRSxNQUFNLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7Ozs7Ozs7R0FFckMsQ0FBQyxDQUFDOztBQUVILElBQUUsQ0FBQyxvQ0FBb0MsRUFBRTtRQUluQyxNQUFNLEVBRU4sS0FBSzs7Ozs7QUFKVCxlQUFLLENBQUMsa0JBQWtCLEVBQUUsQ0FBQzs7QUFFdkIsZ0JBQU0sR0FBRyxJQUFJLElBQUksRUFBRTs7aUJBQ2pCLEtBQUssQ0FBQyxPQUFPLEVBQUU7OztBQUNqQixlQUFLLEdBQUcsSUFBSSxJQUFJLEVBQUU7O0FBQ3RCLFdBQUMsS0FBSyxHQUFDLE1BQU0sQ0FBQSxDQUFFLE1BQU0sQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQzs7Ozs7OztHQUV0QyxDQUFDLENBQUM7O0FBRUgsSUFBRSxDQUFDLDJDQUEyQyxFQUFFO1FBQzFDLElBQUksRUFJSixNQUFNOzs7OztpQkFKTyxLQUFLLENBQUMsOEJBQThCLEVBQUU7OztBQUFuRCxjQUFJOztBQUVSLGdCQUFNLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDO0FBQ25CLG9CQUFVLENBQUMsSUFBSSxDQUFDLENBQUMsTUFBTSxDQUFDLFVBQVUsQ0FBQyxFQUFFLFFBQUssQ0FBQztBQUN2QyxnQkFBTSxHQUFHLGdCQUFnQjs7QUFDN0IsY0FBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDOzs7Ozs7O0dBQ2pELENBQUMsQ0FBQzs7QUFFSCxJQUFFLENBQUMsZ0NBQWdDLEVBQUU7UUFDL0IsT0FBTzs7Ozs7aUJBQVMsS0FBSyxDQUFDLFlBQVksRUFBRTs7O0FBQXBDLGlCQUFPOztBQUVYLGdCQUFNLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0FBQ3RCLFdBQUMsT0FBTyxPQUFPLENBQUEsQ0FBRSxNQUFNLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxDQUFDO0FBQ3hDLFdBQUMsVUFBVSxDQUFDLE9BQU8sQ0FBQyxHQUFDLEdBQUcsQ0FBQSxDQUFFLE1BQU0sQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQzs7Ozs7OztHQUNqRCxDQUFDLENBQUM7Q0FFSixDQUFDLENBQUMiLCJmaWxlIjoidGVzdC94Y29kZS1zcGVjcy5qcyIsInNvdXJjZXNDb250ZW50IjpbInJlcXVpcmUoJ3NvdXJjZS1tYXAtc3VwcG9ydCcpLmluc3RhbGwoKTtcblxuaW1wb3J0ICogYXMgeGNvZGUgZnJvbSAnLi4vbGliL3hjb2RlJztcbmltcG9ydCBjaGFpIGZyb20gJ2NoYWknO1xuaW1wb3J0IGNoYWlBc1Byb21pc2VkIGZyb20gJ2NoYWktYXMtcHJvbWlzZWQnO1xuaW1wb3J0ICdtb2NoYXdhaXQnO1xuaW1wb3J0IGZzIGZyb20gJ2ZzJztcbmltcG9ydCBkZW5vZGVpZnkgZnJvbSAnZGVub2RlaWZ5JztcbmltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5cbmxldCBzaG91bGQgPSBjaGFpLnNob3VsZCgpO1xuY2hhaS51c2UoY2hhaUFzUHJvbWlzZWQpO1xuXG5sZXQgZmlsZUV4aXN0cyA9IGRlbm9kZWlmeShmcy5zdGF0KTtcblxuZGVzY3JpYmUoJ3hjb2RlIEBza2lwLWxpbnV4JywgKCkgPT4ge1xuXG4gIGl0KCdzaG91bGQgZ2V0IHRoZSBwYXRoIHRvIHhjb2RlIGV4ZWN1dGFibGUnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG5cbiAgICBsZXQgcGF0aCA9IGF3YWl0IHhjb2RlLmdldFBhdGgoKTtcbiAgICBzaG91bGQuZXhpc3QocGF0aCk7XG4gICAgYXdhaXQgZmlsZUV4aXN0cyhwYXRoKTtcblxuICB9KTtcblxuICBpdCgnc2hvdWxkIGdldCB0aGUgdmVyc2lvbiBvZiB4Y29kZScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcblxuICAgIGxldCB2ZXJzaW9uID0gYXdhaXQgeGNvZGUuZ2V0VmVyc2lvbigpO1xuICAgIHNob3VsZC5leGlzdCh2ZXJzaW9uKTtcbiAgICBfLmlzU3RyaW5nKHZlcnNpb24pLnNob3VsZC5iZS50cnVlO1xuICAgIC9cXGRcXC5cXGRcXC4qXFxkKi8udGVzdCh2ZXJzaW9uKS5zaG91bGQuYmUudHJ1ZTtcbiAgfSk7XG5cbiAgaXQoJ3Nob3VsZCBnZXQgdGhlIHBhdGggYW5kIHZlcnNpb24gYWdhaW4sIHRoZXNlIHZhbHVlcyBhcmUgY2FjaGVkJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuXG4gICAgYXdhaXQgeGNvZGUuZ2V0UGF0aCgpO1xuICAgIGF3YWl0IHhjb2RlLmdldFZlcnNpb24oKTtcblxuICAgIGxldCBiZWZvcmUgPSBuZXcgRGF0ZSgpO1xuICAgIGxldCBwYXRoID0gYXdhaXQgeGNvZGUuZ2V0UGF0aCgpO1xuICAgIGxldCBhZnRlciA9IG5ldyBEYXRlKCk7XG5cbiAgICBzaG91bGQuZXhpc3QocGF0aCk7XG4gICAgYXdhaXQgZmlsZUV4aXN0cyhwYXRoKTtcbiAgICAoYWZ0ZXItYmVmb3JlKS5zaG91bGQuYmUuYXQubW9zdCgyKTtcblxuICAgIGJlZm9yZSA9IG5ldyBEYXRlKCk7XG4gICAgbGV0IHZlcnNpb24gPSBhd2FpdCB4Y29kZS5nZXRWZXJzaW9uKCk7XG4gICAgYWZ0ZXIgPSBuZXcgRGF0ZSgpO1xuXG4gICAgc2hvdWxkLmV4aXN0KHZlcnNpb24pO1xuICAgIF8uaXNTdHJpbmcodmVyc2lvbikuc2hvdWxkLmJlLnRydWU7XG4gICAgL1xcZFxcLlxcZFxcLipcXGQqLy50ZXN0KHZlcnNpb24pLnNob3VsZC5iZS50cnVlO1xuICAgIChhZnRlci1iZWZvcmUpLnNob3VsZC5iZS5hdC5tb3N0KDIpO1xuXG4gIH0pO1xuXG4gIGl0KCdzaG91bGQgY2xlYXIgdGhlIGNhY2hlIGlmIGFza2VkIHRvJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuXG4gICAgeGNvZGUuY2xlYXJJbnRlcm5hbENhY2hlKCk7XG5cbiAgICBsZXQgYmVmb3JlID0gbmV3IERhdGUoKTtcbiAgICBhd2FpdCB4Y29kZS5nZXRQYXRoKCk7XG4gICAgbGV0IGFmdGVyID0gbmV3IERhdGUoKTtcbiAgICAoYWZ0ZXItYmVmb3JlKS5zaG91bGQuYmUuYXQubGVhc3QoNyk7XG5cbiAgfSk7XG5cbiAgaXQoJ3Nob3VsZCBmaW5kIHRoZSBhdXRvbWF0aW9uIHRyYWNlIHRlbXBsYXRlJywgYXN5bmMgKCkgPT4ge1xuICAgIGxldCBwYXRoID0gYXdhaXQgeGNvZGUuZ2V0QXV0b21hdGlvblRyYWNlVGVtcGxhdGVQYXRoKCk7XG5cbiAgICBzaG91bGQuZXhpc3QocGF0aCk7XG4gICAgZmlsZUV4aXN0cyhwYXRoKS5zaG91bGQuZXZlbnR1YWxseS5iZS50cnVlO1xuICAgIGxldCBzdWZmaXggPSBcIi50cmFjZXRlbXBsYXRlXCI7XG4gICAgcGF0aC5zbGljZSgtc3VmZml4Lmxlbmd0aCkuc2hvdWxkLmVxdWFsKHN1ZmZpeCk7XG4gIH0pO1xuXG4gIGl0KCdzaG91bGQgZ2V0IG1heCBpT1MgU0RLIHZlcnNpb24nLCBhc3luYygpID0+IHtcbiAgICBsZXQgdmVyc2lvbiA9IGF3YWl0IHhjb2RlLmdldE1heElPU1NESygpO1xuXG4gICAgc2hvdWxkLmV4aXN0KHZlcnNpb24pO1xuICAgICh0eXBlb2YgdmVyc2lvbikuc2hvdWxkLmVxdWFsKCdzdHJpbmcnKTtcbiAgICAocGFyc2VGbG9hdCh2ZXJzaW9uKS02LjEpLnNob3VsZC5iZS5hdC5sZWFzdCgwKTtcbiAgfSk7XG5cbn0pO1xuIl19 | ||
//# sourceMappingURL=data:application/json;base64,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LE9BQU8sQ0FBQyxDQUFDO0FBQ3RCLFdBQUMsT0FBTyxPQUFPLENBQUEsQ0FBRSxNQUFNLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxDQUFDOzs7Ozs7O0dBQ3pDLENBQUMsQ0FBQztDQUVKLENBQUMsQ0FBQyIsImZpbGUiOiJ0ZXN0L3hjb2RlLXNwZWNzLmpzIiwic291cmNlc0NvbnRlbnQiOlsicmVxdWlyZSgnc291cmNlLW1hcC1zdXBwb3J0JykuaW5zdGFsbCgpO1xuXG5pbXBvcnQgKiBhcyB4Y29kZSBmcm9tICcuLi9saWIveGNvZGUnO1xuaW1wb3J0IGNoYWkgZnJvbSAnY2hhaSc7XG5pbXBvcnQgY2hhaUFzUHJvbWlzZWQgZnJvbSAnY2hhaS1hcy1wcm9taXNlZCc7XG5pbXBvcnQgJ21vY2hhd2FpdCc7XG5pbXBvcnQgZnMgZnJvbSAnZnMnO1xuaW1wb3J0IGRlbm9kZWlmeSBmcm9tICdkZW5vZGVpZnknO1xuaW1wb3J0IF8gZnJvbSAnbG9kYXNoJztcblxubGV0IHNob3VsZCA9IGNoYWkuc2hvdWxkKCk7XG5jaGFpLnVzZShjaGFpQXNQcm9taXNlZCk7XG5cbmxldCBmaWxlRXhpc3RzID0gZGVub2RlaWZ5KGZzLnN0YXQpO1xuXG5kZXNjcmliZSgneGNvZGUgQHNraXAtbGludXgnLCAoKSA9PiB7XG5cbiAgaXQoJ3Nob3VsZCBnZXQgdGhlIHBhdGggdG8geGNvZGUgZXhlY3V0YWJsZScsIGFzeW5jIGZ1bmN0aW9uICgpIHtcblxuICAgIGxldCBwYXRoID0gYXdhaXQgeGNvZGUuZ2V0UGF0aCgpO1xuICAgIHNob3VsZC5leGlzdChwYXRoKTtcbiAgICBhd2FpdCBmaWxlRXhpc3RzKHBhdGgpO1xuXG4gIH0pO1xuXG4gIGl0KCdzaG91bGQgZ2V0IHRoZSB2ZXJzaW9uIG9mIHhjb2RlJywgYXN5bmMgZnVuY3Rpb24gKCkge1xuXG4gICAgbGV0IHZlcnNpb24gPSBhd2FpdCB4Y29kZS5nZXRWZXJzaW9uKCk7XG4gICAgc2hvdWxkLmV4aXN0KHZlcnNpb24pO1xuICAgIF8uaXNTdHJpbmcodmVyc2lvbikuc2hvdWxkLmJlLnRydWU7XG4gICAgL1xcZFxcLlxcZFxcLipcXGQqLy50ZXN0KHZlcnNpb24pLnNob3VsZC5iZS50cnVlO1xuICB9KTtcblxuICBpdCgnc2hvdWxkIGdldCB0aGUgcGF0aCBhbmQgdmVyc2lvbiBhZ2FpbiwgdGhlc2UgdmFsdWVzIGFyZSBjYWNoZWQnLCBhc3luYyBmdW5jdGlvbiAoKSB7XG5cbiAgICBhd2FpdCB4Y29kZS5nZXRQYXRoKCk7XG4gICAgYXdhaXQgeGNvZGUuZ2V0VmVyc2lvbigpO1xuXG4gICAgbGV0IGJlZm9yZSA9IG5ldyBEYXRlKCk7XG4gICAgbGV0IHBhdGggPSBhd2FpdCB4Y29kZS5nZXRQYXRoKCk7XG4gICAgbGV0IGFmdGVyID0gbmV3IERhdGUoKTtcblxuICAgIHNob3VsZC5leGlzdChwYXRoKTtcbiAgICBhd2FpdCBmaWxlRXhpc3RzKHBhdGgpO1xuICAgIChhZnRlci1iZWZvcmUpLnNob3VsZC5iZS5hdC5tb3N0KDIpO1xuXG4gICAgYmVmb3JlID0gbmV3IERhdGUoKTtcbiAgICBsZXQgdmVyc2lvbiA9IGF3YWl0IHhjb2RlLmdldFZlcnNpb24oKTtcbiAgICBhZnRlciA9IG5ldyBEYXRlKCk7XG5cbiAgICBzaG91bGQuZXhpc3QodmVyc2lvbik7XG4gICAgXy5pc1N0cmluZyh2ZXJzaW9uKS5zaG91bGQuYmUudHJ1ZTtcbiAgICAvXFxkXFwuXFxkXFwuKlxcZCovLnRlc3QodmVyc2lvbikuc2hvdWxkLmJlLnRydWU7XG4gICAgKGFmdGVyLWJlZm9yZSkuc2hvdWxkLmJlLmF0Lm1vc3QoMik7XG5cbiAgfSk7XG5cbiAgaXQoJ3Nob3VsZCBjbGVhciB0aGUgY2FjaGUgaWYgYXNrZWQgdG8nLCBhc3luYyBmdW5jdGlvbiAoKSB7XG5cbiAgICB4Y29kZS5jbGVhckludGVybmFsQ2FjaGUoKTtcblxuICAgIGxldCBiZWZvcmUgPSBuZXcgRGF0ZSgpO1xuICAgIGF3YWl0IHhjb2RlLmdldFBhdGgoKTtcbiAgICBsZXQgYWZ0ZXIgPSBuZXcgRGF0ZSgpO1xuICAgIChhZnRlci1iZWZvcmUpLnNob3VsZC5iZS5hdC5sZWFzdCg3KTtcblxuICB9KTtcblxuICBpdCgnc2hvdWxkIGZpbmQgdGhlIGF1dG9tYXRpb24gdHJhY2UgdGVtcGxhdGUnLCBhc3luYyAoKSA9PiB7XG4gICAgbGV0IHBhdGggPSBhd2FpdCB4Y29kZS5nZXRBdXRvbWF0aW9uVHJhY2VUZW1wbGF0ZVBhdGgoKTtcblxuICAgIHNob3VsZC5leGlzdChwYXRoKTtcbiAgICBmaWxlRXhpc3RzKHBhdGgpLnNob3VsZC5ldmVudHVhbGx5LmJlLnRydWU7XG4gICAgbGV0IHN1ZmZpeCA9IFwiLnRyYWNldGVtcGxhdGVcIjtcbiAgICBwYXRoLnNsaWNlKC1zdWZmaXgubGVuZ3RoKS5zaG91bGQuZXF1YWwoc3VmZml4KTtcbiAgfSk7XG5cbiAgaXQoJ3Nob3VsZCBnZXQgbWF4IGlPUyBTREsgdmVyc2lvbicsIGFzeW5jKCkgPT4ge1xuICAgIGxldCB2ZXJzaW9uID0gYXdhaXQgeGNvZGUuZ2V0TWF4SU9TU0RLKCk7XG5cbiAgICBzaG91bGQuZXhpc3QodmVyc2lvbik7XG4gICAgKHR5cGVvZiB2ZXJzaW9uKS5zaG91bGQuZXF1YWwoJ3N0cmluZycpO1xuICAgIChwYXJzZUZsb2F0KHZlcnNpb24pLTYuMSkuc2hvdWxkLmJlLmF0LmxlYXN0KDApO1xuICB9KTtcblxuICBpdCgnc2hvdWxkIGdldCBhIGxpc3Qgb2YgaU9TIGRldmljZXMnLCBhc3luYygpID0+IHtcbiAgICBsZXQgZGV2aWNlcyA9IGF3YWl0IHhjb2RlLmdldENvbm5lY3RlZERldmljZXMoKTtcbiAgICBzaG91bGQuZXhpc3QoZGV2aWNlcyk7XG4gICAgKHR5cGVvZiBkZXZpY2VzKS5zaG91bGQuZXF1YWwoJ29iamVjdCcpO1xuICB9KTtcblxufSk7XG4iXX0= |
@@ -8,2 +8,3 @@ import npmlog from 'npmlog'; | ||
import _ from 'lodash'; | ||
import plist from 'plist'; | ||
@@ -173,4 +174,2 @@ const exec = support.core.exec; | ||
// TODO remove this function. should really just be using 'getVersion()' everywhere. | ||
// This was added for tech-debt backwards compatibility | ||
async function getMaxIOSSDKWithoutRetry () { | ||
@@ -203,2 +202,32 @@ | ||
async function getConnectedDevices () { | ||
let [stdout] = await exec('/usr/sbin/system_profiler -xml SPUSBDataType', | ||
{maxBuffer: 524288, timeout: XCODE_SELECT_TIMEOUT}); | ||
let plistContent = plist.parse(stdout); | ||
let devicesFound = []; | ||
let entriesToSearch = [plistContent[0]]; | ||
while (entriesToSearch.length > 0) { | ||
let currentEntry = entriesToSearch.pop(); | ||
if (currentEntry instanceof Array) { | ||
entriesToSearch = entriesToSearch.concat(currentEntry); | ||
} else if ((currentEntry._name && | ||
currentEntry._name.substring(0, 4) === "iPad") || | ||
(currentEntry._name && | ||
currentEntry._name.substring(0, 6) === "iPhone")) { | ||
let deviceInfo = { | ||
name: currentEntry._name, | ||
udid: currentEntry.serial_num, | ||
productId: currentEntry.product_id, | ||
deviceVersion: currentEntry.bcd_device | ||
}; | ||
devicesFound.push(deviceInfo); | ||
} else if (currentEntry._items) { | ||
entriesToSearch = entriesToSearch.concat(currentEntry._items); | ||
} | ||
} | ||
return devicesFound; | ||
} | ||
function clearInternalCache () { | ||
@@ -221,2 +250,2 @@ | ||
getAutomationTraceTemplatePathWithoutRetry, getMaxIOSSDKWithoutRetry, | ||
clearInternalCache }; | ||
getConnectedDevices, clearInternalCache }; |
@@ -7,3 +7,3 @@ { | ||
], | ||
"version": "2.0.2", | ||
"version": "2.0.3", | ||
"author": "appium", | ||
@@ -33,2 +33,3 @@ "license": "Apache License 2.0", | ||
"npmlog": "^1.2.0", | ||
"plist": "^1.1.0", | ||
"q": "^1.2.0", | ||
@@ -35,0 +36,0 @@ "source-map-support": "^0.2.10" |
appium-xcode | ||
=================== | ||
Work in progress, stay tuned! | ||
ES7 module for interacting with Xcode and Xcode-related functions. | ||
Used by [Appium](github.com/appium/appium) | ||
API | ||
=== | ||
All functions are `async`, meaning they return promises which can be awaited via `await`. | ||
Most functions are memoized, so after they are called once, they will simply return the same value. Remember that calling `require()` multiple times returns the same instantiation of a module if it has already been instantiated, so the memoization will be preserved across multiple files in the same project. | ||
Some functions have an auto-retry built into them, they will retry silently a number of times. This is because the Xcode commands sometimes just flake and return bad values (or don't return). | ||
To clear the memoized values, call `clearInternalCache` | ||
### getPath() | ||
*memoized* | ||
gets path to Xcode | ||
### getVersion([num_retries]) | ||
*memoized*, *retry* | ||
returns the version of Xcode. Returns strings like `'6.3.1'` | ||
### getAutomationTraceTemplatePath([num_retries]) | ||
*memoized, *retry* | ||
returns a path to the default AutomationTraceTemplate | ||
### getAutomationTraceTemplatePathWithoutRetry() | ||
same as `getAutomationTraceTemplatePath()` but without retry or memoization. | ||
### getMaxIOSSDK([num_retries]) | ||
*memoized*, *retry* | ||
returns largest IOS SDK version supported by Xcode. | ||
eg: `'8.3'` | ||
### getMaxIOSSDKWithoutRetry() | ||
same as `getMaxIOSDK()` but without retry or memoization | ||
### clearInternalCache() | ||
clears the internal cache used for memoizing functions. | ||
Develop | ||
======= | ||
## Watch | ||
@@ -7,0 +54,0 @@ |
@@ -86,2 +86,8 @@ // transpile:mocha | ||
it('should get a list of iOS devices', async() => { | ||
let devices = await xcode.getConnectedDevices(); | ||
should.exist(devices); | ||
(typeof devices).should.equal('object'); | ||
}); | ||
}); |
Sorry, the diff of this file is not supported yet
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
135458
783
64
9
+ Addedplist@^1.1.0
+ Addedbase64-js@0.0.8(transitive)
+ Addedplist@1.2.0(transitive)
+ Addedxmlbuilder@4.0.0(transitive)
+ Addedxmldom@0.1.31(transitive)