Fastfile 10.0 KB

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