Files
letro-ios/Tools/Sources/Commands/CI/ReleaseToGithub.swift
Stefan Ceriu c3ba6113fe Replace the last of the fastlane lanes with swift tooling
- move more sharable code to CI static methods
- merge `release_to_github` and `prepare_next_release` into one single command as they had dependencies on each other
- remove all traces of ruby and fastlane
2026-03-11 10:06:44 +02:00

160 lines
6.5 KiB
Swift

import ArgumentParser
import Foundation
import Yams
struct ReleaseToGitHub: AsyncParsableCommand {
static let configuration = CommandConfiguration(commandName: "release-to-github",
abstract: "Creates a GitHub release and updates CHANGES.md with generated release notes.")
enum ReleaseError: LocalizedError {
case missingGitHubToken
case failedToCreateRelease(String)
case failedToParseResponse
case missingReleaseNotes
case failedToReadVersion
var errorDescription: String? {
switch self {
case .missingGitHubToken:
return "The GITHUB_TOKEN environment variable is not set."
case .failedToCreateRelease(let message):
return "Failed to create GitHub release: \(message)"
case .failedToParseResponse:
return "Failed to parse the GitHub API response."
case .missingReleaseNotes:
return "The generated release notes are empty."
case .failedToReadVersion:
return "Failed to read the marketing version from project.yml."
}
}
}
func run() async throws {
let currentVersion = try CI.readMarketingVersion()
logger.info("Creating GitHub release for version \(currentVersion)")
let releaseBody = try await createGitHubRelease(version: currentVersion)
try updateChangelog(version: currentVersion, generatedNotes: releaseBody)
let changesFilePath = URL.projectDirectory.appendingPathComponent("CHANGES.md").path
try await CI.run(.name("git"), ["add", changesFilePath])
logger.info("Successfully created GitHub release \(currentVersion) and updated CHANGES.md.")
let targetFilePath = "project.yml"
let xcodeProjPath = "ElementX.xcodeproj"
guard let newVersion = bumpPatchVersion(currentVersion) else {
throw ValidationError("Invalid version format: \(currentVersion)")
}
// Bump the patch version using sed (preserves file formatting)
try await CI.run(.name("sed"), ["-i", "''", "'s/MARKETING_VERSION: \(currentVersion)/MARKETING_VERSION: \(newVersion)/g'", targetFilePath])
logger.info("Version updated from \(currentVersion) to \(newVersion)")
try await CI.run(.name("xcodegen"))
try await CI.gitConfigureGlobals()
try await CI.run(.name("git"), ["add", targetFilePath, xcodeProjPath])
try await CI.run(.name("git"), ["commit", "-m", "Prepare next release"])
try await CI.gitPush()
try await rebaseMainOntoCurrentBranch()
}
// MARK: - Private
private func createGitHubRelease(version: String) async throws -> String {
guard let apiToken = ProcessInfo.processInfo.environment["GITHUB_TOKEN"], !apiToken.isEmpty
else {
throw ReleaseError.missingGitHubToken
}
let url = URL(string: "https://api.github.com/repos/element-hq/element-x-ios/releases")!
var request = URLRequest(url: url)
request.httpMethod = "POST"
request.setValue("Bearer \(apiToken)", forHTTPHeaderField: "Authorization")
request.setValue("application/vnd.github+json", forHTTPHeaderField: "Accept")
request.setValue("application/json", forHTTPHeaderField: "Content-Type")
let body: [String: Any] = ["tag_name": "release/\(version)",
"name": version,
"generate_release_notes": true]
request.httpBody = try JSONSerialization.data(withJSONObject: body)
let (data, response) = try await URLSession.shared.data(for: request)
guard let httpResponse = response as? HTTPURLResponse else {
throw ReleaseError.failedToParseResponse
}
guard (200...299).contains(httpResponse.statusCode) else {
let errorBody = String(data: data, encoding: .utf8) ?? "Unknown error"
throw ReleaseError.failedToCreateRelease("HTTP \(httpResponse.statusCode): \(errorBody)")
}
guard let json = try JSONSerialization.jsonObject(with: data) as? [String: Any],
let releaseBody = json["body"] as? String else {
throw ReleaseError.failedToParseResponse
}
return releaseBody
}
private func updateChangelog(version: String, generatedNotes: String) throws {
let changesURL = URL.projectDirectory.appending(component: "CHANGES.md")
// Clean up the generated notes: remove HTML comments and adjust header levels
let cleanedNotes = generatedNotes
.replacingOccurrences(of: "<!-- .*? -->", with: "", options: .regularExpression)
.replacingOccurrences(of: "### ", with: "\n")
.replacingOccurrences(of: "## ", with: "### ")
guard !cleanedNotes.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty else {
throw ReleaseError.missingReleaseNotes
}
let releaseDate = Date().formatted(.iso8601.year().month().day())
let existingContent = try String(contentsOf: changesURL)
let newContent = "## Changes in \(version) (\(releaseDate))\(cleanedNotes)\n\n\(existingContent)"
try newContent.write(to: changesURL, atomically: true, encoding: .utf8)
logger.info("Updated CHANGES.md with release notes.")
}
private func bumpPatchVersion(_ version: String) -> String? {
let regex = /^(\d{2})\.(\d{2})\.(\d+)$/
guard let match = version.firstMatch(of: regex), var patch = Int(match.3) else {
return nil
}
let year = String(match.1)
let month = String(match.2)
patch = patch + 1
return "\(year).\(month).\(patch)"
}
private func rebaseMainOntoCurrentBranch() async throws {
guard let currentBranch = try await CI.run(.name("git"), ["rev-parse", "--abbrev-ref", "HEAD"], output: .string(limit: 4096))
.standardOutput.map({ $0.trimmingCharacters(in: .whitespacesAndNewlines) }) else {
throw ValidationError("Could not determine the current branch.")
}
logger.info("Current branch: \(currentBranch)")
try await CI.run(.name("git"), ["reset", "--hard"])
try await CI.run(.name("git"), ["checkout", "main"])
try await CI.run(.name("git"), ["pull", "origin", "main"])
try await CI.run(.name("git"), ["rebase", "currentBranch"])
try await CI.gitPush()
logger.info("Successfully rebased main onto \(currentBranch)")
}
}