| 
									
										
										
										
											2023-03-09 20:32:46 -08:00
										 |  |  | # typed: true | 
					
						
							| 
									
										
										
										
											2020-04-20 10:17:42 +02:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  | require "abstract_command" | 
					
						
							| 
									
										
										
										
											2023-07-21 22:18:00 +08:00
										 |  |  | require "formula" | 
					
						
							| 
									
										
										
										
											2021-03-11 15:15:35 +00:00
										 |  |  | require "github_packages" | 
					
						
							| 
									
										
										
										
											2021-03-10 16:08:45 +00:00
										 |  |  | require "github_releases" | 
					
						
							| 
									
										
										
										
											2024-01-11 19:22:16 -08:00
										 |  |  | require "extend/hash/deep_merge" | 
					
						
							| 
									
										
										
										
											2020-04-20 10:17:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | module Homebrew | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |   module DevCmd | 
					
						
							|  |  |  |     class PrUpload < AbstractCommand | 
					
						
							|  |  |  |       cmd_args do | 
					
						
							|  |  |  |         description <<~EOS | 
					
						
							|  |  |  |           Apply the bottle commit and publish bottles to a host. | 
					
						
							|  |  |  |         EOS | 
					
						
							|  |  |  |         switch "--keep-old", | 
					
						
							|  |  |  |                description: "If the formula specifies a rebuild version, " \ | 
					
						
							|  |  |  |                             "attempt to preserve its value in the generated DSL." | 
					
						
							|  |  |  |         switch "-n", "--dry-run", | 
					
						
							|  |  |  |                description: "Print what would be done rather than doing it." | 
					
						
							|  |  |  |         switch "--no-commit", | 
					
						
							|  |  |  |                description: "Do not generate a new commit before uploading." | 
					
						
							|  |  |  |         switch "--warn-on-upload-failure", | 
					
						
							|  |  |  |                description: "Warn instead of raising an error if the bottle upload fails. " \ | 
					
						
							|  |  |  |                             "Useful for repairing bottle uploads that previously failed." | 
					
						
							|  |  |  |         switch "--upload-only", | 
					
						
							|  |  |  |                description: "Skip running `brew bottle` before uploading." | 
					
						
							|  |  |  |         flag   "--committer=", | 
					
						
							|  |  |  |                description: "Specify a committer name and email in `git`'s standard author format." | 
					
						
							|  |  |  |         flag   "--root-url=", | 
					
						
							|  |  |  |                description: "Use the specified <URL> as the root of the bottle's URL instead of Homebrew's default." | 
					
						
							|  |  |  |         flag   "--root-url-using=", | 
					
						
							|  |  |  |                description: "Use the specified download strategy class for downloading the bottle's URL instead of " \ | 
					
						
							|  |  |  |                             "Homebrew's default." | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         conflicts "--upload-only", "--keep-old" | 
					
						
							|  |  |  |         conflicts "--upload-only", "--no-commit" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         named_args :none | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-08-20 14:21:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |       sig { override.void } | 
					
						
							|  |  |  |       def run | 
					
						
							|  |  |  |         json_files = Dir["*.bottle.json"] | 
					
						
							|  |  |  |         odie "No bottle JSON files found in the current working directory" if json_files.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Homebrew.install_bundler_gems!(groups: ["pr_upload"]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-30 10:39:35 +01:00
										 |  |  |         bottles_hash = bottles_hash_from_json_files(json_files, args) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |         unless args.upload_only? | 
					
						
							|  |  |  |           bottle_args = ["bottle", "--merge", "--write"] | 
					
						
							|  |  |  |           bottle_args << "--verbose" if args.verbose? | 
					
						
							|  |  |  |           bottle_args << "--debug" if args.debug? | 
					
						
							|  |  |  |           bottle_args << "--keep-old" if args.keep_old? | 
					
						
							|  |  |  |           bottle_args << "--root-url=#{args.root_url}" if args.root_url | 
					
						
							|  |  |  |           bottle_args << "--committer=#{args.committer}" if args.committer | 
					
						
							|  |  |  |           bottle_args << "--no-commit" if args.no_commit? | 
					
						
							|  |  |  |           bottle_args << "--root-url-using=#{args.root_url_using}" if args.root_url_using | 
					
						
							|  |  |  |           bottle_args += json_files | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if args.dry_run? | 
					
						
							|  |  |  |             dry_run_service = if github_packages?(bottles_hash) | 
					
						
							|  |  |  |               # GitHub Packages has its own --dry-run handling. | 
					
						
							|  |  |  |               nil | 
					
						
							|  |  |  |             elsif github_releases?(bottles_hash) | 
					
						
							|  |  |  |               "GitHub Releases" | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |               odie "Service specified by root_url is not recognized" | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if dry_run_service | 
					
						
							|  |  |  |               puts <<~EOS | 
					
						
							|  |  |  |                 brew #{bottle_args.join " "} | 
					
						
							|  |  |  |                 Upload bottles described by these JSON files to #{dry_run_service}: | 
					
						
							|  |  |  |                   #{json_files.join("\n  ")} | 
					
						
							|  |  |  |               EOS | 
					
						
							|  |  |  |               return | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           check_bottled_formulae!(bottles_hash) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           safe_system HOMEBREW_BREW_FILE, *bottle_args | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           json_files = Dir["*.bottle.json"] | 
					
						
							|  |  |  |           if json_files.blank? | 
					
						
							|  |  |  |             puts "No bottle JSON files after merge, no upload needed!" | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           # Reload the JSON files (in case `brew bottle --merge` generated | 
					
						
							|  |  |  |           # `all: $SHA256` bottles) | 
					
						
							|  |  |  |           bottles_hash = bottles_hash_from_json_files(json_files, args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           # Check the bottle commits did not break `brew audit` | 
					
						
							|  |  |  |           unless args.no_commit? | 
					
						
							|  |  |  |             audit_args = ["audit", "--skip-style"] | 
					
						
							|  |  |  |             audit_args << "--verbose" if args.verbose? | 
					
						
							|  |  |  |             audit_args << "--debug" if args.debug? | 
					
						
							|  |  |  |             audit_args += bottles_hash.keys | 
					
						
							|  |  |  |             safe_system HOMEBREW_BREW_FILE, *audit_args | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-08-20 14:21:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |         if github_releases?(bottles_hash) | 
					
						
							|  |  |  |           github_releases = GitHubReleases.new | 
					
						
							|  |  |  |           github_releases.upload_bottles(bottles_hash) | 
					
						
							|  |  |  |         elsif github_packages?(bottles_hash) | 
					
						
							|  |  |  |           github_packages = GitHubPackages.new | 
					
						
							|  |  |  |           github_packages.upload_bottles(bottles_hash, | 
					
						
							|  |  |  |                                          keep_old:      args.keep_old?, | 
					
						
							|  |  |  |                                          dry_run:       args.dry_run?, | 
					
						
							|  |  |  |                                          warn_on_error: args.warn_on_upload_failure?) | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           odie "Service specified by root_url is not recognized" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2021-03-11 15:15:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |       private | 
					
						
							| 
									
										
										
										
											2020-07-26 19:03:02 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |       def check_bottled_formulae!(bottles_hash) | 
					
						
							|  |  |  |         bottles_hash.each do |name, bottle_hash| | 
					
						
							|  |  |  |           formula_path = HOMEBREW_REPOSITORY/bottle_hash["formula"]["path"] | 
					
						
							|  |  |  |           formula_version = Formulary.factory(formula_path).pkg_version | 
					
						
							|  |  |  |           bottle_version = PkgVersion.parse bottle_hash["formula"]["pkg_version"] | 
					
						
							|  |  |  |           next if formula_version == bottle_version | 
					
						
							| 
									
										
										
										
											2020-08-20 14:21:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |           odie "Bottles are for #{name} #{bottle_version} but formula is version #{formula_version}!" | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-03-30 17:35:13 +01:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |       def github_releases?(bottles_hash) | 
					
						
							|  |  |  |         @github_releases ||= bottles_hash.values.all? do |bottle_hash| | 
					
						
							|  |  |  |           root_url = bottle_hash["bottle"]["root_url"] | 
					
						
							|  |  |  |           url_match = root_url.match GitHubReleases::URL_REGEX | 
					
						
							|  |  |  |           _, _, _, tag = *url_match | 
					
						
							| 
									
										
										
										
											2021-04-21 17:10:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |           tag | 
					
						
							| 
									
										
										
										
											2021-10-21 02:39:27 +01:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2021-10-21 02:39:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |       def github_packages?(bottles_hash) | 
					
						
							|  |  |  |         @github_packages ||= bottles_hash.values.all? do |bottle_hash| | 
					
						
							|  |  |  |           bottle_hash["bottle"]["root_url"].match? GitHubPackages::URL_REGEX | 
					
						
							| 
									
										
										
										
											2021-10-21 02:39:27 +01:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-03-30 08:42:15 +01:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-08-20 14:21:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |       def bottles_hash_from_json_files(json_files, args) | 
					
						
							|  |  |  |         puts "Reading JSON files: #{json_files.join(", ")}" if args.verbose? | 
					
						
							| 
									
										
										
										
											2021-04-30 12:00:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |         bottles_hash = json_files.reduce({}) do |hash, json_file| | 
					
						
							|  |  |  |           hash.deep_merge(JSON.parse(File.read(json_file))) | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-04-29 14:48:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |         if args.root_url | 
					
						
							|  |  |  |           bottles_hash.each_value do |bottle_hash| | 
					
						
							|  |  |  |             bottle_hash["bottle"]["root_url"] = args.root_url | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-04-21 17:10:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 21:33:58 -07:00
										 |  |  |         bottles_hash | 
					
						
							| 
									
										
										
										
											2021-10-21 02:39:27 +01:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-11-09 09:59:59 +00:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2020-04-20 10:17:42 +02:00
										 |  |  |   end | 
					
						
							|  |  |  | end |