239 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			239 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # Gets a patch from a GitHub commit or pull request and applies it to Homebrew.
 | |
| # Optionally, installs it too.
 | |
| 
 | |
| require "utils"
 | |
| require "formula"
 | |
| require "cmd/tap"
 | |
| 
 | |
| module Homebrew
 | |
|   HOMEBREW_PULL_API_REGEX = %r{https://api\.github\.com/repos/([\w-]+)/homebrew(-[\w-]+)?/pulls/(\d+)}
 | |
| 
 | |
|   def tap(arg)
 | |
|     match = arg.match(%r{homebrew-([\w-]+)/})
 | |
|     match[1].downcase if match
 | |
|   end
 | |
| 
 | |
|   def pull_url(url)
 | |
|     # GitHub provides commits/pull-requests raw patches using this URL.
 | |
|     url += ".patch"
 | |
| 
 | |
|     patchpath = HOMEBREW_CACHE + File.basename(url)
 | |
|     curl url, "-o", patchpath
 | |
| 
 | |
|     ohai "Applying patch"
 | |
|     patch_args = []
 | |
|     # Normally we don't want whitespace errors, but squashing them can break
 | |
|     # patches so an option is provided to skip this step.
 | |
|     if ARGV.include?("--ignore-whitespace") || ARGV.include?("--clean")
 | |
|       patch_args << "--whitespace=nowarn"
 | |
|     else
 | |
|       patch_args << "--whitespace=fix"
 | |
|     end
 | |
| 
 | |
|     # Fall back to three-way merge if patch does not apply cleanly
 | |
|     patch_args << "-3"
 | |
|     patch_args << patchpath
 | |
| 
 | |
|     begin
 | |
|       safe_system "git", "am", *patch_args
 | |
|     rescue ErrorDuringExecution
 | |
|       if ARGV.include? "--resolve"
 | |
|         odie "Patch failed to apply: try to resolve it."
 | |
|       else
 | |
|         system "git", "am", "--abort"
 | |
|         odie "Patch failed to apply: aborted."
 | |
|       end
 | |
|     ensure
 | |
|       patchpath.unlink
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def pull
 | |
|     if ARGV.empty?
 | |
|       odie "This command requires at least one argument containing a URL or pull request number"
 | |
|     end
 | |
| 
 | |
|     if ARGV[0] == "--rebase"
 | |
|       odie "You meant `git pull --rebase`."
 | |
|     end
 | |
| 
 | |
|     bintray_fetch_formulae =[]
 | |
| 
 | |
|     ARGV.named.each do |arg|
 | |
|       if arg.to_i > 0
 | |
|         url = "https://github.com/Homebrew/homebrew/pull/#{arg}"
 | |
|         issue = arg
 | |
|       elsif (testing_match = arg.match %r{brew.sh/job/Homebrew.*Testing/(\d+)/})
 | |
|         _, testing_job = *testing_match
 | |
|         url = "https://github.com/Homebrew/homebrew/compare/master...BrewTestBot:testing-#{testing_job}"
 | |
|         odie "Testing URLs require `--bottle`!" unless ARGV.include?("--bottle")
 | |
|       else
 | |
|         if (api_match = arg.match HOMEBREW_PULL_API_REGEX)
 | |
|           _, user, tap, pull = *api_match
 | |
|           arg = "https://github.com/#{user}/homebrew#{tap}/pull/#{pull}"
 | |
|         end
 | |
| 
 | |
|         url_match = arg.match HOMEBREW_PULL_OR_COMMIT_URL_REGEX
 | |
|         odie "Not a GitHub pull request or commit: #{arg}" unless url_match
 | |
| 
 | |
|         url = url_match[0]
 | |
|         issue = url_match[3]
 | |
|       end
 | |
| 
 | |
|       if !testing_job && ARGV.include?("--bottle") && issue.nil?
 | |
|         raise "No pull request detected!"
 | |
|       end
 | |
| 
 | |
|       if !testing_job && tap_name = tap(url)
 | |
|         user = url_match[1].downcase
 | |
|         tap_dir = HOMEBREW_REPOSITORY/"Library/Taps/#{user}/homebrew-#{tap_name}"
 | |
|         safe_system "brew", "tap", "#{user}/#{tap_name}" unless tap_dir.exist?
 | |
|         Dir.chdir tap_dir
 | |
|       else
 | |
|         Dir.chdir HOMEBREW_REPOSITORY
 | |
|       end
 | |
| 
 | |
|       # The cache directory seems like a good place to put patches.
 | |
|       HOMEBREW_CACHE.mkpath
 | |
| 
 | |
|       # Store current revision and branch
 | |
|       revision = `git rev-parse --short HEAD`.strip
 | |
|       branch = `git symbolic-ref --short HEAD`.strip
 | |
| 
 | |
|       unless branch == "master"
 | |
|         opoo "Current branch is #{branch}: do you need to pull inside master?" unless ARGV.include? "--clean"
 | |
|       end
 | |
| 
 | |
|       pull_url url
 | |
| 
 | |
|       changed_formulae = []
 | |
| 
 | |
|       if tap_dir
 | |
|         formula_dir = %w[Formula HomebrewFormula].find { |d| tap_dir.join(d).directory? } || ""
 | |
|       else
 | |
|         formula_dir = "Library/Formula"
 | |
|       end
 | |
| 
 | |
|       Utils.popen_read(
 | |
|         "git", "diff-tree", "-r", "--name-only",
 | |
|         "--diff-filter=AM", revision, "HEAD", "--", formula_dir
 | |
|       ).each_line do |line|
 | |
|         name = File.basename(line.chomp, ".rb")
 | |
| 
 | |
|         begin
 | |
|           changed_formulae << Formula[name]
 | |
|         # Make sure we catch syntax errors.
 | |
|         rescue Exception
 | |
|           next
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       fetch_bottles = false
 | |
|       changed_formulae.each do |f|
 | |
|         if ARGV.include? "--bottle"
 | |
|           if f.bottle_unneeded?
 | |
|             ohai "#{f}: skipping unneeded bottle."
 | |
|           elsif f.bottle_disabled?
 | |
|             ohai "#{f}: skipping disabled bottle: #{f.bottle_disable_reason}"
 | |
|           else
 | |
|             fetch_bottles = true
 | |
|           end
 | |
|         else
 | |
|           next unless f.bottle_defined?
 | |
|           opoo "#{f.full_name} has a bottle: do you need to update it with --bottle?"
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       if issue && !ARGV.include?("--clean")
 | |
|         ohai "Patch closes issue ##{issue}"
 | |
|         message = `git log HEAD^.. --format=%B`
 | |
| 
 | |
|         if ARGV.include? "--bump"
 | |
|           odie "Can only bump one changed formula" unless changed_formulae.length == 1
 | |
|           formula = changed_formulae.first
 | |
|           subject = "#{formula.name} #{formula.version}"
 | |
|           ohai "New bump commit subject: #{subject}"
 | |
|           system "/bin/echo -n #{subject} | pbcopy"
 | |
|           message = "#{subject}\n\n#{message}"
 | |
|         end
 | |
| 
 | |
|         # If this is a pull request, append a close message.
 | |
|         unless message.include? "Closes ##{issue}."
 | |
|           message += "\nCloses ##{issue}."
 | |
|           safe_system "git", "commit", "--amend", "--signoff", "--allow-empty", "-q", "-m", message
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       if fetch_bottles
 | |
|         bottle_commit_url = if testing_job
 | |
|           bottle_branch = "testing-bottle-#{testing_job}"
 | |
|           url
 | |
|         else
 | |
|           bottle_branch = "pull-bottle-#{issue}"
 | |
|           if tap_name
 | |
|             "https://github.com/BrewTestBot/homebrew-#{tap_name}/compare/homebrew:master...pr-#{issue}"
 | |
|           else
 | |
|             "https://github.com/BrewTestBot/homebrew/compare/homebrew:master...pr-#{issue}"
 | |
|           end
 | |
|         end
 | |
|         curl "--silent", "--fail", "-o", "/dev/null", "-I", bottle_commit_url
 | |
| 
 | |
|         safe_system "git", "checkout", "-B", bottle_branch, revision
 | |
|         pull_url bottle_commit_url
 | |
|         safe_system "git", "rebase", branch
 | |
|         safe_system "git", "checkout", branch
 | |
|         safe_system "git", "merge", "--ff-only", "--no-edit", bottle_branch
 | |
|         safe_system "git", "branch", "-D", bottle_branch
 | |
| 
 | |
|         # Publish bottles on Bintray
 | |
|         bintray_user = ENV["BINTRAY_USER"]
 | |
|         bintray_key = ENV["BINTRAY_KEY"]
 | |
| 
 | |
|         if bintray_user && bintray_key
 | |
|           repo = Bintray.repository(tap_name)
 | |
|           changed_formulae.each do |f|
 | |
|             next if f.bottle_unneeded? || f.bottle_disabled?
 | |
|             ohai "Publishing on Bintray:"
 | |
|             package = Bintray.package f.name
 | |
|             version = f.pkg_version
 | |
|             curl "-w", '\n', "--silent", "--fail",
 | |
|               "-u#{bintray_user}:#{bintray_key}", "-X", "POST",
 | |
|               "-d", '{"publish_wait_for_secs": -1}',
 | |
|               "https://api.bintray.com/content/homebrew/#{repo}/#{package}/#{version}/publish"
 | |
|             bintray_fetch_formulae << f
 | |
|           end
 | |
|         else
 | |
|           opoo "You must set BINTRAY_USER and BINTRAY_KEY to add or update bottles on Bintray!"
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       ohai "Patch changed:"
 | |
|       safe_system "git", "diff-tree", "-r", "--stat", revision, "HEAD"
 | |
| 
 | |
|       if ARGV.include? "--install"
 | |
|         changed_formulae.each do |f|
 | |
|           ohai "Installing #{f.full_name}"
 | |
|           install = f.installed? ? "upgrade" : "install"
 | |
|           safe_system "brew", install, "--debug", f.full_name
 | |
|         end
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     bintray_fetch_formulae.each do |f|
 | |
|       max_retries = 8
 | |
|       retry_count = 0
 | |
|       begin
 | |
|         success = system "brew", "fetch", "--force-bottle", f.full_name
 | |
|         raise "Failed to download #{f} bottle!" unless success
 | |
|       rescue RuntimeError => e
 | |
|         retry_count += 1
 | |
|         raise e if retry_count >= max_retries
 | |
|         sleep_seconds = 2**retry_count
 | |
|         ohai "That didn't work; sleeping #{sleep_seconds} seconds and trying again..."
 | |
|         sleep sleep_seconds
 | |
|         retry
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| end
 | 
