2026-06-25 23:48:26 -07:00
|
|
|
require "fileutils"
|
2026-06-25 22:50:30 -07:00
|
|
|
require "shellwords"
|
2026-06-25 22:48:59 -07:00
|
|
|
|
2026-06-05 23:19:14 -07:00
|
|
|
default_platform(:ios)
|
|
|
|
|
|
2026-06-25 22:41:00 -07:00
|
|
|
APP_IDENTIFIER = "net.buzzert.sybil2"
|
2026-06-05 23:19:14 -07:00
|
|
|
SCHEME = "Sybil"
|
2026-06-25 22:41:00 -07:00
|
|
|
TEAM_ID = "DQQH5H6GBD"
|
|
|
|
|
PROFILE_NAME = "Sybil AppStore CI"
|
2026-06-25 23:44:13 -07:00
|
|
|
SIGNING_IDENTITY = "Apple Distribution: James Magahern (DQQH5H6GBD)"
|
|
|
|
|
CI_KEYCHAIN_NAME = "sybil_ci_keychain"
|
2026-06-25 23:48:26 -07:00
|
|
|
CI_KEYCHAIN_PASSWORD = ""
|
2026-06-25 22:50:30 -07:00
|
|
|
IOS_ROOT = File.expand_path("..", __dir__)
|
|
|
|
|
PROJECT_FILE = File.join(IOS_ROOT, "Sybil.xcodeproj")
|
|
|
|
|
PROJECT_SPEC = File.join(IOS_ROOT, "project.yml")
|
2026-06-25 23:48:26 -07:00
|
|
|
CI_KEYCHAIN_PATH = File.join(File.expand_path("~/Library/Keychains"), CI_KEYCHAIN_NAME)
|
|
|
|
|
CI_KEYCHAIN_DB_PATH = "#{CI_KEYCHAIN_PATH}-db"
|
2026-06-25 23:51:33 -07:00
|
|
|
LOGIN_KEYCHAIN_PATH = File.expand_path("~/Library/Keychains/login.keychain")
|
|
|
|
|
LOGIN_KEYCHAIN_DB_PATH = "#{LOGIN_KEYCHAIN_PATH}-db"
|
2026-06-05 23:19:14 -07:00
|
|
|
|
|
|
|
|
def present?(value)
|
|
|
|
|
!value.to_s.strip.empty?
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def release_version
|
2026-06-25 22:41:00 -07:00
|
|
|
tag = ENV["SYBIL_VERSION_TAG"].to_s
|
|
|
|
|
tag = ENV["GITHUB_REF_NAME"].to_s if !present?(tag)
|
|
|
|
|
tag = ENV["GITHUB_REF"].to_s.sub(%r{\Arefs/tags/}, "") if !present?(tag)
|
|
|
|
|
tag = sh("git describe --tags --abbrev=0").strip if !present?(tag)
|
|
|
|
|
version = tag.sub(%r{\Arelease/}, "").sub(/\Av/, "")
|
2026-06-25 21:03:43 -07:00
|
|
|
|
2026-06-25 22:41:00 -07:00
|
|
|
unless version.match?(/\A\d+\.\d+\.\d+\z/)
|
|
|
|
|
UI.user_error!("Release tag must look like v1.2.3; got #{tag.inspect}")
|
2026-06-25 21:03:43 -07:00
|
|
|
end
|
|
|
|
|
|
2026-06-25 22:41:00 -07:00
|
|
|
version
|
2026-06-25 21:03:43 -07:00
|
|
|
end
|
|
|
|
|
|
2026-06-25 23:44:13 -07:00
|
|
|
def ci?
|
|
|
|
|
present?(ENV["CI"])
|
2026-06-25 23:18:01 -07:00
|
|
|
end
|
|
|
|
|
|
2026-06-25 23:44:13 -07:00
|
|
|
def ci_keychain_path
|
2026-06-25 23:48:26 -07:00
|
|
|
File.file?(CI_KEYCHAIN_DB_PATH) ? CI_KEYCHAIN_DB_PATH : CI_KEYCHAIN_PATH
|
2026-06-25 23:44:13 -07:00
|
|
|
end
|
2026-06-25 23:06:00 -07:00
|
|
|
|
2026-06-25 23:44:13 -07:00
|
|
|
platform :ios do
|
2026-06-25 22:41:00 -07:00
|
|
|
private_lane :app_store_api_key do
|
|
|
|
|
app_store_connect_api_key(
|
|
|
|
|
key_id: ENV.fetch("APP_STORE_CONNECT_KEY_ID"),
|
|
|
|
|
issuer_id: ENV.fetch("APP_STORE_CONNECT_ISSUER_ID"),
|
|
|
|
|
key_content: ENV.fetch("APP_STORE_CONNECT_KEY_CONTENT"),
|
|
|
|
|
is_key_content_base64: true
|
|
|
|
|
)
|
2026-06-25 21:18:54 -07:00
|
|
|
end
|
|
|
|
|
|
2026-06-25 23:44:13 -07:00
|
|
|
private_lane :setup_ci_signing do
|
|
|
|
|
next unless ci?
|
|
|
|
|
|
2026-06-25 23:48:26 -07:00
|
|
|
FileUtils.mkdir_p(File.dirname(CI_KEYCHAIN_PATH))
|
|
|
|
|
sh("security delete-keychain #{CI_KEYCHAIN_PATH.shellescape} || true", log: false)
|
|
|
|
|
FileUtils.rm_f(CI_KEYCHAIN_PATH)
|
|
|
|
|
FileUtils.rm_f(CI_KEYCHAIN_DB_PATH)
|
|
|
|
|
|
|
|
|
|
create_keychain(
|
|
|
|
|
path: CI_KEYCHAIN_PATH,
|
|
|
|
|
password: CI_KEYCHAIN_PASSWORD,
|
2026-06-25 23:51:33 -07:00
|
|
|
default_keychain: false,
|
2026-06-25 23:48:26 -07:00
|
|
|
unlock: true,
|
|
|
|
|
timeout: 3600,
|
|
|
|
|
lock_when_sleeps: true,
|
2026-06-25 23:51:33 -07:00
|
|
|
add_to_search_list: false
|
2026-06-25 23:44:13 -07:00
|
|
|
)
|
2026-06-25 23:48:26 -07:00
|
|
|
|
2026-06-25 23:51:33 -07:00
|
|
|
sh("security default-keychain -d user -s #{CI_KEYCHAIN_PATH.shellescape}", log: false)
|
|
|
|
|
sh("security list-keychains -d user -s #{ci_keychain_path.shellescape}", log: false)
|
|
|
|
|
|
2026-06-25 23:48:26 -07:00
|
|
|
ENV["MATCH_KEYCHAIN_NAME"] = CI_KEYCHAIN_PATH
|
|
|
|
|
ENV["MATCH_KEYCHAIN_PASSWORD"] = CI_KEYCHAIN_PASSWORD
|
|
|
|
|
ENV["MATCH_READONLY"] = "true"
|
2026-06-25 23:44:13 -07:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
private_lane :cleanup_ci_signing do
|
|
|
|
|
next unless ci?
|
|
|
|
|
|
2026-06-25 23:51:33 -07:00
|
|
|
if File.file?(LOGIN_KEYCHAIN_DB_PATH) || File.file?(LOGIN_KEYCHAIN_PATH)
|
|
|
|
|
sh("security default-keychain -d user -s #{LOGIN_KEYCHAIN_PATH.shellescape} || true", log: false)
|
|
|
|
|
sh("security list-keychains -d user -s #{LOGIN_KEYCHAIN_DB_PATH.shellescape} || true", log: false)
|
|
|
|
|
end
|
|
|
|
|
sh("security delete-keychain #{ci_keychain_path.shellescape} || true", log: false)
|
|
|
|
|
FileUtils.rm_f(CI_KEYCHAIN_PATH)
|
|
|
|
|
FileUtils.rm_f(CI_KEYCHAIN_DB_PATH)
|
2026-06-25 23:44:13 -07:00
|
|
|
rescue => error
|
|
|
|
|
UI.message("Unable to delete temporary CI keychain: #{error.message}")
|
|
|
|
|
ensure
|
|
|
|
|
ENV.delete("MATCH_KEYCHAIN_NAME")
|
|
|
|
|
ENV.delete("MATCH_KEYCHAIN_PASSWORD")
|
2026-06-25 23:48:26 -07:00
|
|
|
ENV.delete("MATCH_READONLY")
|
2026-06-25 23:44:13 -07:00
|
|
|
end
|
2026-06-25 22:41:00 -07:00
|
|
|
|
2026-06-25 23:44:13 -07:00
|
|
|
private_lane :sync_signing do |options|
|
2026-06-25 23:48:26 -07:00
|
|
|
match_options = {
|
2026-06-25 22:41:00 -07:00
|
|
|
type: "appstore",
|
|
|
|
|
readonly: options.fetch(:readonly),
|
|
|
|
|
app_identifier: APP_IDENTIFIER,
|
|
|
|
|
team_id: TEAM_ID,
|
|
|
|
|
profile_name: PROFILE_NAME,
|
2026-06-25 23:44:13 -07:00
|
|
|
git_url: ENV.fetch("MATCH_GIT_URL"),
|
|
|
|
|
git_branch: "master",
|
2026-06-25 22:41:00 -07:00
|
|
|
git_full_name: "Sybil Release Bot",
|
|
|
|
|
git_user_email: "james.magahern@me.com",
|
2026-06-25 23:44:13 -07:00
|
|
|
api_key: options.fetch(:api_key)
|
2026-06-25 23:48:26 -07:00
|
|
|
}
|
|
|
|
|
match_options[:keychain_name] = ENV["MATCH_KEYCHAIN_NAME"] if present?(ENV["MATCH_KEYCHAIN_NAME"])
|
|
|
|
|
match_options[:keychain_password] = ENV["MATCH_KEYCHAIN_PASSWORD"] if ENV.key?("MATCH_KEYCHAIN_PASSWORD")
|
|
|
|
|
|
|
|
|
|
match(match_options)
|
2026-06-25 23:44:13 -07:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
private_lane :verify_ci_signing do
|
|
|
|
|
next unless ci?
|
|
|
|
|
|
|
|
|
|
if File.file?(ci_keychain_path)
|
|
|
|
|
password = ENV.fetch("MATCH_KEYCHAIN_PASSWORD", "")
|
|
|
|
|
sh("security unlock-keychain -p #{password.shellescape} #{ci_keychain_path.shellescape}", log: false)
|
|
|
|
|
sh("security set-key-partition-list -S apple-tool:,apple:,codesign: -s -k #{password.shellescape} #{ci_keychain_path.shellescape}", log: false)
|
|
|
|
|
end
|
2026-06-25 20:51:01 -07:00
|
|
|
|
2026-06-25 23:44:13 -07:00
|
|
|
identities = sh("security find-identity -v -p codesigning", log: false)
|
|
|
|
|
UI.message(identities)
|
|
|
|
|
|
|
|
|
|
unless identities.include?(SIGNING_IDENTITY)
|
|
|
|
|
UI.user_error!("The CI keychain search list does not contain #{SIGNING_IDENTITY}")
|
|
|
|
|
end
|
2026-06-25 20:51:01 -07:00
|
|
|
end
|
|
|
|
|
|
2026-06-25 22:41:00 -07:00
|
|
|
desc "Create or update match signing assets"
|
|
|
|
|
lane :setup_signing do
|
2026-06-25 23:44:13 -07:00
|
|
|
sync_signing(api_key: app_store_api_key, readonly: false)
|
2026-06-05 23:19:14 -07:00
|
|
|
end
|
|
|
|
|
|
2026-06-25 22:41:00 -07:00
|
|
|
desc "Build and upload to TestFlight"
|
2026-06-05 23:19:14 -07:00
|
|
|
lane :beta do
|
2026-06-25 23:44:13 -07:00
|
|
|
setup_ci_signing
|
2026-06-05 23:19:14 -07:00
|
|
|
|
2026-06-25 22:41:00 -07:00
|
|
|
api_key = app_store_api_key
|
2026-06-05 23:19:14 -07:00
|
|
|
|
2026-06-25 22:50:30 -07:00
|
|
|
sh("xcodegen --spec #{PROJECT_SPEC.shellescape}")
|
2026-06-05 23:19:14 -07:00
|
|
|
|
2026-06-25 22:41:00 -07:00
|
|
|
increment_version_number(
|
|
|
|
|
version_number: release_version,
|
2026-06-25 22:50:30 -07:00
|
|
|
xcodeproj: PROJECT_FILE
|
2026-06-25 22:41:00 -07:00
|
|
|
)
|
2026-06-05 23:19:14 -07:00
|
|
|
|
2026-06-25 22:41:00 -07:00
|
|
|
latest_build_number = latest_testflight_build_number(
|
|
|
|
|
app_identifier: APP_IDENTIFIER,
|
|
|
|
|
api_key: api_key,
|
|
|
|
|
initial_build_number: 0
|
|
|
|
|
)
|
2026-06-05 23:19:14 -07:00
|
|
|
|
2026-06-25 22:41:00 -07:00
|
|
|
increment_build_number(
|
|
|
|
|
build_number: latest_build_number + 1,
|
2026-06-25 22:50:30 -07:00
|
|
|
xcodeproj: PROJECT_FILE
|
2026-06-25 22:41:00 -07:00
|
|
|
)
|
|
|
|
|
|
2026-06-25 23:44:13 -07:00
|
|
|
sync_signing(api_key: api_key, readonly: true)
|
|
|
|
|
verify_ci_signing
|
|
|
|
|
|
2026-06-25 23:51:33 -07:00
|
|
|
xcargs = [
|
|
|
|
|
"DEVELOPMENT_TEAM=#{TEAM_ID.shellescape}",
|
|
|
|
|
"CODE_SIGN_STYLE=Manual",
|
|
|
|
|
"CODE_SIGN_IDENTITY=#{SIGNING_IDENTITY.shellescape}",
|
|
|
|
|
"PROVISIONING_PROFILE_SPECIFIER=#{PROFILE_NAME.shellescape}"
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
if ci?
|
|
|
|
|
xcargs << "CODE_SIGN_KEYCHAIN=#{ci_keychain_path.shellescape}"
|
|
|
|
|
xcargs << "OTHER_CODE_SIGN_FLAGS=#{("--keychain #{ci_keychain_path}").shellescape}"
|
|
|
|
|
end
|
|
|
|
|
|
2026-06-25 23:44:13 -07:00
|
|
|
build_app(
|
2026-06-25 22:50:30 -07:00
|
|
|
project: PROJECT_FILE,
|
2026-06-05 23:19:14 -07:00
|
|
|
scheme: SCHEME,
|
|
|
|
|
export_method: "app-store",
|
2026-06-25 23:44:13 -07:00
|
|
|
codesigning_identity: SIGNING_IDENTITY,
|
2026-06-25 23:51:33 -07:00
|
|
|
xcargs: xcargs.join(" "),
|
2026-06-05 23:19:14 -07:00
|
|
|
export_options: {
|
2026-06-25 23:44:13 -07:00
|
|
|
signingStyle: "manual",
|
|
|
|
|
teamID: TEAM_ID,
|
2026-06-25 20:51:01 -07:00
|
|
|
provisioningProfiles: {
|
2026-06-25 22:41:00 -07:00
|
|
|
APP_IDENTIFIER => PROFILE_NAME
|
|
|
|
|
}
|
2026-06-05 23:19:14 -07:00
|
|
|
}
|
2026-06-25 23:44:13 -07:00
|
|
|
)
|
2026-06-05 23:19:14 -07:00
|
|
|
|
2026-06-25 20:51:01 -07:00
|
|
|
upload_to_testflight(
|
|
|
|
|
api_key: api_key,
|
|
|
|
|
skip_waiting_for_build_processing: true
|
|
|
|
|
)
|
2026-06-25 23:44:13 -07:00
|
|
|
ensure
|
|
|
|
|
cleanup_ci_signing
|
2026-06-05 23:19:14 -07:00
|
|
|
end
|
|
|
|
|
end
|