Fastfile 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  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(name: name, identifier: identifier)
  154. capabilities.each { |capability|
  155. bundle_id.create_capability(capability)
  156. }
  157. end
  158. configure_bundle_id("Trio", "#{BUNDLE_ID}", [
  159. Spaceship::ConnectAPI::BundleIdCapability::Type::APP_GROUPS,
  160. Spaceship::ConnectAPI::BundleIdCapability::Type::HEALTHKIT,
  161. Spaceship::ConnectAPI::BundleIdCapability::Type::NFC_TAG_READING,
  162. Spaceship::ConnectAPI::BundleIdCapability::Type::PUSH_NOTIFICATIONS
  163. ])
  164. configure_bundle_id("Trio Watch Complication", "#{BUNDLE_ID}.watchkitapp.TrioWatchComplication", [
  165. Spaceship::ConnectAPI::BundleIdCapability::Type::APP_GROUPS
  166. ])
  167. configure_bundle_id("Trio Watch App", "#{BUNDLE_ID}.watchkitapp", [
  168. Spaceship::ConnectAPI::BundleIdCapability::Type::APP_GROUPS,
  169. Spaceship::ConnectAPI::BundleIdCapability::Type::HEALTHKIT
  170. ])
  171. configure_bundle_id("Trio LiveActivity", "#{BUNDLE_ID}.LiveActivity", [])
  172. end
  173. desc "Provision Certificates"
  174. lane :certs do
  175. setup_ci if ENV['CI']
  176. ENV["MATCH_READONLY"] = false.to_s
  177. BUNDLE_ID = ENV["BUNDLE_ID"]
  178. app_store_connect_api_key(
  179. key_id: "#{FASTLANE_KEY_ID}",
  180. issuer_id: "#{FASTLANE_ISSUER_ID}",
  181. key_content: "#{FASTLANE_KEY}"
  182. )
  183. match(
  184. type: "appstore",
  185. force: false,
  186. verbose: true,
  187. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}"),
  188. app_identifier: [
  189. "#{BUNDLE_ID}",
  190. "#{BUNDLE_ID}.watchkitapp",
  191. "#{BUNDLE_ID}.watchkitapp.TrioWatchComplication",
  192. "#{BUNDLE_ID}.LiveActivity"
  193. ]
  194. )
  195. end
  196. desc "Validate Secrets"
  197. lane :validate_secrets do
  198. setup_ci if ENV['CI']
  199. ENV["MATCH_READONLY"] = true.to_s
  200. BUNDLE_ID = ENV["BUNDLE_ID"]
  201. app_store_connect_api_key(
  202. key_id: "#{FASTLANE_KEY_ID}",
  203. issuer_id: "#{FASTLANE_ISSUER_ID}",
  204. key_content: "#{FASTLANE_KEY}"
  205. )
  206. def find_bundle_id(identifier)
  207. bundle_id = Spaceship::ConnectAPI::BundleId.find(identifier)
  208. end
  209. find_bundle_id("#{BUNDLE_ID}")
  210. match(
  211. type: "appstore",
  212. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}"),
  213. app_identifier: [],
  214. )
  215. end
  216. desc "Nuke Certs"
  217. lane :nuke_certs do
  218. setup_ci if ENV['CI']
  219. ENV["MATCH_READONLY"] = false.to_s
  220. app_store_connect_api_key(
  221. key_id: "#{FASTLANE_KEY_ID}",
  222. issuer_id: "#{FASTLANE_ISSUER_ID}",
  223. key_content: "#{FASTLANE_KEY}"
  224. )
  225. match_nuke(
  226. type: "appstore",
  227. team_id: "#{TEAMID}",
  228. skip_confirmation: true,
  229. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}")
  230. )
  231. end
  232. desc "Check Certificates and Trigger Workflow for Expired or Missing Certificates"
  233. lane :check_and_renew_certificates do
  234. setup_ci if ENV['CI']
  235. ENV["MATCH_READONLY"] = false.to_s
  236. # Authenticate using App Store Connect API Key
  237. api_key = app_store_connect_api_key(
  238. key_id: ENV["FASTLANE_KEY_ID"],
  239. issuer_id: ENV["FASTLANE_ISSUER_ID"],
  240. key_content: ENV["FASTLANE_KEY"] # Ensure valid key content
  241. )
  242. # Initialize flag to track if renewal of certificates is needed
  243. new_certificate_needed = false
  244. # Fetch all certificates
  245. certificates = Spaceship::ConnectAPI::Certificate.all
  246. # Filter for Distribution Certificates
  247. distribution_certs = certificates.select { |cert| cert.certificate_type == "DISTRIBUTION" }
  248. # Handle case where no distribution certificates are found
  249. if distribution_certs.empty?
  250. puts "No Distribution certificates found! Triggering action to create certificate."
  251. new_certificate_needed = true
  252. else
  253. # Check for expiration
  254. distribution_certs.each do |cert|
  255. expiration_date = Time.parse(cert.expiration_date)
  256. puts "Current Distribution Certificate: #{cert.id}, Expiration date: #{expiration_date}"
  257. if expiration_date < Time.now
  258. puts "Distribution Certificate #{cert.id} is expired! Triggering action to renew certificate."
  259. new_certificate_needed = true
  260. else
  261. puts "Distribution certificate #{cert.id} is valid. No action required."
  262. end
  263. end
  264. end
  265. # Write result to new_certificate_needed.txt
  266. file_path = File.expand_path('new_certificate_needed.txt')
  267. File.write(file_path, new_certificate_needed ? 'true' : 'false')
  268. # Log the absolute path and contents of the new_certificate_needed.txt file
  269. puts ""
  270. puts "Absolute path of new_certificate_needed.txt: #{file_path}"
  271. new_certificate_needed_content = File.read(file_path)
  272. puts "Certificate creation or renewal needed: #{new_certificate_needed_content}"
  273. end
  274. end