/
ListDevicesCommand.swift
116 lines (93 loc) · 3.79 KB
/
ListDevicesCommand.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
import ArgumentParser
import Foundation
// import ShellOut
// import TSCBasic
import Blocks
struct ListDevicesCommand: ParsableCommand {
static var configuration = CommandConfiguration(
commandName: "list-simulators",
abstract: "List (or filter) the available simulators Xcode provide."
)
@Option var osFilter: String?
@Option var deviceFilter: String?
mutating func run() throws {
let rawAvailableDevicesJSON = CLIUtils.shell("xcrun simctl list --json devices available")
let filteredZippedResults = try find(osName: osFilter, deviceName: deviceFilter, in: rawAvailableDevicesJSON)
switch (osFilter, deviceFilter) {
case (.some, .some):
// We want to output the UDID
precondition(filteredZippedResults.count == 1)
print(filteredZippedResults.first!.simulator.udid)
default:
let output = filteredZippedResults
.sorted()
.map(\.description)
print(output.joined(separator: "\n"))
}
}
func find(osName: String?, deviceName: String?, in rawOutput: String) throws -> [ZipOSSimulator] {
try zip(rawOutput: rawOutput)
.filter { osName == nil ? true : $0.formattedOS == osName }
.filter { deviceName == nil ? true : $0.simulator.name == deviceName }
}
private func zip(rawOutput: String) throws -> [ZipOSSimulator] {
try JSONDecoder()
.decode(DeviceContainer.self, from: Data(rawOutput.utf8))
.devices
.reduce(into: [ZipOSSimulator]()) { partialResult, newItem in
partialResult.append(contentsOf: newItem.value.map {
ZipOSSimulator(osIdentifier: newItem.key,
simulator: $0)
})
}
}
}
struct DeviceContainer: Codable {
let devices: [String: [Simulator]]
}
struct Simulator: Codable, Equatable {
let lastBootedAt: String?
let dataPath: String
let dataPathSize: Int
let logPath: String
let udid: String
let isAvailable: Bool
let logPathSize: Int?
let deviceTypeIdentifier: String
let state: String
let name: String
}
struct ZipOSSimulator: Comparable, CustomStringConvertible {
let osIdentifier: String
let simulator: Simulator
// MARK: - CustomStringConvertible
var description: String {
"\(simulator.udid): \(formattedOSAndSimulator)"
}
var formattedOS: String {
(try? formatOSIdentifier(osIdentifier)) ?? "n/a"
}
var formattedOSAndSimulator: String {
"\(formattedOS), \(simulator.name)"
}
// MARK: - Comparable
static func < (lhs: ZipOSSimulator, rhs: ZipOSSimulator) -> Bool {
lhs.formattedOSAndSimulator < rhs.formattedOSAndSimulator
}
private func formatOSIdentifier(_ identifier: String) throws -> String {
let pattern = #"com\.apple\.CoreSimulator\.SimRuntime\.([a-zA-Z]+)-(\d+)-(\d+)"#
let regex = try? NSRegularExpression(pattern: pattern, options: [])
let nsrange = NSRange(identifier.startIndex ..< identifier.endIndex, in: identifier)
if let match = regex?.firstMatch(in: identifier, options: [], range: nsrange) {
if let osRange = Range(match.range(at: 1), in: identifier),
let majorVersionRange = Range(match.range(at: 2), in: identifier),
let minorVersionRange = Range(match.range(at: 3), in: identifier) {
let os = identifier[osRange]
let majorVersion = identifier[majorVersionRange]
let minorVersion = identifier[minorVersionRange]
return "\(os) \(majorVersion).\(minorVersion)"
}
}
throw SimpleMessageError(message: "formatDeviceIdentifier could not format \(identifier)")
}
}