forked from mapbox/mapbox-events-ios
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathPackage.swift
157 lines (133 loc) · 5.65 KB
/
Package.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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
// swift-tools-version:5.3
// The swift-tools-version declares the minimum version of Swift required to build this package.
import PackageDescription
import Foundation
let registry = SDKRegistry()
let version = "1.0.10"
let checksum = "a23a14c445a4d4b81184ad22e778810034b6e82999a6eba21617e76a5cf0f991"
let package = Package(
name: "MapboxMobileEvents",
platforms: [.iOS(.v9)],
products: [
.library(
name: "MapboxMobileEvents",
targets: ["MapboxMobileEvents"]
)
],
targets: [
registry.mapboxEventsTarget(version: version, checksum: checksum),
.testTarget(
name: "MapboxMobileEventsTests",
dependencies: ["MapboxMobileEvents"],
cSettings: [
.headerSearchPath("Fakes"),
.headerSearchPath("Utilities"),
.headerSearchPath("../../Sources"),
.headerSearchPath("../../Sources/MapboxMobileEvents")
]
),
.testTarget(
name: "MapboxMobileEventsSwiftTests",
dependencies: ["MapboxMobileEvents"]
),
],
cxxLanguageStandard: .cxx14
)
struct SDKRegistry {
let host = "api.mapbox.com"
func binaryTarget(name: String, version: String, path: String, filename: String, checksum: String) -> Target {
var url = "https://\(host)/downloads/v2/\(path)/releases/ios/packages/\(version)/\(filename)"
if let token = netrcToken {
url += "?access_token=\(token)"
} else {
debugPrint("Mapbox token wasn't founded in ~/.netrc. Fix this issue to integrate Mapbox SDK. Otherwise, you will see 'invalid status code 401' or 'no XCFramework found. To clean issue in Xcode, remove ~/Library/Developer/Xcode/DerivedData folder")
}
return .binaryTarget(name: name, url: url, checksum: checksum)
}
var netrcToken: String? {
var mapboxToken: String?
do {
let netrc = try Netrc.load().get()
mapboxToken = netrc.machines.first(where: { $0.name == host })?.password
} catch {
// Do nothing on client machines
}
return mapboxToken
}
}
extension SDKRegistry {
func mapboxEventsTarget(version: String, checksum: String) -> Target {
return binaryTarget(name: "MapboxMobileEvents",
version: version,
path: "mapbox-events-ios",
filename: "MapboxMobileEvents.zip",
checksum: checksum)
}
}
// Reference: https://github.com/apple/swift-tools-support-core/pull/88
// Sub-reference: https://github.com/Carthage/Carthage/pull/2774
struct NetrcMachine {
let name: String
let login: String
let password: String
}
struct Netrc {
enum NetrcError: Error {
case fileNotFound(URL)
case unreadableFile(URL)
case machineNotFound
case missingToken(String)
case missingValueForToken(String)
}
public let machines: [NetrcMachine]
init(machines: [NetrcMachine]) {
self.machines = machines
}
static func load(from fileURL: URL = URL(fileURLWithPath: "\(NSHomeDirectory())/.netrc")) -> Result<Netrc, Error> {
guard FileManager.default.fileExists(atPath: fileURL.path) else { return .failure(NetrcError.fileNotFound(fileURL)) }
guard FileManager.default.isReadableFile(atPath: fileURL.path) else { return .failure(NetrcError.unreadableFile(fileURL)) }
return Result(catching: { try String(contentsOf: fileURL, encoding: .utf8) })
.flatMap { Netrc.from($0) }
}
static func from(_ content: String) -> Result<Netrc, Error> {
let trimmedCommentsContent = trimComments(from: content)
let tokens = trimmedCommentsContent
.trimmingCharacters(in: .whitespacesAndNewlines)
.components(separatedBy: .whitespacesAndNewlines)
.filter({ $0 != "" })
var machines: [NetrcMachine] = []
let machineTokens = tokens.split { $0 == "machine" }
guard tokens.contains("machine"), machineTokens.count > 0 else { return .failure(NetrcError.machineNotFound) }
for machine in machineTokens {
let values = Array(machine)
guard let name = values.first else { continue }
guard let login = values["login"] else { return .failure(NetrcError.missingValueForToken("login")) }
guard let password = values["password"] else { return .failure(NetrcError.missingValueForToken("password")) }
machines.append(NetrcMachine(name: name, login: login, password: password))
}
guard machines.count > 0 else { return .failure(NetrcError.machineNotFound) }
return .success(Netrc(machines: machines))
}
private static func trimComments(from text: String) -> String {
let regex = try! NSRegularExpression(pattern: "\\#[\\s\\S]*?.*$", options: .anchorsMatchLines)
let nsString = text as NSString
let range = NSRange(location: 0, length: nsString.length)
let matches = regex.matches(in: text, range: range)
var trimmedCommentsText = text
matches.forEach {
trimmedCommentsText = trimmedCommentsText
.replacingOccurrences(of: nsString.substring(with: $0.range), with: "")
}
return trimmedCommentsText
}
}
fileprivate extension Array where Element == String {
subscript(_ token: String) -> String? {
guard let tokenIndex = firstIndex(of: token),
count > tokenIndex,
!["machine", "login", "password"].contains(self[tokenIndex + 1]) else {
return nil
}
return self[tokenIndex + 1]
}
}