214 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			214 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| require 'cmd/tap'
 | |
| require 'cmd/untap'
 | |
| 
 | |
| module Homebrew extend self
 | |
|   def update
 | |
|     unless ARGV.named.empty?
 | |
|       abort <<-EOS.undent
 | |
|         This command updates brew itself, and does not take formula names.
 | |
|         Use `brew upgrade <formula>`.
 | |
|       EOS
 | |
|     end
 | |
|     abort "Please `brew install git' first." unless which "git"
 | |
| 
 | |
|     # ensure GIT_CONFIG is unset as we need to operate on .git/config
 | |
|     ENV.delete('GIT_CONFIG')
 | |
| 
 | |
|     cd HOMEBREW_REPOSITORY
 | |
|     git_init_if_necessary
 | |
| 
 | |
|     tapped_formulae = Dir['Library/Formula/*'].map do |formula|
 | |
|       path = Pathname.new formula
 | |
|       next unless path.symlink?
 | |
|       Pathname.new(path.realpath.to_s.gsub(/.*Taps\//, '')) rescue nil
 | |
|     end
 | |
|     tapped_formulae.compact!
 | |
|     unlink_tap_formula(tapped_formulae)
 | |
| 
 | |
|     report = Report.new
 | |
|     master_updater = Updater.new
 | |
|     begin
 | |
|       master_updater.pull!
 | |
|     ensure
 | |
|       link_tap_formula(tapped_formulae)
 | |
|     end
 | |
|     report.merge!(master_updater.report)
 | |
| 
 | |
|     Dir["Library/Taps/*"].each do |tapd|
 | |
|       next unless File.directory?(tapd)
 | |
| 
 | |
|       cd tapd do
 | |
|         begin
 | |
|           updater = Updater.new
 | |
|           updater.pull!
 | |
|           report.merge!(updater.report) do |key, oldval, newval|
 | |
|             oldval.concat(newval)
 | |
|           end
 | |
|         rescue
 | |
|           tapd =~ %r{^Library/Taps/(\w+)-(\w+)}
 | |
|           onoe "Failed to update tap: #$1/#$2"
 | |
|         end
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     # we unlink first in case the formula has moved to another tap
 | |
|     Homebrew.unlink_tap_formula(report.removed_tapped_formula)
 | |
|     Homebrew.link_tap_formula(report.new_tapped_formula)
 | |
| 
 | |
|     if report.empty?
 | |
|       puts "Already up-to-date."
 | |
|     else
 | |
|       puts "Updated Homebrew from #{master_updater.initial_revision[0,8]} to #{master_updater.current_revision[0,8]}."
 | |
|       report.dump
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def git_init_if_necessary
 | |
|     if Dir['.git/*'].empty?
 | |
|       safe_system "git init"
 | |
|       safe_system "git config core.autocrlf false"
 | |
|       safe_system "git remote add origin https://github.com/mxcl/homebrew.git"
 | |
|       safe_system "git fetch origin"
 | |
|       safe_system "git reset --hard origin/master"
 | |
|     end
 | |
|   rescue Exception
 | |
|     FileUtils.rm_rf ".git"
 | |
|     raise
 | |
|   end
 | |
| 
 | |
| end
 | |
| 
 | |
| class Updater
 | |
|   attr_reader :initial_revision, :current_revision
 | |
| 
 | |
|   def pull!
 | |
|     safe_system "git checkout -q master"
 | |
| 
 | |
|     @initial_revision = read_current_revision
 | |
| 
 | |
|     # ensure we don't munge line endings on checkout
 | |
|     safe_system "git config core.autocrlf false"
 | |
| 
 | |
|     args = ["pull"]
 | |
|     args << "--rebase" if ARGV.include? "--rebase"
 | |
|     args << "-q" unless ARGV.verbose?
 | |
|     args << "origin"
 | |
|     # the refspec ensures that 'origin/master' gets updated
 | |
|     args << "refs/heads/master:refs/remotes/origin/master"
 | |
| 
 | |
|     reset_on_interrupt { safe_system "git", *args }
 | |
| 
 | |
|     @current_revision = read_current_revision
 | |
|   end
 | |
| 
 | |
|   def reset_on_interrupt
 | |
|     ignore_interrupts { yield }
 | |
|   ensure
 | |
|     if $?.signaled? && $?.termsig == 2 # SIGINT
 | |
|       safe_system "git", "reset", "--hard", @initial_revision
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # Matches raw git diff format (see `man git-diff-tree`)
 | |
|   DIFFTREE_RX = /^:[0-7]{6} [0-7]{6} [0-9a-fA-F]{40} [0-9a-fA-F]{40} ([ACDMR])\d{0,3}\t(.+?)(?:\t(.+))?$/
 | |
| 
 | |
|   def report
 | |
|     map = Hash.new{ |h,k| h[k] = [] }
 | |
| 
 | |
|     if initial_revision && initial_revision != current_revision
 | |
|       `git diff-tree -r --raw -M85% #{initial_revision} #{current_revision}`.each_line do |line|
 | |
|         DIFFTREE_RX.match line
 | |
|         path = case status = $1.to_sym
 | |
|           when :R then $3
 | |
|           else $2
 | |
|           end
 | |
|         path = Pathname.pwd.join(path).relative_path_from(HOMEBREW_REPOSITORY)
 | |
|         map[status] << path.to_s
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     map
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def read_current_revision
 | |
|     `git rev-parse -q --verify HEAD`.chomp
 | |
|   end
 | |
| 
 | |
|   def `(cmd)
 | |
|     out = Kernel.`(cmd) #`
 | |
|     if $? && !$?.success?
 | |
|       $stderr.puts out
 | |
|       raise ErrorDuringExecution, "Failure while executing: #{cmd}"
 | |
|     end
 | |
|     ohai(cmd, out) if ARGV.verbose?
 | |
|     out
 | |
|   end
 | |
| end
 | |
| 
 | |
| 
 | |
| class Report < Hash
 | |
| 
 | |
|   def dump
 | |
|     # Key Legend: Added (A), Copied (C), Deleted (D), Modified (M), Renamed (R)
 | |
| 
 | |
|     dump_formula_report :A, "New Formulae"
 | |
|     dump_formula_report :M, "Updated Formulae"
 | |
|     dump_formula_report :D, "Deleted Formulae"
 | |
|     dump_formula_report :R, "Renamed Formulae"
 | |
| #    dump_new_commands
 | |
| #    dump_deleted_commands
 | |
|   end
 | |
| 
 | |
|   def tapped_formula_for key
 | |
|     fetch(key, []).map do |path|
 | |
|       case path when %r{^Library/Taps/(\w+-\w+/.*)}
 | |
|         relative_path = $1
 | |
|         if valid_formula_location?(relative_path)
 | |
|           Pathname.new(relative_path)
 | |
|         end
 | |
|       end
 | |
|     end.compact
 | |
|   end
 | |
| 
 | |
|   def valid_formula_location?(relative_path)
 | |
|     ruby_file = /\A.*\.rb\Z/
 | |
|     parts = relative_path.split('/')[1..-1]
 | |
|     [
 | |
|       parts.length == 1 && parts.first =~ ruby_file,
 | |
|       parts.length == 2 && parts.first == 'Formula' && parts.last =~ ruby_file,
 | |
|       parts.length == 2 && parts.first == 'HomebrewFormula' && parts.last =~ ruby_file,
 | |
|     ].any?
 | |
|   end
 | |
| 
 | |
|   def new_tapped_formula
 | |
|     tapped_formula_for :A
 | |
|   end
 | |
| 
 | |
|   def removed_tapped_formula
 | |
|     tapped_formula_for :D
 | |
|   end
 | |
| 
 | |
|   def select_formula key
 | |
|     fetch(key, []).map do |path|
 | |
|       case path when %r{^Library/Formula}
 | |
|         File.basename(path, ".rb")
 | |
|       when %r{^Library/Taps/(\w+)-(\w+)/(.*)\.rb}
 | |
|         "#$1/#$2/#{File.basename(path, '.rb')}"
 | |
|       end
 | |
|     end.compact.sort
 | |
|   end
 | |
| 
 | |
|   def dump_formula_report key, title
 | |
|     formula = select_formula(key)
 | |
|     unless formula.empty?
 | |
|       ohai title
 | |
|       puts_columns formula.uniq
 | |
|     end
 | |
|   end
 | |
| 
 | |
| end
 | 
