Fastfile 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  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. mapping = Actions.lane_context[
  79. SharedValues::MATCH_PROVISIONING_PROFILE_MAPPING
  80. ]
  81. update_code_signing_settings(
  82. path: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  83. profile_name: mapping["#{BUNDLE_ID}"],
  84. code_sign_identity: "iPhone Distribution",
  85. targets: ["Trio"]
  86. )
  87. update_code_signing_settings(
  88. path: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  89. profile_name: mapping["#{BUNDLE_ID}.watchkitapp"],
  90. code_sign_identity: "iPhone Distribution",
  91. targets: ["Trio Watch App"]
  92. )
  93. update_code_signing_settings(
  94. path: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  95. profile_name: mapping["#{BUNDLE_ID}.watchkitapp.TrioWatchComplication"],
  96. code_sign_identity: "iPhone Distribution",
  97. targets: ["Trio Watch Complication Extension"]
  98. )
  99. update_code_signing_settings(
  100. path: "#{GITHUB_WORKSPACE}/Trio.xcodeproj",
  101. profile_name: mapping["#{BUNDLE_ID}.LiveActivity"],
  102. code_sign_identity: "iPhone Distribution",
  103. targets: ["LiveActivityExtension"]
  104. )
  105. gym(
  106. export_method: "app-store",
  107. scheme: "Trio",
  108. output_name: "Trio.ipa",
  109. configuration: "Release",
  110. destination: 'generic/platform=iOS',
  111. buildlog_path: 'buildlog'
  112. )
  113. copy_artifacts(
  114. target_path: "artifacts",
  115. artifacts: ["*.mobileprovision", "*.ipa", "*.dSYM.zip"]
  116. )
  117. end
  118. desc "Push to TestFlight"
  119. lane :release do
  120. api_key = app_store_connect_api_key(
  121. key_id: "#{FASTLANE_KEY_ID}",
  122. issuer_id: "#{FASTLANE_ISSUER_ID}",
  123. key_content: "#{FASTLANE_KEY}"
  124. )
  125. upload_to_testflight(
  126. api_key: api_key,
  127. skip_submission: false,
  128. ipa: "Trio.ipa",
  129. skip_waiting_for_build_processing: true,
  130. changelog: git_branch+" "+last_git_commit[:abbreviated_commit_hash],
  131. )
  132. end
  133. desc "Provision Identifiers and Certificates"
  134. lane :identifiers do
  135. setup_ci if ENV['CI']
  136. ENV["MATCH_READONLY"] = false.to_s
  137. BUNDLE_ID = ENV["BUNDLE_ID"]
  138. app_store_connect_api_key(
  139. key_id: "#{FASTLANE_KEY_ID}",
  140. issuer_id: "#{FASTLANE_ISSUER_ID}",
  141. key_content: "#{FASTLANE_KEY}"
  142. )
  143. def configure_bundle_id(name, identifier, capabilities)
  144. bundle_id = Spaceship::ConnectAPI::BundleId.find(identifier) || Spaceship::ConnectAPI::BundleId.create(
  145. name: name,
  146. identifier: identifier,
  147. platform: "IOS"
  148. )
  149. existing = bundle_id.get_capabilities.map(&:capability_type)
  150. capabilities.reject { |c| existing.include?(c) }.each do |cap|
  151. bundle_id.create_capability(cap)
  152. end
  153. end
  154. configure_bundle_id("Trio", "#{BUNDLE_ID}", [
  155. Spaceship::ConnectAPI::BundleIdCapability::Type::APP_GROUPS,
  156. Spaceship::ConnectAPI::BundleIdCapability::Type::HEALTHKIT,
  157. Spaceship::ConnectAPI::BundleIdCapability::Type::NFC_TAG_READING,
  158. Spaceship::ConnectAPI::BundleIdCapability::Type::PUSH_NOTIFICATIONS
  159. ])
  160. configure_bundle_id("Trio Watch Complication", "#{BUNDLE_ID}.watchkitapp.TrioWatchComplication", [
  161. Spaceship::ConnectAPI::BundleIdCapability::Type::APP_GROUPS
  162. ])
  163. configure_bundle_id("Trio Watch App", "#{BUNDLE_ID}.watchkitapp", [
  164. Spaceship::ConnectAPI::BundleIdCapability::Type::APP_GROUPS,
  165. Spaceship::ConnectAPI::BundleIdCapability::Type::HEALTHKIT
  166. ])
  167. configure_bundle_id("Trio LiveActivity", "#{BUNDLE_ID}.LiveActivity", [])
  168. end
  169. desc "Provision Certificates"
  170. lane :certs do
  171. setup_ci if ENV['CI']
  172. ENV["MATCH_READONLY"] = false.to_s
  173. BUNDLE_ID = ENV["BUNDLE_ID"]
  174. app_store_connect_api_key(
  175. key_id: "#{FASTLANE_KEY_ID}",
  176. issuer_id: "#{FASTLANE_ISSUER_ID}",
  177. key_content: "#{FASTLANE_KEY}"
  178. )
  179. match(
  180. type: "appstore",
  181. force: false,
  182. verbose: true,
  183. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}"),
  184. app_identifier: [
  185. "#{BUNDLE_ID}",
  186. "#{BUNDLE_ID}.watchkitapp",
  187. "#{BUNDLE_ID}.watchkitapp.TrioWatchComplication",
  188. "#{BUNDLE_ID}.LiveActivity"
  189. ]
  190. )
  191. end
  192. desc "Validate Secrets"
  193. lane :validate_secrets do
  194. setup_ci if ENV['CI']
  195. ENV["MATCH_READONLY"] = true.to_s
  196. BUNDLE_ID = ENV["BUNDLE_ID"]
  197. app_store_connect_api_key(
  198. key_id: "#{FASTLANE_KEY_ID}",
  199. issuer_id: "#{FASTLANE_ISSUER_ID}",
  200. key_content: "#{FASTLANE_KEY}"
  201. )
  202. def find_bundle_id(identifier)
  203. bundle_id = Spaceship::ConnectAPI::BundleId.find(identifier)
  204. end
  205. find_bundle_id("#{BUNDLE_ID}")
  206. match(
  207. type: "appstore",
  208. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}"),
  209. app_identifier: [],
  210. )
  211. end
  212. desc "Nuke Certs"
  213. lane :nuke_certs do
  214. setup_ci if ENV['CI']
  215. ENV["MATCH_READONLY"] = false.to_s
  216. app_store_connect_api_key(
  217. key_id: "#{FASTLANE_KEY_ID}",
  218. issuer_id: "#{FASTLANE_ISSUER_ID}",
  219. key_content: "#{FASTLANE_KEY}"
  220. )
  221. match_nuke(
  222. type: "appstore",
  223. team_id: "#{TEAMID}",
  224. skip_confirmation: true,
  225. git_basic_authorization: Base64.strict_encode64("#{GITHUB_REPOSITORY_OWNER}:#{GH_PAT}")
  226. )
  227. end
  228. desc "Check Certificates and Trigger Workflow for Expired or Missing Certificates"
  229. lane :check_and_renew_certificates do
  230. setup_ci if ENV['CI']
  231. ENV["MATCH_READONLY"] = false.to_s
  232. # Authenticate using App Store Connect API Key
  233. api_key = app_store_connect_api_key(
  234. key_id: ENV["FASTLANE_KEY_ID"],
  235. issuer_id: ENV["FASTLANE_ISSUER_ID"],
  236. key_content: ENV["FASTLANE_KEY"] # Ensure valid key content
  237. )
  238. # Initialize flag to track if renewal of certificates is needed
  239. new_certificate_needed = false
  240. # Fetch all certificates
  241. certificates = Spaceship::ConnectAPI::Certificate.all
  242. # Filter for Distribution Certificates
  243. distribution_certs = certificates.select { |cert| cert.certificate_type == "DISTRIBUTION" }
  244. # Handle case where no distribution certificates are found
  245. if distribution_certs.empty?
  246. puts "No Distribution certificates found! Triggering action to create certificate."
  247. new_certificate_needed = true
  248. else
  249. # Check for expiration
  250. distribution_certs.each do |cert|
  251. expiration_date = Time.parse(cert.expiration_date)
  252. puts "Current Distribution Certificate: #{cert.id}, Expiration date: #{expiration_date}"
  253. if expiration_date < Time.now
  254. puts "Distribution Certificate #{cert.id} is expired! Triggering action to renew certificate."
  255. new_certificate_needed = true
  256. else
  257. puts "Distribution certificate #{cert.id} is valid. No action required."
  258. end
  259. end
  260. end
  261. # Write result to new_certificate_needed.txt
  262. file_path = File.expand_path('new_certificate_needed.txt')
  263. File.write(file_path, new_certificate_needed ? 'true' : 'false')
  264. # Log the absolute path and contents of the new_certificate_needed.txt file
  265. puts ""
  266. puts "Absolute path of new_certificate_needed.txt: #{file_path}"
  267. new_certificate_needed_content = File.read(file_path)
  268. puts "Certificate creation or renewal needed: #{new_certificate_needed_content}"
  269. end
  270. end