253 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			253 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: true
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| require "formula"
 | |
| require "fetch"
 | |
| require "cli/parser"
 | |
| require "cask/download"
 | |
| 
 | |
| module Homebrew
 | |
|   extend Fetch
 | |
| 
 | |
|   FETCH_MAX_TRIES = 5
 | |
| 
 | |
|   sig { returns(CLI::Parser) }
 | |
|   def self.fetch_args
 | |
|     Homebrew::CLI::Parser.new do
 | |
|       description <<~EOS
 | |
|         Download a bottle (if available) or source packages for <formula>e
 | |
|         and binaries for <cask>s. For files, also print SHA-256 checksums.
 | |
|       EOS
 | |
|       flag   "--os=",
 | |
|              description: "Download for the given operating system." \
 | |
|                           "(Pass `all` to download for all operating systems.)"
 | |
|       flag   "--arch=",
 | |
|              description: "Download for the given CPU architecture." \
 | |
|                           "(Pass `all` to download for all architectures.)"
 | |
|       flag   "--bottle-tag=",
 | |
|              description: "Download a bottle for given tag."
 | |
|       switch "--HEAD",
 | |
|              description: "Fetch HEAD 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. Tries at most #{FETCH_MAX_TRIES} times with " \
 | |
|                           "exponential backoff."
 | |
|       switch "--deps",
 | |
|              description: "Also download dependencies for any listed <formula>."
 | |
|       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 "--build-from-source", "--build-bottle", "--force-bottle", "--bottle-tag"
 | |
|       conflicts "--cask", "--HEAD"
 | |
|       conflicts "--cask", "--deps"
 | |
|       conflicts "--cask", "-s"
 | |
|       conflicts "--cask", "--build-bottle"
 | |
|       conflicts "--cask", "--force-bottle"
 | |
|       conflicts "--cask", "--bottle-tag"
 | |
|       conflicts "--formula", "--cask"
 | |
|       conflicts "--os", "--bottle-tag"
 | |
|       conflicts "--arch", "--bottle-tag"
 | |
| 
 | |
|       named_args [:formula, :cask], min: 1
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def self.fetch
 | |
|     args = fetch_args.parse
 | |
| 
 | |
|     Formulary.enable_factory_cache!
 | |
| 
 | |
|     bucket = if args.deps?
 | |
|       args.named.to_formulae_and_casks.flat_map do |formula_or_cask|
 | |
|         case formula_or_cask
 | |
|         when Formula
 | |
|           formula = formula_or_cask
 | |
|           [formula, *formula.recursive_dependencies.map(&:to_formula)]
 | |
|         else
 | |
|           formula_or_cask
 | |
|         end
 | |
|       end
 | |
|     else
 | |
|       args.named.to_formulae_and_casks
 | |
|     end.uniq
 | |
| 
 | |
|     os_arch_combinations = args.os_arch_combinations
 | |
| 
 | |
|     puts "Fetching: #{bucket * ", "}" if bucket.size > 1
 | |
|     bucket.each do |formula_or_cask|
 | |
|       case formula_or_cask
 | |
|       when Formula
 | |
|         formula = T.cast(formula_or_cask, Formula)
 | |
|         ref = formula.loaded_from_api? ? formula.full_name : formula.path
 | |
| 
 | |
|         os_arch_combinations.each do |os, arch|
 | |
|           SimulateSystem.with os: os, arch: arch do
 | |
|             Formulary.clear_cache
 | |
|             formula = Formulary.factory(ref)
 | |
| 
 | |
|             formula.print_tap_action verb: "Fetching"
 | |
| 
 | |
|             fetched_bottle = false
 | |
|             if fetch_bottle?(
 | |
|               formula,
 | |
|               force_bottle:               args.force_bottle?,
 | |
|               bottle_tag:                 args.bottle_tag&.to_sym,
 | |
|               build_from_source_formulae: args.build_from_source_formulae,
 | |
|               os:                         args.os&.to_sym,
 | |
|               arch:                       args.arch&.to_sym,
 | |
|             )
 | |
|               begin
 | |
|                 formula.clear_cache if args.force?
 | |
| 
 | |
|                 bottle_tag = if (bottle_tag = args.bottle_tag&.to_sym)
 | |
|                   Utils::Bottles::Tag.from_symbol(bottle_tag)
 | |
|                 else
 | |
|                   Utils::Bottles::Tag.new(system: os, arch: arch)
 | |
|                 end
 | |
| 
 | |
|                 bottle = formula.bottle_for_tag(bottle_tag)
 | |
| 
 | |
|                 if bottle.nil?
 | |
|                   opoo "Bottle for tag #{bottle_tag.to_sym.inspect} is unavailable."
 | |
|                   next
 | |
|                 end
 | |
| 
 | |
|                 formula.fetch_bottle_tab
 | |
|                 fetch_formula(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 instead."
 | |
|               else
 | |
|                 fetched_bottle = true
 | |
|               end
 | |
|             end
 | |
| 
 | |
|             next if fetched_bottle
 | |
| 
 | |
|             fetch_formula(formula, args: args)
 | |
| 
 | |
|             formula.resources.each do |r|
 | |
|               fetch_resource(r, args: args)
 | |
|               r.patches.each { |p| fetch_patch(p, args: args) if p.external? }
 | |
|             end
 | |
| 
 | |
|             formula.patchlist.each { |p| fetch_patch(p, args: args) if p.external? }
 | |
|           end
 | |
|         end
 | |
|       else
 | |
|         cask = formula_or_cask
 | |
|         ref = cask.loaded_from_api? ? cask.full_name : cask.sourcefile_path
 | |
| 
 | |
|         os_arch_combinations.each do |os, arch|
 | |
|           next if os == :linux
 | |
| 
 | |
|           SimulateSystem.with os: os, arch: arch do
 | |
|             cask = Cask::CaskLoader.load(ref)
 | |
| 
 | |
|             if cask.depends_on.macos&.satisfied? == false
 | |
|               opoo "#{cask.token}: #{cask.depends_on.macos.message(type: :cask)}"
 | |
|               next
 | |
|             end
 | |
| 
 | |
|             quarantine = args.quarantine?
 | |
|             quarantine = true if quarantine.nil?
 | |
| 
 | |
|             download = Cask::Download.new(cask, quarantine: quarantine)
 | |
|             fetch_cask(download, args: args)
 | |
|           end
 | |
|         end
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def self.fetch_resource(resource, args:)
 | |
|     puts "Resource: #{resource.name}"
 | |
|     fetch_fetchable resource, args: args
 | |
|   rescue ChecksumMismatchError => e
 | |
|     retry if retry_fetch?(resource, args: args)
 | |
|     opoo "Resource #{resource.name} reports different sha256: #{e.expected}"
 | |
|   end
 | |
| 
 | |
|   def self.fetch_formula(formula, args:)
 | |
|     fetch_fetchable formula, args: args
 | |
|   rescue ChecksumMismatchError => e
 | |
|     retry if retry_fetch?(formula, args: args)
 | |
|     opoo "Formula reports different sha256: #{e.expected}"
 | |
|   end
 | |
| 
 | |
|   def self.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 sha256: #{e.expected}"
 | |
|   end
 | |
| 
 | |
|   def self.fetch_patch(patch, args:)
 | |
|     fetch_fetchable patch, args: args
 | |
|   rescue ChecksumMismatchError => e
 | |
|     opoo "Patch reports different sha256: #{e.expected}"
 | |
|     Homebrew.failed = true
 | |
|   end
 | |
| 
 | |
|   def self.retry_fetch?(formula, args:)
 | |
|     @fetch_tries ||= Hash.new { |h, k| h[k] = 1 }
 | |
|     if args.retry? && (@fetch_tries[formula] < FETCH_MAX_TRIES)
 | |
|       wait = 2 ** @fetch_tries[formula]
 | |
|       remaining = FETCH_MAX_TRIES - @fetch_tries[formula]
 | |
|       what = Utils.pluralize("tr", remaining, plural: "ies", singular: "y")
 | |
| 
 | |
|       ohai "Retrying download in #{wait}s... (#{remaining} #{what} left)"
 | |
|       sleep wait
 | |
| 
 | |
|       formula.clear_cache
 | |
|       @fetch_tries[formula] += 1
 | |
|       true
 | |
|     else
 | |
|       Homebrew.failed = true
 | |
|       false
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def self.fetch_fetchable(formula, args:)
 | |
|     formula.clear_cache if args.force?
 | |
| 
 | |
|     already_fetched = formula.cached_download.exist?
 | |
| 
 | |
|     begin
 | |
|       download = formula.fetch(verify_download_integrity: false)
 | |
|     rescue DownloadError
 | |
|       retry if retry_fetch?(formula, args: args)
 | |
|       raise
 | |
|     end
 | |
| 
 | |
|     return unless download.file?
 | |
| 
 | |
|     puts "Downloaded to: #{download}" unless already_fetched
 | |
|     puts "SHA256: #{download.sha256}"
 | |
| 
 | |
|     formula.verify_download_integrity(download)
 | |
|   end
 | |
| end
 | 
