diff --git a/Library/Homebrew/brew.rb b/Library/Homebrew/brew.rb index 527dec79e7bca1d8232ae5e1bf307916b8dce8d2..162fae41e11349866b2df455b10d7c6b0e3a3cfb 100644 --- a/Library/Homebrew/brew.rb +++ b/Library/Homebrew/brew.rb @@ -14,6 +14,8 @@ end require_relative "global" +require "update_migrator" + begin trap("INT", std_trap) # restore default CTRL-C handler @@ -76,7 +78,7 @@ begin end # Migrate LinkedKegs/PinnedKegs if update didn't already do so - migrate_legacy_keg_symlinks_if_necessary + UpdateMigrator.migrate_legacy_keg_symlinks_if_necessary # Uninstall old brew-cask if it's still around; we just use the tap now. if cmd == "cask" && (HOMEBREW_CELLAR/"brew-cask").exist? diff --git a/Library/Homebrew/cask/lib/hbc/download.rb b/Library/Homebrew/cask/lib/hbc/download.rb index d16e0ebe5dc08104322f4bd7250ed1becc412057..1429611783531a391062b9bd9da929be562e1e06 100644 --- a/Library/Homebrew/cask/lib/hbc/download.rb +++ b/Library/Homebrew/cask/lib/hbc/download.rb @@ -1,4 +1,5 @@ require "fileutils" +require "hbc/cache" require "hbc/quarantine" require "hbc/verify" @@ -19,11 +20,6 @@ module Hbc downloaded_path end - private - - attr_reader :force - attr_accessor :downloaded_path - def downloader @downloader ||= begin strategy = DownloadStrategyDetector.detect(cask.url.to_s, cask.url.using) @@ -31,6 +27,11 @@ module Hbc end end + private + + attr_reader :force + attr_accessor :downloaded_path + def clear_cache downloader.clear_cache if force || cask.version.latest? end @@ -39,7 +40,9 @@ module Hbc downloader.fetch @downloaded_path = downloader.cached_location rescue StandardError => e - raise CaskError, "Download failed on Cask '#{cask}' with message: #{e}" + error = CaskError.new("Download failed on Cask '#{cask}' with message: #{e}") + error.set_backtrace e.backtrace + raise error end def quarantine diff --git a/Library/Homebrew/cleanup.rb b/Library/Homebrew/cleanup.rb index 3d977cef897a638974cafdc38fc5d2c76def129d..d89ecf20d7570038939c79f82ff922c3d1a29812 100644 --- a/Library/Homebrew/cleanup.rb +++ b/Library/Homebrew/cleanup.rb @@ -48,8 +48,7 @@ module CleanupRefinement end def stale?(scrub = false) - return false unless file? - + return false unless resolved_path.file? stale_formula?(scrub) || stale_cask?(scrub) end @@ -209,6 +208,20 @@ module Homebrew end end + def cleanup_unreferenced_downloads + return if dry_run? + return unless (cache/"downloads").directory? + + # We can't use `.reject(&:incomplete?) here due to the refinement scope. + downloads = (cache/"downloads").children.reject { |path| path.incomplete? } # rubocop:disable Style/SymbolProc + referenced_downloads = [cache, cache/"Cask"].select(&:directory?) + .flat_map(&:children) + .select(&:symlink?) + .map(&:resolved_path) + + (downloads - referenced_downloads).each(&:unlink) + end + def cleanup_cache(entries = nil) entries ||= [cache, cache/"Cask"].select(&:directory?).flat_map(&:children) @@ -217,7 +230,7 @@ module Homebrew next cleanup_path(path) { FileUtils.rm_rf path } if path.nested_cache? if path.prune?(days) - if path.file? + if path.file? || path.symlink? cleanup_path(path) { path.unlink } elsif path.directory? && path.to_s.include?("--") cleanup_path(path) { FileUtils.rm_rf path } @@ -227,6 +240,8 @@ module Homebrew next cleanup_path(path) { path.unlink } if path.stale?(scrub?) end + + cleanup_unreferenced_downloads end def cleanup_path(path) diff --git a/Library/Homebrew/cmd/update-report.rb b/Library/Homebrew/cmd/update-report.rb index 13571da01ef4e707b284212acc687d4dcdc3c791..009180407b07dcfcf55afa49a468fad2c1c0a293 100644 --- a/Library/Homebrew/cmd/update-report.rb +++ b/Library/Homebrew/cmd/update-report.rb @@ -7,6 +7,7 @@ require "migrator" require "formulary" require "descriptions" require "cleanup" +require "update_migrator" module Homebrew module_function @@ -110,9 +111,10 @@ module Homebrew updated = true end - migrate_legacy_cache_if_necessary - migrate_cache_entries_to_double_dashes(initial_version) - migrate_legacy_keg_symlinks_if_necessary + UpdateMigrator.migrate_legacy_cache_if_necessary + UpdateMigrator.migrate_cache_entries_to_double_dashes(initial_version) + UpdateMigrator.migrate_cache_entries_to_symlinks(initial_version) + UpdateMigrator.migrate_legacy_keg_symlinks_if_necessary if !updated if !ARGV.include?("--preinstall") && !ENV["HOMEBREW_UPDATE_FAILED"] @@ -138,7 +140,7 @@ module Homebrew # This should always be the last thing to run (but skip on auto-update). if !ARGV.include?("--preinstall") || ENV["HOMEBREW_ENABLE_AUTO_UPDATE_MIGRATION"] - migrate_legacy_repository_if_necessary + UpdateMigrator.migrate_legacy_repository_if_necessary end end @@ -156,222 +158,6 @@ module Homebrew ENV["HOMEBREW_UPDATE_AFTER_HOMEBREW_HOMEBREW_CORE"] = revision end - def migrate_legacy_cache_if_necessary - legacy_cache = Pathname.new "/Library/Caches/Homebrew" - return if HOMEBREW_CACHE.to_s == legacy_cache.to_s - return unless legacy_cache.directory? - return unless legacy_cache.readable_real? - - migration_attempted_file = legacy_cache/".migration_attempted" - return if migration_attempted_file.exist? - - return unless legacy_cache.writable_real? - FileUtils.touch migration_attempted_file - - # This directory could have been compromised if it's world-writable/ - # a symlink/owned by another user so don't copy files in those cases. - world_writable = legacy_cache.stat.mode & 0777 == 0777 - return if world_writable - return if legacy_cache.symlink? - return if !legacy_cache.owned? && legacy_cache.lstat.uid.nonzero? - - ohai "Migrating #{legacy_cache} to #{HOMEBREW_CACHE}..." - HOMEBREW_CACHE.mkpath - legacy_cache.cd do - legacy_cache.entries.each do |f| - next if [".", "..", ".migration_attempted"].include? f.to_s - begin - FileUtils.cp_r f, HOMEBREW_CACHE - rescue - @migration_failed ||= true - end - end - end - - if @migration_failed - opoo <<~EOS - Failed to migrate #{legacy_cache} to - #{HOMEBREW_CACHE}. Please do so manually. - EOS - else - ohai "Deleting #{legacy_cache}..." - FileUtils.rm_rf legacy_cache - if legacy_cache.exist? - FileUtils.touch migration_attempted_file - opoo <<~EOS - Failed to delete #{legacy_cache}. - Please do so manually. - EOS - end - end - end - - def migrate_cache_entries_to_double_dashes(initial_version) - return if initial_version && initial_version > "1.7.1" - - return if ENV.key?("HOMEBREW_DISABLE_LOAD_FORMULA") - - ohai "Migrating cache entries..." - - Formula.each do |formula| - specs = [*formula.stable, *formula.devel, *formula.head] - - resources = [*formula.bottle&.resource] + specs.flat_map do |spec| - [ - spec, - *spec.resources.values, - *spec.patches.select(&:external?).map(&:resource), - ] - end - - resources.each do |resource| - downloader = resource.downloader - - name = resource.download_name - version = resource.version - - new_location = downloader.cached_location - extname = new_location.extname - old_location = downloader.cached_location.dirname/"#{name}-#{version}#{extname}" - - next unless old_location.file? - - if new_location.exist? - begin - FileUtils.rm_rf old_location - rescue Errno::EACCES - opoo "Could not remove #{old_location}, please do so manually." - end - else - begin - FileUtils.mv old_location, new_location - rescue Errno::EACCES - opoo "Could not move #{old_location} to #{new_location}, please do so manually." - end - end - end - end - end - - def migrate_legacy_repository_if_necessary - return unless HOMEBREW_PREFIX.to_s == "/usr/local" - return unless HOMEBREW_REPOSITORY.to_s == "/usr/local" - - ohai "Migrating HOMEBREW_REPOSITORY (please wait)..." - - unless HOMEBREW_PREFIX.writable_real? - ofail <<~EOS - #{HOMEBREW_PREFIX} is not writable. - - You should change the ownership and permissions of #{HOMEBREW_PREFIX} - temporarily back to your user account so we can complete the Homebrew - repository migration: - sudo chown -R $(whoami) #{HOMEBREW_PREFIX} - EOS - return - end - - new_homebrew_repository = Pathname.new "/usr/local/Homebrew" - new_homebrew_repository.rmdir_if_possible - if new_homebrew_repository.exist? - ofail <<~EOS - #{new_homebrew_repository} already exists. - Please remove it manually or uninstall and reinstall Homebrew into a new - location as the migration cannot be done automatically. - EOS - return - end - new_homebrew_repository.mkpath - - repo_files = HOMEBREW_REPOSITORY.cd do - Utils.popen_read("git ls-files").lines.map(&:chomp) - end - - unless Utils.popen_read("git status --untracked-files=all --porcelain").empty? - HOMEBREW_REPOSITORY.cd do - quiet_system "git", "merge", "--abort" - quiet_system "git", "rebase", "--abort" - quiet_system "git", "reset", "--mixed" - safe_system "git", "-c", "user.email=brew-update@localhost", - "-c", "user.name=brew update", - "stash", "save", "--include-untracked" - end - stashed = true - end - - FileUtils.cp_r "#{HOMEBREW_REPOSITORY}/.git", "#{new_homebrew_repository}/.git" - new_homebrew_repository.cd do - safe_system "git", "checkout", "--force", "." - safe_system "git", "stash", "pop" if stashed - end - - if (HOMEBREW_REPOSITORY/"Library/Locks").exist? - FileUtils.cp_r "#{HOMEBREW_REPOSITORY}/Library/Locks", "#{new_homebrew_repository}/Library/Locks" - end - - if (HOMEBREW_REPOSITORY/"Library/Taps").exist? - FileUtils.cp_r "#{HOMEBREW_REPOSITORY}/Library/Taps", "#{new_homebrew_repository}/Library/Taps" - end - - unremovable_paths = [] - extra_remove_paths = [".git", "Library/Locks", "Library/Taps", - "Library/Homebrew/cask", "Library/Homebrew/test"] - (repo_files + extra_remove_paths).each do |file| - path = Pathname.new "#{HOMEBREW_REPOSITORY}/#{file}" - begin - FileUtils.rm_rf path - rescue Errno::EACCES - unremovable_paths << path - end - quiet_system "rmdir", "-p", path.parent if path.parent.exist? - end - - unless unremovable_paths.empty? - ofail <<~EOS - Could not remove old HOMEBREW_REPOSITORY paths! - Please do this manually with: - sudo rm -rf #{unremovable_paths.join " "} - EOS - end - - (Keg::ALL_TOP_LEVEL_DIRECTORIES + ["Cellar"]).each do |dir| - FileUtils.mkdir_p "#{HOMEBREW_PREFIX}/#{dir}" - end - - src = Pathname.new("#{new_homebrew_repository}/bin/brew") - dst = Pathname.new("#{HOMEBREW_PREFIX}/bin/brew") - begin - FileUtils.ln_s(src.relative_path_from(dst.parent), dst) - rescue Errno::EACCES, Errno::ENOENT - ofail <<~EOS - Could not create symlink at #{dst}! - Please do this manually with: - sudo ln -sf #{src} #{dst} - sudo chown $(whoami) #{dst} - EOS - end - - link_completions_manpages_and_docs(new_homebrew_repository) - - ohai "Migrated HOMEBREW_REPOSITORY to #{new_homebrew_repository}!" - puts <<~EOS - Homebrew no longer needs to have ownership of /usr/local. If you wish you can - return /usr/local to its default ownership with: - sudo chown root:wheel #{HOMEBREW_PREFIX} - EOS - rescue => e - ofail <<~EOS - #{Tty.bold}Failed to migrate HOMEBREW_REPOSITORY to #{new_homebrew_repository}!#{Tty.reset} - The error was: - #{e} - Please try to resolve this error yourself and then run `brew update` again to - complete the migration. If you need help please +1 an existing error or comment - with your new error in issue: - #{Formatter.url("https://github.com/Homebrew/brew/issues/987")} - EOS - $stderr.puts e.backtrace - end - def link_completions_manpages_and_docs(repository = HOMEBREW_REPOSITORY) command = "brew update" Utils::Link.link_completions(repository, command) diff --git a/Library/Homebrew/dev-cmd/tests.rb b/Library/Homebrew/dev-cmd/tests.rb index f2310202bb4ffbc0d1cdcbb0fd09f8df23554b9a..bf721db0c60b3472a878ff6f6fbf1c053990ac27 100644 --- a/Library/Homebrew/dev-cmd/tests.rb +++ b/Library/Homebrew/dev-cmd/tests.rb @@ -37,6 +37,7 @@ module Homebrew ENV.delete("HOMEBREW_COLOR") ENV.delete("HOMEBREW_NO_COLOR") ENV.delete("HOMEBREW_VERBOSE") + ENV.delete("HOMEBREW_DEBUG") ENV.delete("VERBOSE") ENV.delete("HOMEBREW_CASK_OPTS") ENV.delete("HOMEBREW_TEMP") diff --git a/Library/Homebrew/download_strategy.rb b/Library/Homebrew/download_strategy.rb index 256af3b32a5356f2792f1cac12413ab95858c5f2..6319fdf4f9324ee530f58a3bf3d218642f5b55bf 100644 --- a/Library/Homebrew/download_strategy.rb +++ b/Library/Homebrew/download_strategy.rb @@ -2,6 +2,8 @@ require "json" require "rexml/document" require "time" require "unpack_strategy" +require "lazy_object" +require "cgi" class AbstractDownloadStrategy extend Forwardable @@ -9,12 +11,12 @@ class AbstractDownloadStrategy module Pourable def stage - ohai "Pouring #{cached_location.basename}" + ohai "Pouring #{basename}" super end end - attr_reader :cached_location + attr_reader :cache, :cached_location, :url attr_reader :meta, :name, :version, :shutup private :meta, :name, :version, :shutup @@ -51,7 +53,7 @@ class AbstractDownloadStrategy UnpackStrategy.detect(cached_location, extension_only: true, ref_type: @ref_type, ref: @ref) - .extract_nestedly(basename: basename_without_params, + .extract_nestedly(basename: basename, extension_only: true, verbose: ARGV.verbose? && !shutup) chdir @@ -82,11 +84,8 @@ class AbstractDownloadStrategy rm_rf(cached_location) end - def basename_without_params - return unless @url - - # Strip any ?thing=wad out of .c?thing=wad style extensions - File.basename(@url)[/[^?]+/] + def basename + cached_location.basename end private @@ -122,7 +121,7 @@ class VCSDownloadStrategy < AbstractDownloadStrategy end def fetch - ohai "Cloning #{@url}" + ohai "Cloning #{url}" if cached_location.exist? && repo_valid? puts "Updating #{cached_location}" @@ -193,30 +192,79 @@ class AbstractFileDownloadStrategy < AbstractDownloadStrategy def initialize(url, name, version, **meta) super - @cached_location = @cache/"#{name}--#{version}#{ext}" @temporary_path = Pathname.new("#{cached_location}.incomplete") end + def symlink_location + return @symlink_location if defined?(@symlink_location) + ext = Pathname(parse_basename(url)).extname + @symlink_location = @cache/"#{name}--#{version}#{ext}" + end + + def cached_location + return @cached_location if defined?(@cached_location) + + url_sha256 = Digest::SHA256.hexdigest(url) + downloads = Pathname.glob(HOMEBREW_CACHE/"downloads/#{url_sha256}--*") + .reject { |path| path.extname.end_with?(".incomplete") } + + @cached_location = if downloads.count == 1 + downloads.first + else + HOMEBREW_CACHE/"downloads/#{url_sha256}--#{resolved_basename}" + end + end + + def basename + cached_location.basename.sub(/^[\da-f]{64}\-\-/, "") + end + private - def ext - uri_path = if URI::DEFAULT_PARSER.make_regexp =~ @url - uri = URI(@url) + def resolved_url + resolved_url, = resolved_url_and_basename + resolved_url + end + + def resolved_basename + _, resolved_basename = resolved_url_and_basename + resolved_basename + end + + def resolved_url_and_basename + return @resolved_url_and_basename if defined?(@resolved_url_and_basename) + @resolved_url_and_basename = [url, parse_basename(url)] + end + + def parse_basename(url) + uri_path = if URI::DEFAULT_PARSER.make_regexp =~ url + uri = URI(url) + + if uri.query + query_params = CGI.parse(uri.query) + query_params["response-content-disposition"].each do |param| + query_basename = param[/attachment;\s*filename=(["']?)(.+)\1/i, 2] + return query_basename if query_basename + end + end + uri.query ? "#{uri.path}?#{uri.query}" : uri.path else - @url + url end + uri_path = URI.decode_www_form_component(uri_path) + # We need a Pathname because we've monkeypatched extname to support double # extensions (e.g. tar.gz). - # We can't use basename_without_params, because given a URL like - # https://example.com/download.php?file=foo-1.0.tar.gz - # the extension we want is ".tar.gz", not ".php". + # Given a URL like https://example.com/download.php?file=foo-1.0.tar.gz + # the basename we want is "foo-1.0.tar.gz", not "download.php". Pathname.new(uri_path).ascend do |path| ext = path.extname[/[^?&]+/] - return ext if ext + return path.basename.to_s[/[^?&]+#{Regexp.escape(ext)}/] if ext end - nil + + File.basename(uri_path) end end @@ -229,23 +277,35 @@ class CurlDownloadStrategy < AbstractFileDownloadStrategy end def fetch - ohai "Downloading #{@url}" + urls = [url, *mirrors] - if cached_location.exist? - puts "Already downloaded: #{cached_location}" - else - begin - _fetch - rescue ErrorDuringExecution - raise CurlDownloadStrategyError, @url + begin + url = urls.shift + + ohai "Downloading #{url}" + + if cached_location.exist? + puts "Already downloaded: #{cached_location}" + else + begin + resolved_url, = resolve_url_and_basename(url) + + _fetch(url: url, resolved_url: resolved_url) + rescue ErrorDuringExecution + raise CurlDownloadStrategyError, url + end + ignore_interrupts do + temporary_path.rename(cached_location) + symlink_location.dirname.mkpath + end end - ignore_interrupts { temporary_path.rename(cached_location) } + + FileUtils.ln_s cached_location.relative_path_from(symlink_location.dirname), symlink_location, force: true + rescue CurlDownloadStrategyError + raise if urls.empty? + puts "Trying a mirror..." + retry end - rescue CurlDownloadStrategyError - raise if mirrors.empty? - puts "Trying a mirror..." - @url = mirrors.shift - retry end def clear_cache @@ -255,18 +315,52 @@ class CurlDownloadStrategy < AbstractFileDownloadStrategy private - # Private method, can be overridden if needed. - def _fetch - url = @url + def resolved_url_and_basename + return @resolved_url_and_basename if defined?(@resolved_url_and_basename) + @resolved_url_and_basename = resolve_url_and_basename(url) + end + def resolve_url_and_basename(url) if ENV["HOMEBREW_ARTIFACT_DOMAIN"] url = url.sub(%r{^((ht|f)tps?://)?}, ENV["HOMEBREW_ARTIFACT_DOMAIN"].chomp("/") + "/") - ohai "Downloading from #{url}" end + out, _, status= curl_output("--location", "--silent", "--head", url.to_s) + + lines = status.success? ? out.lines.map(&:chomp) : [] + + locations = lines.map { |line| line[/^Location:\s*(.*)$/i, 1] } + .compact + + redirect_url = locations.reduce(url) do |current_url, location| + if location.start_with?("/") + uri = URI(current_url) + "#{uri.scheme}://#{uri.host}#{location}" + else + location + end + end + + filenames = lines.map { |line| line[/^Content\-Disposition:\s*attachment;\s*filename=(["']?)(.+)\1$/i, 2] } + .compact + + basename = filenames.last || parse_basename(redirect_url) + + [redirect_url, basename] + end + + def _fetch(url:, resolved_url:) temporary_path.dirname.mkpath - curl_download resolved_url(url), to: temporary_path + ohai "Downloading from #{resolved_url}" if url != resolved_url + + if ENV["HOMEBREW_NO_INSECURE_REDIRECT"] && + url.start_with?("https://") && !resolved_url.start_with?("https://") + $stderr.puts "HTTPS to HTTP redirect detected & HOMEBREW_NO_INSECURE_REDIRECT is set." + raise CurlDownloadStrategyError, url + end + + curl_download resolved_url, to: temporary_path end # Curl options to be always passed to curl, @@ -291,27 +385,6 @@ class CurlDownloadStrategy < AbstractFileDownloadStrategy {} end - def resolved_url(url) - redirect_url, _, status = curl_output( - "--silent", "--head", - "--write-out", "%{redirect_url}", - "--output", "/dev/null", - url.to_s - ) - - return url unless status.success? - return url if redirect_url.empty? - - ohai "Downloading from #{redirect_url}" - if ENV["HOMEBREW_NO_INSECURE_REDIRECT"] && - url.start_with?("https://") && !redirect_url.start_with?("https://") - puts "HTTPS to HTTP redirect detected & HOMEBREW_NO_INSECURE_REDIRECT is set." - raise CurlDownloadStrategyError, url - end - - redirect_url - end - def curl_output(*args, **options) super(*_curl_args, *args, **_curl_opts, **options) end @@ -324,23 +397,30 @@ end # Detect and download from Apache Mirror class CurlApacheMirrorDownloadStrategy < CurlDownloadStrategy - def apache_mirrors - mirrors, = curl_output("--silent", "--location", "#{@url}&asjson=1") - JSON.parse(mirrors) + def mirrors + return @combined_mirrors if defined?(@combined_mirrors) + + backup_mirrors = apache_mirrors.fetch("backup", []) + .map { |mirror| "#{mirror}#{apache_mirrors["path_info"]}" } + + @combined_mirrors = [*@mirrors, *backup_mirrors] end - def _fetch - return super if @tried_apache_mirror - @tried_apache_mirror = true + private - mirrors = apache_mirrors - path_info = mirrors.fetch("path_info") - @url = mirrors.fetch("preferred") + path_info - @mirrors |= %W[https://archive.apache.org/dist/#{path_info}] + def resolved_url_and_basename + return @resolved_url_and_basename if defined?(@resolved_url_and_basename) + @resolved_url_and_basename = [ + "#{apache_mirrors["preferred"]}#{apache_mirrors["path_info"]}", + File.basename(apache_mirrors["path_info"]), + ] + end - ohai "Best Mirror #{@url}" - super - rescue IndexError, JSON::ParserError + def apache_mirrors + return @apache_mirrors if defined?(@apache_mirrors) + json, = curl_output("--silent", "--location", "#{url}&asjson=1") + @apache_mirrors = JSON.parse(json) + rescue JSON::ParserError raise CurlDownloadStrategyError, "Couldn't determine mirror, try again later." end end @@ -348,12 +428,14 @@ end # Download via an HTTP POST. # Query parameters on the URL are converted into POST parameters class CurlPostDownloadStrategy < CurlDownloadStrategy - def _fetch + private + + def _fetch(url:, resolved_url:) args = if meta.key?(:data) escape_data = ->(d) { ["-d", URI.encode_www_form([d])] } - [@url, *meta[:data].flat_map(&escape_data)] + [url, *meta[:data].flat_map(&escape_data)] else - url, query = @url.split("?", 2) + url, query = url.split("?", 2) query.nil? ? [url, "-X", "POST"] : [url, "-d", query] end @@ -366,7 +448,7 @@ end class NoUnzipCurlDownloadStrategy < CurlDownloadStrategy def stage UnpackStrategy::Uncompressed.new(cached_location) - .extract(basename: basename_without_params, + .extract(basename: basename, verbose: ARGV.verbose? && !shutup) end end @@ -386,10 +468,10 @@ end # because it lets you use a private S3 bucket as a repo for internal # distribution. (It will work for public buckets as well.) class S3DownloadStrategy < CurlDownloadStrategy - def _fetch - if @url !~ %r{^https?://([^.].*)\.s3\.amazonaws\.com/(.+)$} && - @url !~ %r{^s3://([^.].*?)/(.+)$} - raise "Bad S3 URL: " + @url + def _fetch(url:, resolved_url:) + if url !~ %r{^https?://([^.].*)\.s3\.amazonaws\.com/(.+)$} && + url !~ %r{^s3://([^.].*?)/(.+)$} + raise "Bad S3 URL: " + url end bucket = Regexp.last_match(1) key = Regexp.last_match(2) @@ -402,7 +484,7 @@ class S3DownloadStrategy < CurlDownloadStrategy s3url = signer.presigned_url :get_object, bucket: bucket, key: key rescue Aws::Sigv4::Errors::MissingCredentialsError ohai "AWS credentials missing, trying public URL instead." - s3url = @url + s3url = url end curl_download s3url, to: temporary_path @@ -428,24 +510,23 @@ class GitHubPrivateRepositoryDownloadStrategy < CurlDownloadStrategy end def parse_url_pattern - url_pattern = %r{https://github.com/([^/]+)/([^/]+)/(\S+)} - unless @url =~ url_pattern + unless match = url.match(%r{https://github.com/([^/]+)/([^/]+)/(\S+)}) raise CurlDownloadStrategyError, "Invalid url pattern for GitHub Repository." end - _, @owner, @repo, @filepath = *@url.match(url_pattern) + _, @owner, @repo, @filepath = *match end def download_url "https://#{@github_token}@github.com/#{@owner}/#{@repo}/#{@filepath}" end - def _fetch + private + + def _fetch(url:, resolved_url:) curl_download download_url, to: temporary_path end - private - def set_github_token @github_token = ENV["HOMEBREW_GITHUB_API_TOKEN"] unless @github_token @@ -486,14 +567,14 @@ class GitHubPrivateRepositoryReleaseDownloadStrategy < GitHubPrivateRepositoryDo "https://#{@github_token}@api.github.com/repos/#{@owner}/#{@repo}/releases/assets/#{asset_id}" end - def _fetch + private + + def _fetch(url:, resolved_url:) # HTTP request header `Accept: application/octet-stream` is required. # Without this, the GitHub API will respond with metadata, not binary. curl_download download_url, "--header", "Accept: application/octet-stream", to: temporary_path end - private - def asset_id @asset_id ||= resolve_asset_id end diff --git a/Library/Homebrew/extend/pathname.rb b/Library/Homebrew/extend/pathname.rb index 51bb6eec79aacfa992c75a274777d4ed72f787db..d4820c7a3d122ff26c7066c641c0591794548156 100644 --- a/Library/Homebrew/extend/pathname.rb +++ b/Library/Homebrew/extend/pathname.rb @@ -24,6 +24,12 @@ module DiskUsageExtension private def compute_disk_usage + if symlink? && !exist? + @file_count = 1 + @disk_usage = 0 + return + end + path = if symlink? resolved_path else diff --git a/Library/Homebrew/test/cask/cli/reinstall_spec.rb b/Library/Homebrew/test/cask/cli/reinstall_spec.rb index f2d1322ad5bab55942df86285a60196126de0514..ed8061fb6dddba45d324c80c3222e1c2db828402 100644 --- a/Library/Homebrew/test/cask/cli/reinstall_spec.rb +++ b/Library/Homebrew/test/cask/cli/reinstall_spec.rb @@ -10,7 +10,7 @@ describe Hbc::CLI::Reinstall, :cask do output = Regexp.new <<~EOS ==> Downloading file:.*caffeine.zip - Already downloaded: .*local-caffeine--1.2.3.zip + Already downloaded: .*caffeine.zip ==> Verifying checksum for Cask local-caffeine ==> Uninstalling Cask local-caffeine ==> Backing App 'Caffeine.app' up to '.*Caffeine.app'. diff --git a/Library/Homebrew/test/cmd/--cache_spec.rb b/Library/Homebrew/test/cmd/--cache_spec.rb index fb3c9cee6c4918808faba91d13edf5136aa51b8a..ad3af474eb1c743d90d1d4b243892b1b139fdf0a 100644 --- a/Library/Homebrew/test/cmd/--cache_spec.rb +++ b/Library/Homebrew/test/cmd/--cache_spec.rb @@ -8,7 +8,7 @@ describe "brew --cache", :integration_test do it "prints all cache files for a given Formula" do expect { brew "--cache", testball } - .to output(%r{#{HOMEBREW_CACHE}/testball-}).to_stdout + .to output(%r{#{HOMEBREW_CACHE}/downloads/[\da-f]{64}\-\-testball\-}).to_stdout .and not_to_output.to_stderr .and be_a_success end diff --git a/Library/Homebrew/test/cmd/fetch_spec.rb b/Library/Homebrew/test/cmd/fetch_spec.rb index da5c5ce4130b7506e878f0b86548f319026c5140..8ea15f163b0de383088a71fa71bc6a83efd8ebf4 100644 --- a/Library/Homebrew/test/cmd/fetch_spec.rb +++ b/Library/Homebrew/test/cmd/fetch_spec.rb @@ -2,10 +2,9 @@ describe "brew fetch", :integration_test do it "downloads the Formula's URL" do setup_test_formula "testball" - expect(HOMEBREW_CACHE/"testball--0.1.tbz").not_to exist - expect { brew "fetch", "testball" }.to be_a_success + expect(HOMEBREW_CACHE/"testball--0.1.tbz").to be_a_symlink expect(HOMEBREW_CACHE/"testball--0.1.tbz").to exist end end diff --git a/Library/Homebrew/test/cmd/update-report_spec.rb b/Library/Homebrew/test/cmd/update-report_spec.rb deleted file mode 100644 index a796d392f1f02e5ea307d57e5bafed8d12437492..0000000000000000000000000000000000000000 --- a/Library/Homebrew/test/cmd/update-report_spec.rb +++ /dev/null @@ -1,46 +0,0 @@ -require "cmd/update-report" - -describe "brew update-report" do - describe "::migrate_cache_entries_to_double_dashes" do - let(:formula_name) { "foo" } - let(:f) { - formula formula_name do - url "https://example.com/foo-1.2.3.tar.gz" - version "1.2.3" - end - } - let(:old_cache_file) { HOMEBREW_CACHE/"#{formula_name}-1.2.3.tar.gz" } - let(:new_cache_file) { HOMEBREW_CACHE/"#{formula_name}--1.2.3.tar.gz" } - - before(:each) do - FileUtils.touch old_cache_file - allow(Formula).to receive(:each).and_yield(f) - end - - it "moves old files to use double dashes when upgrading from <= 1.7.1" do - Homebrew.migrate_cache_entries_to_double_dashes(Version.new("1.7.1")) - - expect(old_cache_file).not_to exist - expect(new_cache_file).to exist - end - - context "when the formula name contains dashes" do - let(:formula_name) { "foo-bar" } - - it "does not introduce extra double dashes when called multiple times" do - Homebrew.migrate_cache_entries_to_double_dashes(Version.new("1.7.1")) - Homebrew.migrate_cache_entries_to_double_dashes(Version.new("1.7.1")) - - expect(old_cache_file).not_to exist - expect(new_cache_file).to exist - end - end - - it "does not move files if upgrading from > 1.7.1" do - Homebrew.migrate_cache_entries_to_double_dashes(Version.new("1.7.2")) - - expect(old_cache_file).to exist - expect(new_cache_file).not_to exist - end - end -end diff --git a/Library/Homebrew/test/download_strategies_spec.rb b/Library/Homebrew/test/download_strategies_spec.rb index bfeaee1aa9982dcf76789be8534d17cfaf468329..f654f17692f0d24262fc0039401a9d6a6d93cca5 100644 --- a/Library/Homebrew/test/download_strategies_spec.rb +++ b/Library/Homebrew/test/download_strategies_spec.rb @@ -207,16 +207,12 @@ describe S3DownloadStrategy do let(:url) { "https://bucket.s3.amazonaws.com/foo.tar.gz" } let(:version) { nil } - describe "#_fetch" do - subject { described_class.new(url, name, version)._fetch } - + describe "#fetch" do context "when given Bad S3 URL" do let(:url) { "https://example.com/foo.tar.gz" } it "raises Bad S3 URL error" do - expect { - subject._fetch - }.to raise_error(RuntimeError) + expect { subject.fetch }.to raise_error(RuntimeError, /S3/) end end end @@ -238,18 +234,19 @@ describe CurlDownloadStrategy do subject { described_class.new(url, name, version, **specs).cached_location } context "when URL ends with file" do - it { is_expected.to eq(HOMEBREW_CACHE/"foo--1.2.3.tar.gz") } + it { is_expected.to eq(HOMEBREW_CACHE/"downloads/3d1c0ae7da22be9d83fb1eb774df96b7c4da71d3cf07e1cb28555cf9a5e5af70--foo.tar.gz") } end context "when URL file is in middle" do let(:url) { "https://example.com/foo.tar.gz/from/this/mirror" } - it { is_expected.to eq(HOMEBREW_CACHE/"foo--1.2.3.tar.gz") } + it { is_expected.to eq(HOMEBREW_CACHE/"downloads/1ab61269ba52c83994510b1e28dd04167a2f2e8393a35a9c50c1f7d33fd8f619--foo.tar.gz") } end end describe "#fetch" do before(:each) do + subject.temporary_path.dirname.mkpath FileUtils.touch subject.temporary_path end @@ -330,11 +327,6 @@ describe CurlDownloadStrategy do its("cached_location.extname") { is_expected.to eq(".dmg") } end - context "with no discernible file name in it" do - let(:url) { "https://example.com/download" } - its("cached_location.basename.to_path") { is_expected.to eq("foo--1.2.3") } - end - context "with a file name trailing the first query parameter" do let(:url) { "https://example.com/download?file=cask.zip&a=1" } its("cached_location.extname") { is_expected.to eq(".zip") } @@ -380,6 +372,7 @@ describe CurlPostDownloadStrategy do describe "#fetch" do before(:each) do + subject.temporary_path.dirname.mkpath FileUtils.touch subject.temporary_path end diff --git a/Library/Homebrew/test/update_migrator_spec.rb b/Library/Homebrew/test/update_migrator_spec.rb new file mode 100644 index 0000000000000000000000000000000000000000..811d035e1ec83b485fe08d6073537522b7502d07 --- /dev/null +++ b/Library/Homebrew/test/update_migrator_spec.rb @@ -0,0 +1,84 @@ +require "update_migrator" + +describe UpdateMigrator do + describe "::migrate_cache_entries_to_double_dashes" do + let(:formula_name) { "foo" } + let(:f) { + formula formula_name do + url "https://example.com/foo-1.2.3.tar.gz" + version "1.2.3" + end + } + let(:old_cache_file) { HOMEBREW_CACHE/"#{formula_name}-1.2.3.tar.gz" } + let(:new_cache_file) { HOMEBREW_CACHE/"#{formula_name}--1.2.3.tar.gz" } + + before(:each) do + FileUtils.touch old_cache_file + allow(Formula).to receive(:each).and_yield(f) + end + + it "moves old files to use double dashes when upgrading from <= 1.7.1" do + described_class.migrate_cache_entries_to_double_dashes(Version.new("1.7.1")) + + expect(old_cache_file).not_to exist + expect(new_cache_file).to exist + end + + context "when the formula name contains dashes" do + let(:formula_name) { "foo-bar" } + + it "does not introduce extra double dashes when called multiple times" do + described_class.migrate_cache_entries_to_double_dashes(Version.new("1.7.1")) + described_class.migrate_cache_entries_to_double_dashes(Version.new("1.7.1")) + + expect(old_cache_file).not_to exist + expect(new_cache_file).to exist + end + end + + it "does not move files if upgrading from > 1.7.1" do + described_class.migrate_cache_entries_to_double_dashes(Version.new("1.7.2")) + + expect(old_cache_file).to exist + expect(new_cache_file).not_to exist + end + end + + describe "::migrate_cache_entries_to_symlinks" do + let(:formula_name) { "foo" } + let(:f) { + formula formula_name do + url "https://example.com/foo-1.2.3.tar.gz" + version "1.2.3" + end + } + let(:old_cache_file) { HOMEBREW_CACHE/"#{formula_name}--1.2.3.tar.gz" } + let(:new_cache_symlink) { HOMEBREW_CACHE/"#{formula_name}--1.2.3.tar.gz" } + let(:new_cache_file) { HOMEBREW_CACHE/"downloads/5994e3a27baa3f448a001fb071ab1f0bf25c87aebcb254d91a6d0b02f46eef86--foo-1.2.3.tar.gz" } + + before(:each) do + old_cache_file.dirname.mkpath + FileUtils.touch old_cache_file + allow(Formula).to receive(:[]).and_return(f) + end + + it "moves old files to use symlinks when upgrading from <= 1.7.2" do + described_class.migrate_cache_entries_to_symlinks(Version.new("1.7.2")) + + expect(old_cache_file).to eq(new_cache_symlink) + expect(new_cache_symlink).to be_a_symlink + expect(new_cache_symlink.readlink.to_s) + .to eq "downloads/5994e3a27baa3f448a001fb071ab1f0bf25c87aebcb254d91a6d0b02f46eef86--foo-1.2.3.tar.gz" + expect(new_cache_file).to exist + expect(new_cache_file).to be_a_file + end + + it "does not move files if upgrading from > 1.7.2" do + described_class.migrate_cache_entries_to_symlinks(Version.new("1.7.3")) + + expect(old_cache_file).to exist + expect(new_cache_file).not_to exist + expect(new_cache_symlink).not_to be_a_symlink + end + end +end diff --git a/Library/Homebrew/update_migrator.rb b/Library/Homebrew/update_migrator.rb new file mode 100644 index 0000000000000000000000000000000000000000..5a04fb6b1f692dc7b67292270eb59cce41617d49 --- /dev/null +++ b/Library/Homebrew/update_migrator.rb @@ -0,0 +1,374 @@ +require "hbc/cask_loader" +require "hbc/download" + +module UpdateMigrator + class << self + def formula_resources(formula) + specs = [formula.stable, formula.devel, formula.head].compact + + [*formula.bottle&.resource] + specs.flat_map do |spec| + [ + spec, + *spec.resources.values, + *spec.patches.select(&:external?).map(&:resource), + ] + end + end + private :formula_resources + + def parse_extname(url) + uri_path = if URI::DEFAULT_PARSER.make_regexp =~ url + uri = URI(url) + uri.query ? "#{uri.path}?#{uri.query}" : uri.path + else + url + end + + # Given a URL like https://example.com/download.php?file=foo-1.0.tar.gz + # the extension we want is ".tar.gz", not ".php". + Pathname.new(uri_path).ascend do |path| + ext = path.extname[/[^?&]+/] + return ext if ext + end + + nil + end + private :parse_extname + + def migrate_cache_entries_to_double_dashes(initial_version) + return if initial_version && initial_version > "1.7.1" + + return if ENV.key?("HOMEBREW_DISABLE_LOAD_FORMULA") + + ohai "Migrating cache entries..." + + Formula.each do |formula| + formula_resources(formula).each do |resource| + downloader = resource.downloader + + url = downloader.url + name = resource.download_name + version = resource.version + + extname = parse_extname(url) + old_location = downloader.cache/"#{name}-#{version}#{extname}" + new_location = downloader.cache/"#{name}--#{version}#{extname}" + + next unless old_location.file? + + if new_location.exist? + begin + FileUtils.rm_rf old_location + rescue Errno::EACCES + opoo "Could not remove #{old_location}, please do so manually." + end + else + begin + FileUtils.mv old_location, new_location + rescue Errno::EACCES + opoo "Could not move #{old_location} to #{new_location}, please do so manually." + end + end + end + end + end + + def migrate_cache_entries_to_symlinks(initial_version) + return if initial_version && initial_version > "1.7.2" + + return if ENV.key?("HOMEBREW_DISABLE_LOAD_FORMULA") + + ohai "Migrating cache entries..." + + cache_entries = lambda do |path| + if path.directory? + path.children + .reject(&:symlink?) + .select(&:file?) + .map { |child| child.basename.to_s.sub(/\-\-.*/, "") } + .uniq + else + [] + end + end + + load_formula = lambda do |formula| + begin + Formula[formula] + rescue FormulaUnavailableError + nil + end + end + + load_cask = lambda do |cask| + begin + Hbc::CaskLoader.load(cask) + rescue Hbc::CaskUnavailableError + nil + end + end + + formula_downloaders = + cache_entries.call(HOMEBREW_CACHE) + .map(&load_formula) + .compact + .flat_map { |formula| formula_resources(formula) } + .map { |resource| [resource.downloader, resource.download_name, resource.version] } + + cask_downloaders = + cache_entries.call(HOMEBREW_CACHE/"Cask") + .map(&load_cask) + .compact + .map { |cask| [Hbc::Download.new(cask).downloader, cask.token, cask.version] } + + downloaders = formula_downloaders + cask_downloaders + + downloaders.each do |downloader, name, version| + next unless downloader.respond_to?(:symlink_location) + + url = downloader.url + extname = parse_extname(url) + old_location = downloader.cache/"#{name}--#{version}#{extname}" + next unless old_location.file? + + new_symlink_location = downloader.symlink_location + new_location = downloader.cached_location + + if new_location.exist? && new_symlink_location.symlink? + begin + FileUtils.rm_rf old_location unless old_location == new_symlink_location + rescue Errno::EACCES + opoo "Could not remove #{old_location}, please do so manually." + end + else + begin + new_location.dirname.mkpath + if new_location.exist? + FileUtils.rm_rf old_location + else + FileUtils.mv old_location, new_location + end + symlink_target = new_location.relative_path_from(new_symlink_location.dirname) + new_symlink_location.dirname.mkpath + FileUtils.ln_s symlink_target, new_symlink_location, force: true + rescue Errno::EACCES + opoo "Could not move #{old_location} to #{new_location}, please do so manually." + end + end + end + end + + def migrate_legacy_cache_if_necessary + legacy_cache = Pathname.new "/Library/Caches/Homebrew" + return if HOMEBREW_CACHE.to_s == legacy_cache.to_s + return unless legacy_cache.directory? + return unless legacy_cache.readable_real? + + migration_attempted_file = legacy_cache/".migration_attempted" + return if migration_attempted_file.exist? + + return unless legacy_cache.writable_real? + FileUtils.touch migration_attempted_file + + # This directory could have been compromised if it's world-writable/ + # a symlink/owned by another user so don't copy files in those cases. + world_writable = legacy_cache.stat.mode & 0777 == 0777 + return if world_writable + return if legacy_cache.symlink? + return if !legacy_cache.owned? && legacy_cache.lstat.uid.nonzero? + + ohai "Migrating #{legacy_cache} to #{HOMEBREW_CACHE}..." + HOMEBREW_CACHE.mkpath + legacy_cache.cd do + legacy_cache.entries.each do |f| + next if [".", "..", ".migration_attempted"].include? f.to_s + begin + FileUtils.cp_r f, HOMEBREW_CACHE + rescue + @migration_failed ||= true + end + end + end + + if @migration_failed + opoo <<~EOS + Failed to migrate #{legacy_cache} to + #{HOMEBREW_CACHE}. Please do so manually. + EOS + else + ohai "Deleting #{legacy_cache}..." + FileUtils.rm_rf legacy_cache + if legacy_cache.exist? + FileUtils.touch migration_attempted_file + opoo <<~EOS + Failed to delete #{legacy_cache}. + Please do so manually. + EOS + end + end + end + + def migrate_legacy_keg_symlinks_if_necessary + legacy_linked_kegs = HOMEBREW_LIBRARY/"LinkedKegs" + return unless legacy_linked_kegs.directory? + + HOMEBREW_LINKED_KEGS.mkpath unless legacy_linked_kegs.children.empty? + legacy_linked_kegs.children.each do |link| + name = link.basename.to_s + src = begin + link.realpath + rescue Errno::ENOENT + begin + (HOMEBREW_PREFIX/"opt/#{name}").realpath + rescue Errno::ENOENT + begin + Formulary.factory(name).installed_prefix + rescue + next + end + end + end + dst = HOMEBREW_LINKED_KEGS/name + dst.unlink if dst.exist? + FileUtils.ln_sf(src.relative_path_from(dst.parent), dst) + end + FileUtils.rm_rf legacy_linked_kegs + + legacy_pinned_kegs = HOMEBREW_LIBRARY/"PinnedKegs" + return unless legacy_pinned_kegs.directory? + + HOMEBREW_PINNED_KEGS.mkpath unless legacy_pinned_kegs.children.empty? + legacy_pinned_kegs.children.each do |link| + name = link.basename.to_s + src = link.realpath + dst = HOMEBREW_PINNED_KEGS/name + FileUtils.ln_sf(src.relative_path_from(dst.parent), dst) + end + FileUtils.rm_rf legacy_pinned_kegs + end + + def migrate_legacy_repository_if_necessary + return unless HOMEBREW_PREFIX.to_s == "/usr/local" + return unless HOMEBREW_REPOSITORY.to_s == "/usr/local" + + ohai "Migrating HOMEBREW_REPOSITORY (please wait)..." + + unless HOMEBREW_PREFIX.writable_real? + ofail <<~EOS + #{HOMEBREW_PREFIX} is not writable. + + You should change the ownership and permissions of #{HOMEBREW_PREFIX} + temporarily back to your user account so we can complete the Homebrew + repository migration: + sudo chown -R $(whoami) #{HOMEBREW_PREFIX} + EOS + return + end + + new_homebrew_repository = Pathname.new "/usr/local/Homebrew" + new_homebrew_repository.rmdir_if_possible + if new_homebrew_repository.exist? + ofail <<~EOS + #{new_homebrew_repository} already exists. + Please remove it manually or uninstall and reinstall Homebrew into a new + location as the migration cannot be done automatically. + EOS + return + end + new_homebrew_repository.mkpath + + repo_files = HOMEBREW_REPOSITORY.cd do + Utils.popen_read("git ls-files").lines.map(&:chomp) + end + + unless Utils.popen_read("git status --untracked-files=all --porcelain").empty? + HOMEBREW_REPOSITORY.cd do + quiet_system "git", "merge", "--abort" + quiet_system "git", "rebase", "--abort" + quiet_system "git", "reset", "--mixed" + safe_system "git", "-c", "user.email=brew-update@localhost", + "-c", "user.name=brew update", + "stash", "save", "--include-untracked" + end + stashed = true + end + + FileUtils.cp_r "#{HOMEBREW_REPOSITORY}/.git", "#{new_homebrew_repository}/.git" + new_homebrew_repository.cd do + safe_system "git", "checkout", "--force", "." + safe_system "git", "stash", "pop" if stashed + end + + if (HOMEBREW_REPOSITORY/"Library/Locks").exist? + FileUtils.cp_r "#{HOMEBREW_REPOSITORY}/Library/Locks", "#{new_homebrew_repository}/Library/Locks" + end + + if (HOMEBREW_REPOSITORY/"Library/Taps").exist? + FileUtils.cp_r "#{HOMEBREW_REPOSITORY}/Library/Taps", "#{new_homebrew_repository}/Library/Taps" + end + + unremovable_paths = [] + extra_remove_paths = [ + ".git", + "Library/Locks", + "Library/Taps", + "Library/Homebrew/cask", + "Library/Homebrew/test", + ] + (repo_files + extra_remove_paths).each do |file| + path = Pathname.new "#{HOMEBREW_REPOSITORY}/#{file}" + begin + FileUtils.rm_rf path + rescue Errno::EACCES + unremovable_paths << path + end + quiet_system "rmdir", "-p", path.parent if path.parent.exist? + end + + unless unremovable_paths.empty? + ofail <<~EOS + Could not remove old HOMEBREW_REPOSITORY paths! + Please do this manually with: + sudo rm -rf #{unremovable_paths.join " "} + EOS + end + + (Keg::ALL_TOP_LEVEL_DIRECTORIES + ["Cellar"]).each do |dir| + FileUtils.mkdir_p "#{HOMEBREW_PREFIX}/#{dir}" + end + + src = Pathname.new("#{new_homebrew_repository}/bin/brew") + dst = Pathname.new("#{HOMEBREW_PREFIX}/bin/brew") + begin + FileUtils.ln_s(src.relative_path_from(dst.parent), dst) + rescue Errno::EACCES, Errno::ENOENT + ofail <<~EOS + Could not create symlink at #{dst}! + Please do this manually with: + sudo ln -sf #{src} #{dst} + sudo chown $(whoami) #{dst} + EOS + end + + link_completions_manpages_and_docs(new_homebrew_repository) + + ohai "Migrated HOMEBREW_REPOSITORY to #{new_homebrew_repository}!" + puts <<~EOS + Homebrew no longer needs to have ownership of /usr/local. If you wish you can + return /usr/local to its default ownership with: + sudo chown root:wheel #{HOMEBREW_PREFIX} + EOS + rescue => e + ofail <<~EOS + #{Tty.bold}Failed to migrate HOMEBREW_REPOSITORY to #{new_homebrew_repository}!#{Tty.reset} + The error was: + #{e} + Please try to resolve this error yourself and then run `brew update` again to + complete the migration. If you need help please +1 an existing error or comment + with your new error in issue: + #{Formatter.url("https://github.com/Homebrew/brew/issues/987")} + EOS + $stderr.puts e.backtrace + end + end +end diff --git a/Library/Homebrew/utils.rb b/Library/Homebrew/utils.rb index 20319c4c35e901dec2e38d6d66520d3e640b45d1..b3ecb0d09157be40e8f37bcda44455389cd9ff19 100644 --- a/Library/Homebrew/utils.rb +++ b/Library/Homebrew/utils.rb @@ -498,45 +498,6 @@ def truncate_text_to_approximate_size(s, max_bytes, options = {}) out end -def migrate_legacy_keg_symlinks_if_necessary - legacy_linked_kegs = HOMEBREW_LIBRARY/"LinkedKegs" - return unless legacy_linked_kegs.directory? - - HOMEBREW_LINKED_KEGS.mkpath unless legacy_linked_kegs.children.empty? - legacy_linked_kegs.children.each do |link| - name = link.basename.to_s - src = begin - link.realpath - rescue Errno::ENOENT - begin - (HOMEBREW_PREFIX/"opt/#{name}").realpath - rescue Errno::ENOENT - begin - Formulary.factory(name).installed_prefix - rescue - next - end - end - end - dst = HOMEBREW_LINKED_KEGS/name - dst.unlink if dst.exist? - FileUtils.ln_sf(src.relative_path_from(dst.parent), dst) - end - FileUtils.rm_rf legacy_linked_kegs - - legacy_pinned_kegs = HOMEBREW_LIBRARY/"PinnedKegs" - return unless legacy_pinned_kegs.directory? - - HOMEBREW_PINNED_KEGS.mkpath unless legacy_pinned_kegs.children.empty? - legacy_pinned_kegs.children.each do |link| - name = link.basename.to_s - src = link.realpath - dst = HOMEBREW_PINNED_KEGS/name - FileUtils.ln_sf(src.relative_path_from(dst.parent), dst) - end - FileUtils.rm_rf legacy_pinned_kegs -end - # Calls the given block with the passed environment variables # added to ENV, then restores ENV afterwards. # Example: