brew/Library/Homebrew/cmd/update.rb

374 lines
10 KiB
Ruby
Raw Normal View History

require "cmd/tap"
require "formula_versions"
2015-08-09 15:09:33 +03:00
require "migrator"
require "formulary"
module Homebrew
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
# ensure GIT_CONFIG is unset as we need to operate on .git/config
ENV.delete("GIT_CONFIG")
2012-03-06 02:23:01 +00:00
cd HOMEBREW_REPOSITORY
git_init_if_necessary
2011-09-20 02:25:50 +01:00
# migrate to new directories based tap structure
migrate_taps
2012-03-06 02:23:01 +00:00
report = Report.new
2014-06-23 18:41:50 -05:00
master_updater = Updater.new(HOMEBREW_REPOSITORY)
master_updater.pull!
report.update(master_updater.report)
# rename Taps directories
# this procedure will be removed in the future if it seems unnecessasry
rename_taps_dir_if_necessary
Tap.each do |tap|
next unless tap.git?
2015-06-08 18:05:58 +08:00
tap.path.cd do
updater = Updater.new(tap.path)
2014-04-25 12:34:28 -05:00
begin
updater.pull!
2014-04-25 12:34:28 -05:00
rescue
2015-06-08 18:05:58 +08:00
onoe "Failed to update tap: #{tap}"
2014-04-25 12:34:28 -05:00
else
report.update(updater.report) do |_key, oldval, newval|
oldval.concat(newval)
end
2012-03-06 02:23:01 +00:00
end
end
2012-03-06 02:23:01 +00:00
end
# automatically tap any migrated formulae's new tap
report.select_formula(:D).each do |f|
2015-08-09 15:09:33 +03:00
next unless (dir = HOMEBREW_CELLAR/f).exist?
migration = TAP_MIGRATIONS[f]
next unless migration
tap_user, tap_repo = migration.split "/"
install_tap tap_user, tap_repo
2015-08-09 15:09:33 +03:00
# update tap for each Tab
tabs = dir.subdirs.each.map { |d| Tab.for_keg(Keg.new(d)) }
next if tabs.first.source["tap"] != "Homebrew/homebrew"
2015-08-09 15:09:33 +03:00
tabs.each { |tab| tab.source["tap"] = "#{tap_user}/homebrew-#{tap_repo}" }
tabs.each(&:write)
end if load_tap_migrations
2015-08-17 21:49:55 +03:00
load_formula_renames
report.update_renamed
# Migrate installed renamed formulae from core and taps.
report.select_formula(:R).each do |oldname, newname|
if oldname.include?("/")
user, repo, oldname = oldname.split("/", 3)
newname = newname.split("/", 3).last
else
2015-08-22 13:15:33 +08:00
user = "homebrew"
repo = "homebrew"
2015-08-09 15:09:33 +03:00
end
next unless (dir = HOMEBREW_CELLAR/oldname).directory? && !dir.subdirs.empty?
begin
migrator = Migrator.new(Formulary.factory("#{user}/#{repo}/#{newname}"))
migrator.migrate
rescue Migrator::MigratorDifferentTapsError
end
end
2012-03-06 02:23:01 +00:00
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]}."
2012-03-06 02:23:01 +00:00
report.dump
end
end
2012-03-06 02:23:01 +00:00
private
2012-03-06 02:23:01 +00:00
def git_init_if_necessary
2014-06-20 18:36:18 -05:00
if Dir[".git/*"].empty?
safe_system "git", "init"
safe_system "git", "config", "core.autocrlf", "false"
safe_system "git", "config", "remote.origin.url", "https://github.com/Homebrew/homebrew.git"
safe_system "git", "config", "remote.origin.fetch", "+refs/heads/*:refs/remotes/origin/*"
2014-06-20 18:36:18 -05:00
safe_system "git", "fetch", "origin"
safe_system "git", "reset", "--hard", "origin/master"
end
if `git remote show origin -n` =~ /Fetch URL: \S+mxcl\/homebrew/
2014-06-20 18:36:18 -05:00
safe_system "git", "remote", "set-url", "origin", "https://github.com/Homebrew/homebrew.git"
safe_system "git", "remote", "set-url", "--delete", "origin", ".*mxcl\/homebrew.*"
end
2012-03-06 02:23:01 +00:00
rescue Exception
FileUtils.rm_rf ".git"
raise
end
def rename_taps_dir_if_necessary
Dir.glob("#{HOMEBREW_LIBRARY}/Taps/*/") do |tapd|
begin
if File.directory?(tapd + "/.git")
tapd_basename = File.basename(tapd)
if tapd_basename.include?("-")
# only replace the *last* dash: yes, tap filenames suck
user, repo = tapd_basename.reverse.sub("-", "/").reverse.split("/")
FileUtils.mkdir_p("#{HOMEBREW_LIBRARY}/Taps/#{user.downcase}")
FileUtils.mv(tapd, "#{HOMEBREW_LIBRARY}/Taps/#{user.downcase}/homebrew-#{repo.downcase}")
if tapd_basename.count("-") >= 2
opoo "Homebrew changed the structure of Taps like <someuser>/<sometap>. "\
+ "So you may need to rename #{HOMEBREW_LIBRARY}/Taps/#{user.downcase}/homebrew-#{repo.downcase} manually."
end
else
opoo "Homebrew changed the structure of Taps like <someuser>/<sometap>. "\
"#{tapd} is incorrect name format. You may need to rename it like <someuser>/<sometap> manually."
end
end
rescue => ex
onoe ex.message
next # next tap directory
end
end
end
def load_tap_migrations
2015-08-10 17:43:21 +03:00
load "tap_migrations.rb"
2015-08-09 15:09:33 +03:00
rescue LoadError
false
end
def load_formula_renames
load "formula_renames.rb"
rescue LoadError
false
end
2012-03-06 02:23:01 +00:00
end
2012-03-06 02:23:01 +00:00
class Updater
2014-06-23 18:41:50 -05:00
attr_reader :initial_revision, :current_revision, :repository
def initialize(repository)
@repository = repository
@stashed = false
2014-06-23 18:41:50 -05:00
end
def pull!(options = {})
quiet = []
quiet << "--quiet" unless ARGV.verbose?
unless system "git", "diff", "--quiet"
unless options[:silent]
puts "Stashing your changes:"
system "git", "status", "--short", "--untracked-files"
end
safe_system "git", "stash", "save", "--include-untracked", *quiet
@stashed = true
end
@initial_branch = `git symbolic-ref --short HEAD`.chomp
if @initial_branch != "master" && !@initial_branch.empty?
safe_system "git", "checkout", "master", *quiet
end
2012-03-06 02:23:01 +00:00
@initial_revision = read_current_revision
2012-03-06 02:23:01 +00:00
# ensure we don't munge line endings on checkout
2014-06-20 18:36:18 -05:00
safe_system "git", "config", "core.autocrlf", "false"
2015-07-09 23:50:55 +01:00
args = ["pull"]
2012-03-06 02:23:01 +00:00
args << "--rebase" if ARGV.include? "--rebase"
2015-07-09 23:50:55 +01:00
args += quiet
args << "origin"
2012-03-06 02:23:01 +00:00
# the refspec ensures that 'origin/master' gets updated
args << "refs/heads/master:refs/remotes/origin/master"
reset_on_interrupt { safe_system "git", *args }
if @initial_branch != "master" && !@initial_branch.empty?
safe_system "git", "checkout", @initial_branch, *quiet
end
if @stashed
safe_system "git", "stash", "pop", *quiet
unless options[:silent]
puts "Restored your changes:"
system "git", "status", "--short", "--untracked-files"
end
@stashed = false
end
2012-03-06 02:23:01 +00:00
@current_revision = read_current_revision
end
def reset_on_interrupt
ignore_interrupts { yield }
ensure
if $?.signaled? && $?.termsig == 2 # SIGINT
safe_system "git", "checkout", @initial_branch
safe_system "git", "reset", "--hard", @initial_revision
safe_system "git", "stash", "pop" if @stashed
end
end
def report
map = Hash.new { |h, k| h[k] = [] }
2012-03-06 02:23:01 +00:00
if initial_revision && initial_revision != current_revision
2014-07-26 20:11:53 -05:00
diff.each_line do |line|
status, *paths = line.split
src = paths.first
dst = paths.last
next unless File.extname(dst) == ".rb"
next unless paths.any? { |p| File.dirname(p) == formula_directory }
case status
when "A", "D"
map[status.to_sym] << repository.join(src)
when "M"
file = repository.join(src)
begin
formula = Formulary.factory(file)
new_version = formula.pkg_version
old_version = FormulaVersions.new(formula).formula_at_revision(@initial_revision, &:pkg_version)
next if new_version == old_version
rescue FormulaUnavailableError, *FormulaVersions::IGNORED_EXCEPTIONS => e
onoe e if ARGV.homebrew_developer?
end
map[:M] << file
when /^R\d{0,3}/
map[:D] << repository.join(src) if File.dirname(src) == formula_directory
map[:A] << repository.join(dst) if File.dirname(dst) == formula_directory
end
2012-03-06 02:23:01 +00:00
end
end
2012-03-06 02:23:01 +00:00
map
end
private
2010-07-23 17:37:03 -07:00
def formula_directory
if repository == HOMEBREW_REPOSITORY
"Library/Formula"
elsif repository.join("Formula").directory?
"Formula"
elsif repository.join("HomebrewFormula").directory?
"HomebrewFormula"
else
"."
end
end
2012-03-06 02:23:01 +00:00
def read_current_revision
`git rev-parse -q --verify HEAD`.chomp
end
2014-07-26 20:11:53 -05:00
def diff
Utils.popen_read(
"git", "diff-tree", "-r", "--name-status", "--diff-filter=AMDR",
"-M85%", initial_revision, current_revision
)
2014-07-26 20:11:53 -05:00
end
2012-03-06 02:23:01 +00:00
def `(cmd)
2014-06-23 18:50:55 -05:00
out = super
unless $?.success?
$stderr.puts(out) unless out.empty?
raise ErrorDuringExecution.new(cmd)
2012-03-06 02:23:01 +00:00
end
ohai(cmd, out) if ARGV.verbose?
out
end
2012-03-06 02:23:01 +00:00
end
class Report
def initialize
@hash = {}
end
def fetch(*args, &block)
@hash.fetch(*args, &block)
end
def update(*args, &block)
@hash.update(*args, &block)
end
def empty?
@hash.empty?
end
2012-03-06 02:23:01 +00:00
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"
2015-08-17 21:49:55 +03:00
dump_formula_report :R, "Renamed Formulae"
dump_formula_report :D, "Deleted Formulae"
end
2015-08-17 21:49:55 +03:00
def update_renamed
renamed_formulae = []
2015-08-17 21:49:55 +03:00
fetch(:D, []).each do |path|
case path.to_s
when HOMEBREW_TAP_PATH_REGEX
2015-08-17 21:49:55 +03:00
user = $1
repo = $2.sub("homebrew-", "")
oldname = path.basename(".rb").to_s
next unless newname = Tap.new(user, repo).formula_renames[oldname]
else
oldname = path.basename(".rb").to_s
next unless newname = FORMULA_RENAMES[oldname]
end
if fetch(:A, []).include?(newpath = path.dirname.join("#{newname}.rb"))
renamed_formulae << [path, newpath]
2015-08-17 21:49:55 +03:00
end
end
unless renamed_formulae.empty?
@hash[:A] -= renamed_formulae.map(&:last) if @hash[:A]
@hash[:D] -= renamed_formulae.map(&:first) if @hash[:D]
@hash[:R] = renamed_formulae
end
2015-08-17 21:49:55 +03:00
end
def select_formula(key)
fetch(key, []).map do |path, newpath|
if path.to_s =~ HOMEBREW_TAP_PATH_REGEX
tap = "#{$1}/#{$2.sub("homebrew-", "")}"
if newpath
["#{tap}/#{path.basename(".rb")}", "#{tap}/#{newpath.basename(".rb")}"]
else
"#{tap}/#{path.basename(".rb")}"
end
elsif newpath
["#{path.basename(".rb")}", "#{newpath.basename(".rb")}"]
else
path.basename(".rb").to_s
2012-03-06 02:23:01 +00:00
end
end.sort
end
def dump_formula_report(key, title)
2012-03-06 02:23:01 +00:00
formula = select_formula(key)
2015-08-17 21:49:55 +03:00
formula.map! { |oldname, newname| "#{oldname} -> #{newname}" } if key == :R
2012-03-06 02:23:01 +00:00
unless formula.empty?
ohai title
puts_columns formula
end
end
2009-09-17 18:40:21 +01:00
end