# typed: false # frozen_string_literal: true require "formula" require "fetch" require "cli/parser" require "cask/download" module Homebrew extend T::Sig extend Fetch module_function sig { returns(CLI::Parser) } def fetch_args Homebrew::CLI::Parser.new do usage_banner <<~EOS `fetch` [] Download a bottle (if available) or source packages for e and binaries for s. For files, also print SHA-256 checksums. EOS switch "--HEAD", description: "Fetch HEAD version instead of stable version." switch "--devel", description: "Fetch development version instead of stable version." switch "-f", "--force", description: "Remove a previously cached version and re-fetch." switch "-v", "--verbose", description: "Do a verbose VCS checkout, if the URL represents a VCS. This is useful for "\ "seeing if an existing VCS cache has been updated." switch "--retry", description: "Retry if downloading fails or re-download if the checksum of a previously cached "\ "version no longer matches." switch "--deps", description: "Also download dependencies for any listed ." switch "-s", "--build-from-source", description: "Download source packages rather than a bottle." switch "--build-bottle", description: "Download source packages (for eventual bottling) rather than a bottle." switch "--force-bottle", description: "Download a bottle if it exists for the current or newest version of macOS, "\ "even if it would not be used during installation." switch "--[no-]quarantine", description: "Disable/enable quarantining of downloads (default: enabled).", env: :cask_opts_quarantine switch "--formula", "--formulae", description: "Treat all named arguments as formulae." switch "--cask", "--casks", description: "Treat all named arguments as casks." conflicts "--formula", "--cask" conflicts "--devel", "--HEAD" conflicts "--build-from-source", "--build-bottle", "--force-bottle" conflicts "--cask", "--HEAD" conflicts "--cask", "--devel" conflicts "--cask", "--deps" conflicts "--cask", "-s" conflicts "--cask", "--build-bottle" conflicts "--cask", "--force-bottle" min_named :formula_or_cask end end def fetch args = fetch_args.parse only = :formula if args.formula? && !args.cask? only = :cask if args.cask? && !args.formula? bucket = if args.deps? args.named.to_formulae_and_casks.flat_map do |formula_or_cask| case formula_or_cask when Formula f = formula_or_cask [f, *f.recursive_dependencies.map(&:to_formula)] else formula_or_cask end end else args.named.to_formulae_and_casks(only: only) end.uniq puts "Fetching: #{bucket * ", "}" if bucket.size > 1 bucket.each do |formula_or_cask| case formula_or_cask when Formula f = formula_or_cask f.print_tap_action verb: "Fetching" fetched_bottle = false if fetch_bottle?(f, args: args) begin fetch_formula(f.bottle, args: args) rescue Interrupt raise rescue => e raise if Homebrew::EnvConfig.developer? fetched_bottle = false onoe e.message opoo "Bottle fetch failed: fetching the source." else fetched_bottle = true end end next if fetched_bottle fetch_formula(f, args: args) f.resources.each do |r| fetch_resource(r, args: args) r.patches.each { |p| fetch_patch(p, args: args) if p.external? } end f.patchlist.each { |p| fetch_patch(p, args: args) if p.external? } else cask = formula_or_cask options = { force: args.force?, quarantine: args.quarantine?, }.compact options[:quarantine] = true if options[:quarantine].nil? download = Cask::Download.new(cask, **options) fetch_cask(download, args: args) end end end def fetch_resource(r, args:) puts "Resource: #{r.name}" fetch_fetchable r, args: args rescue ChecksumMismatchError => e retry if retry_fetch?(r, args: args) opoo "Resource #{r.name} reports different #{e.hash_type}: #{e.expected}" end def fetch_formula(f, args:) fetch_fetchable f, args: args rescue ChecksumMismatchError => e retry if retry_fetch?(f, args: args) opoo "Formula reports different #{e.hash_type}: #{e.expected}" end def fetch_cask(cask_download, args:) fetch_fetchable cask_download, args: args rescue ChecksumMismatchError => e retry if retry_fetch?(cask_download, args: args) opoo "Cask reports different #{e.hash_type}: #{e.expected}" end def fetch_patch(p, args:) fetch_fetchable p, args: args rescue ChecksumMismatchError => e Homebrew.failed = true opoo "Patch reports different #{e.hash_type}: #{e.expected}" end def retry_fetch?(f, args:) @fetch_failed ||= Set.new if args.retry? && @fetch_failed.add?(f) ohai "Retrying download" f.clear_cache true else Homebrew.failed = true false end end def fetch_fetchable(f, args:) f.clear_cache if args.force? already_fetched = f.cached_download.exist? begin download = f.fetch(verify_download_integrity: false) rescue DownloadError retry if retry_fetch?(f, args: args) raise end return unless download.file? puts "Downloaded to: #{download}" unless already_fetched puts Checksum::TYPES.map { |t| "#{t.to_s.upcase}: #{download.send(t)}" } f.verify_download_integrity(download) end end