Fastfile 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. # This file contains the fastlane.tools configuration
  2. # You can find the documentation at https://docs.fastlane.tools
  3. #
  4. # For a list of all available actions, check out
  5. #
  6. # https://docs.fastlane.tools/actions
  7. #
  8. # For a list of all available plugins, check out
  9. #
  10. # https://docs.fastlane.tools/plugins/available-plugins
  11. #
  12. default_platform(:ios)
  13. TEAMID = ENV["TEAMID"]
  14. GH_PAT = ENV["GH_PAT"]
  15. GITHUB_WORKSPACE = ENV["GITHUB_WORKSPACE"]
  16. GITHUB_REPOSITORY_OWNER = ENV["GITHUB_REPOSITORY_OWNER"]
  17. FASTLANE_KEY_ID = ENV["FASTLANE_KEY_ID"]
  18. FASTLANE_ISSUER_ID = ENV["FASTLANE_ISSUER_ID"]
  19. FASTLANE_KEY = ENV["FASTLANE_KEY"]
  20. DEVICE_NAME = ENV["DEVICE_NAME"]
  21. DEVICE_ID = ENV["DEVICE_ID"]
  22. ENV["FASTLANE_XCODEBUILD_SETTINGS_TIMEOUT"] = "120"
  23. # Define method to parse xcconfig file, and replace $(DEVELOPMENT_TEAM) with ENV["TEAMID"]
  24. def parse_xcconfig_file(path)
  25. xcconfig = {}
  26. File.open(path).each_line do |line|
  27. line.strip!
  28. next if line.empty? || line.start_with?('//')
  29. parts = line.split('=')
  30. next if parts.length < 2 # Skip lines without '='
  31. key, value = parts.map(&:strip)
  32. # Replace $(DEVELOPMENT_TEAM) with ENV["TEAMID"]
  33. value = value.gsub('$(DEVELOPMENT_TEAM)', TEAMID)
  34. xcconfig[key] = value
  35. end
  36. xcconfig
  37. end
  38. # Path to config.xcconfig file
  39. xcconfig_path = "#{GITHUB_WORKSPACE}/Config.xcconfig"
  40. # Load the variables from config.xcconfig
  41. xcconfig = parse_xcconfig_file(xcconfig_path)
  42. # Access BUNDLE_IDENTIFIER from the xcconfig file after replacing $(DEVELOPMENT_TEAM) with ENV["TEAMID"]
  43. ENV["BUNDLE_ID"] = xcconfig["BUNDLE_IDENTIFIER"]
  44. # limit lane names to letters and underscores
  45. platform :ios do
  46. desc "Build Trio"
  47. lane :build_trio do
  48. setup_ci if ENV['CI']
  49. BUNDLE_ID = ENV["BUNDLE_ID"]
  50. update_project_team(
  51. path: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  52. teamid: "#{TEAMID}"
  53. )
  54. api_key = app_store_connect_api_key(
  55. key_id: "#{FASTLANE_KEY_ID}",
  56. issuer_id: "#{FASTLANE_ISSUER_ID}",
  57. key_content: "#{FASTLANE_KEY}"
  58. )
  59. previous_build_number = latest_testflight_build_number(
  60. app_identifier: "#{BUNDLE_ID}",
  61. api_key: api_key,
  62. )
  63. current_build_number = previous_build_number + 1
  64. increment_build_number(
  65. xcodeproj: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  66. build_number: current_build_number
  67. )
  68. match(
  69. type: "appstore",
  70. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}"),
  71. app_identifier: [
  72. "#{BUNDLE_ID}",
  73. "#{BUNDLE_ID}.watchkitapp",
  74. "#{BUNDLE_ID}.watchkitapp.TrioWatchComplication",
  75. "#{BUNDLE_ID}.LiveActivity"
  76. ]
  77. )
  78. previous_build_number = latest_testflight_build_number(
  79. app_identifier: "#{BUNDLE_ID}",
  80. api_key: api_key,
  81. )
  82. current_build_number = previous_build_number + 1
  83. increment_build_number(
  84. xcodeproj: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  85. build_number: current_build_number
  86. )
  87. mapping = Actions.lane_context[
  88. SharedValues::MATCH_PROVISIONING_PROFILE_MAPPING
  89. ]
  90. update_code_signing_settings(
  91. path: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  92. profile_name: mapping["#{BUNDLE_ID}"],
  93. code_sign_identity: "iPhone Distribution",
  94. targets: ["Trio"]
  95. )
  96. update_code_signing_settings(
  97. path: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  98. profile_name: mapping["#{BUNDLE_ID}.watchkitapp"],
  99. code_sign_identity: "iPhone Distribution",
  100. targets: ["Trio Watch App"]
  101. )
  102. update_code_signing_settings(
  103. path: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  104. profile_name: mapping["#{BUNDLE_ID}.watchkitapp.TrioWatchComplication"],
  105. code_sign_identity: "iPhone Distribution",
  106. targets: ["Trio Watch Complication Extension"]
  107. )
  108. update_code_signing_settings(
  109. path: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  110. profile_name: mapping["#{BUNDLE_ID}.LiveActivity"],
  111. code_sign_identity: "iPhone Distribution",
  112. targets: ["LiveActivityExtension"]
  113. )
  114. gym(
  115. export_method: "app-store",
  116. scheme: "Trio",
  117. output_name: "Trio.ipa",
  118. configuration: "Release",
  119. destination: 'generic/platform=iOS',
  120. buildlog_path: 'buildlog'
  121. )
  122. copy_artifacts(
  123. target_path: "artifacts",
  124. artifacts: ["*.mobileprovision", "*.ipa", "*.dSYM.zip"]
  125. )
  126. end
  127. desc "Push to TestFlight"
  128. lane :release do
  129. api_key = app_store_connect_api_key(
  130. key_id: "#{FASTLANE_KEY_ID}",
  131. issuer_id: "#{FASTLANE_ISSUER_ID}",
  132. key_content: "#{FASTLANE_KEY}"
  133. )
  134. upload_to_testflight(
  135. api_key: api_key,
  136. skip_submission: false,
  137. ipa: "Trio.ipa",
  138. skip_waiting_for_build_processing: true,
  139. changelog: git_branch+" "+last_git_commit[:abbreviated_commit_hash],
  140. )
  141. end
  142. desc "Provision Identifiers and Certificates"
  143. lane :identifiers do
  144. setup_ci if ENV['CI']
  145. ENV["MATCH_READONLY"] = false.to_s
  146. BUNDLE_ID = ENV["BUNDLE_ID"]
  147. app_store_connect_api_key(
  148. key_id: "#{FASTLANE_KEY_ID}",
  149. issuer_id: "#{FASTLANE_ISSUER_ID}",
  150. key_content: "#{FASTLANE_KEY}"
  151. )
  152. def configure_bundle_id(name, identifier, capabilities)
  153. bundle_id = Spaceship::ConnectAPI::BundleId.find(identifier) || Spaceship::ConnectAPI::BundleId.create(
  154. name: name,
  155. identifier: identifier,
  156. platform: "IOS"
  157. )
  158. existing = bundle_id.get_capabilities.map(&:capability_type)
  159. capabilities.reject { |c| existing.include?(c) }.each do |cap|
  160. bundle_id.create_capability(cap)
  161. end
  162. end
  163. configure_bundle_id("Trio", "#{BUNDLE_ID}", [
  164. Spaceship::ConnectAPI::BundleIdCapability::Type::APP_GROUPS,
  165. Spaceship::ConnectAPI::BundleIdCapability::Type::HEALTHKIT,
  166. Spaceship::ConnectAPI::BundleIdCapability::Type::NFC_TAG_READING,
  167. Spaceship::ConnectAPI::BundleIdCapability::Type::PUSH_NOTIFICATIONS
  168. ])
  169. configure_bundle_id("Trio Watch Complication", "#{BUNDLE_ID}.watchkitapp.TrioWatchComplication", [
  170. Spaceship::ConnectAPI::BundleIdCapability::Type::APP_GROUPS
  171. ])
  172. configure_bundle_id("Trio Watch App", "#{BUNDLE_ID}.watchkitapp", [
  173. Spaceship::ConnectAPI::BundleIdCapability::Type::APP_GROUPS,
  174. Spaceship::ConnectAPI::BundleIdCapability::Type::HEALTHKIT
  175. ])
  176. configure_bundle_id("Trio LiveActivity", "#{BUNDLE_ID}.LiveActivity", [])
  177. end
  178. desc "Provision Certificates"
  179. lane :certs do
  180. setup_ci if ENV['CI']
  181. ENV["MATCH_READONLY"] = false.to_s
  182. BUNDLE_ID = ENV["BUNDLE_ID"]
  183. app_store_connect_api_key(
  184. key_id: "#{FASTLANE_KEY_ID}",
  185. issuer_id: "#{FASTLANE_ISSUER_ID}",
  186. key_content: "#{FASTLANE_KEY}"
  187. )
  188. match(
  189. type: "appstore",
  190. force: false,
  191. verbose: true,
  192. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}"),
  193. app_identifier: [
  194. "#{BUNDLE_ID}",
  195. "#{BUNDLE_ID}.watchkitapp",
  196. "#{BUNDLE_ID}.watchkitapp.TrioWatchComplication",
  197. "#{BUNDLE_ID}.LiveActivity"
  198. ]
  199. )
  200. end
  201. desc "Validate Secrets"
  202. lane :validate_secrets do
  203. setup_ci if ENV['CI']
  204. ENV["MATCH_READONLY"] = true.to_s
  205. BUNDLE_ID = ENV["BUNDLE_ID"]
  206. app_store_connect_api_key(
  207. key_id: "#{FASTLANE_KEY_ID}",
  208. issuer_id: "#{FASTLANE_ISSUER_ID}",
  209. key_content: "#{FASTLANE_KEY}"
  210. )
  211. def find_bundle_id(identifier)
  212. bundle_id = Spaceship::ConnectAPI::BundleId.find(identifier)
  213. end
  214. find_bundle_id("#{BUNDLE_ID}")
  215. match(
  216. type: "appstore",
  217. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}"),
  218. app_identifier: [],
  219. )
  220. end
  221. desc "Nuke Certs"
  222. lane :nuke_certs do
  223. setup_ci if ENV['CI']
  224. ENV["MATCH_READONLY"] = false.to_s
  225. app_store_connect_api_key(
  226. key_id: "#{FASTLANE_KEY_ID}",
  227. issuer_id: "#{FASTLANE_ISSUER_ID}",
  228. key_content: "#{FASTLANE_KEY}"
  229. )
  230. match_nuke(
  231. type: "appstore",
  232. team_id: "#{TEAMID}",
  233. skip_confirmation: true,
  234. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}")
  235. )
  236. end
  237. desc "Check Certificates and Trigger Workflow for Expired or Missing Certificates"
  238. lane :check_and_renew_certificates do
  239. setup_ci if ENV['CI']
  240. ENV["MATCH_READONLY"] = false.to_s
  241. # Authenticate using App Store Connect API Key
  242. api_key = app_store_connect_api_key(
  243. key_id: ENV["FASTLANE_KEY_ID"],
  244. issuer_id: ENV["FASTLANE_ISSUER_ID"],
  245. key_content: ENV["FASTLANE_KEY"] # Ensure valid key content
  246. )
  247. # Initialize flag to track if renewal of certificates is needed
  248. new_certificate_needed = false
  249. # Fetch all certificates
  250. certificates = Spaceship::ConnectAPI::Certificate.all
  251. # Filter for Distribution Certificates
  252. distribution_certs = certificates.select { |cert| cert.certificate_type == "DISTRIBUTION" }
  253. # Handle case where no distribution certificates are found
  254. if distribution_certs.empty?
  255. puts "No Distribution certificates found! Triggering action to create certificate."
  256. new_certificate_needed = true
  257. else
  258. # Check for expiration
  259. distribution_certs.each do |cert|
  260. expiration_date = Time.parse(cert.expiration_date)
  261. puts "Current Distribution Certificate: #{cert.id}, Expiration date: #{expiration_date}"
  262. if expiration_date < Time.now
  263. puts "Distribution Certificate #{cert.id} is expired! Triggering action to renew certificate."
  264. new_certificate_needed = true
  265. else
  266. puts "Distribution certificate #{cert.id} is valid. No action required."
  267. end
  268. end
  269. end
  270. # Write result to new_certificate_needed.txt
  271. file_path = File.expand_path('new_certificate_needed.txt')
  272. File.write(file_path, new_certificate_needed ? 'true' : 'false')
  273. # Log the absolute path and contents of the new_certificate_needed.txt file
  274. puts ""
  275. puts "Absolute path of new_certificate_needed.txt: #{file_path}"
  276. new_certificate_needed_content = File.read(file_path)
  277. puts "Certificate creation or renewal needed: #{new_certificate_needed_content}"
  278. end
  279. end