Files
letro-ios/Tools/Sources/BuildSDK.swift
2022-12-15 09:35:00 +00:00

130 lines
5.1 KiB
Swift

import ArgumentParser
import Foundation
import Yams
struct BuildSDK: ParsableCommand {
static var configuration = CommandConfiguration(abstract: "A tool to checkout and build MatrixRustSDK locally for development.")
@Argument(help: "An optional argument to specify a branch of the SDK.")
var branch: String?
@Option(help: "The target to build for such as aarch64-apple-ios. Omit this option to build for all targets.")
var target: String?
private var projectDirectoryURL: URL { URL(filePath: FileManager.default.currentDirectoryPath) }
private var parentDirectoryURL: URL { projectDirectoryURL.deletingLastPathComponent() }
private var sdkDirectoryURL: URL { parentDirectoryURL.appending(path: "matrix-rust-sdk") }
enum Error: LocalizedError {
case scriptFailed
case rustupOutputFailure
case missingRustTargets([String])
case failureParsingProjectYAML
var errorDescription: String? {
switch self {
case .missingRustTargets(let missingTargets):
return """
Rust is missing the necessary targets to build the SDK.
Run the following command to install them:
rustup target add \(missingTargets.joined(separator: " ")) --toolchain nightly
"""
default:
return nil
}
}
}
func run() throws {
try checkRustupTargets()
try cloneSDKIfNeeded()
try checkoutBranchIfSupplied()
try buildFramework()
try updateXcodeProject()
}
/// Checks that all of the required targets have been added through rustup
/// but only when the ``target`` option hasn't been supplied.
func checkRustupTargets() throws {
guard target == nil else { return }
guard let output = try zsh("rustup show", workingDirectoryURL: projectDirectoryURL) else { throw Error.rustupOutputFailure }
var requiredTargets = [
"aarch64-apple-darwin": false,
"aarch64-apple-ios": false,
"aarch64-apple-ios-sim": false,
"x86_64-apple-darwin": false,
"x86_64-apple-ios": false
]
output.enumerateLines { line, _ in
if requiredTargets.keys.contains(line) {
requiredTargets[line] = true
}
}
let missingTargets = requiredTargets.compactMap { !$0.value ? $0.key : nil }
guard missingTargets.isEmpty else { throw Error.missingRustTargets(missingTargets) }
}
/// Clones the Rust SDK if a copy isn't found in the parent directory.
func cloneSDKIfNeeded() throws {
guard !FileManager.default.fileExists(atPath: sdkDirectoryURL.path) else { return }
try zsh("git clone https://github.com/matrix-org/matrix-rust-sdk", workingDirectoryURL: parentDirectoryURL)
}
/// Checkout the specified branch of the SDK if supplied.
func checkoutBranchIfSupplied() throws {
guard let branch else { return }
try zsh("git checkout \(branch)", workingDirectoryURL: sdkDirectoryURL)
}
/// Build the Rust SDK as an XCFramework with the debug profile.
func buildFramework() throws {
var buildCommand = "cargo xtask swift build-framework --profile dbg"
if let target {
buildCommand.append(" --only-target \(target)")
}
try zsh(buildCommand, workingDirectoryURL: sdkDirectoryURL)
}
/// Update the Xcode project to use the build of the SDK.
func updateXcodeProject() throws {
try updateProjectYAML()
try zsh("xcodegen", workingDirectoryURL: projectDirectoryURL)
}
/// Update project.yml with the local path of the SDK.
func updateProjectYAML() throws {
let yamlURL = projectDirectoryURL.appending(path: "project.yml")
let yamlString = try String(contentsOf: yamlURL)
guard var projectConfig = try Yams.compose(yaml: yamlString) else { throw Error.failureParsingProjectYAML }
projectConfig["packages"]?.mapping?["MatrixRustSDK"]? = ["path": "../matrix-rust-sdk"]
let updatedYAMLString = try Yams.serialize(node: projectConfig)
try updatedYAMLString.write(to: yamlURL, atomically: true, encoding: .utf8)
}
/// Runs a command in zsh.
@discardableResult
func zsh(_ command: String, workingDirectoryURL: URL) throws -> String? {
let process = Process()
process.executableURL = URL(filePath: "/bin/zsh")
process.arguments = ["-c", command]
process.currentDirectoryURL = workingDirectoryURL
let outputPipe = Pipe()
process.standardOutput = outputPipe
try process.run()
process.waitUntilExit()
guard process.terminationReason == .exit, process.terminationStatus == 0 else { throw Error.scriptFailed }
guard let outputData = try outputPipe.fileHandleForReading.readToEnd() else { return nil }
return String(data: outputData, encoding: .utf8)
}
}