2024-08-12 10:30:59 +01:00
|
|
|
# typed: true # rubocop:todo Sorbet/StrictSigil
|
2019-04-19 15:38:03 +09:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
require "abstract_command"
|
2015-08-03 13:09:07 +01:00
|
|
|
require "formula"
|
2018-06-05 23:19:18 -04:00
|
|
|
require "fetch"
|
2020-11-19 18:12:16 +01:00
|
|
|
require "cask/download"
|
2024-07-14 11:42:22 -04:00
|
|
|
require "retryable_download"
|
2011-03-12 09:40:10 -08:00
|
|
|
|
2014-06-18 22:41:47 -05:00
|
|
|
module Homebrew
|
2024-03-29 18:24:18 -07:00
|
|
|
module Cmd
|
|
|
|
class FetchCmd < AbstractCommand
|
|
|
|
include Fetch
|
|
|
|
FETCH_MAX_TRIES = 5
|
|
|
|
|
|
|
|
cmd_args 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."
|
2024-07-14 11:42:22 -04:00
|
|
|
flag "--concurrency=", description: "Number of concurrent downloads.", hidden: true
|
2024-03-29 18:24:18 -07:00
|
|
|
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
|
2018-10-27 23:44:32 +05:30
|
|
|
|
2024-07-14 11:42:22 -04:00
|
|
|
def concurrency
|
|
|
|
@concurrency ||= args.concurrency&.to_i || 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def download_queue
|
|
|
|
@download_queue ||= begin
|
|
|
|
require "download_queue"
|
|
|
|
DownloadQueue.new(concurrency)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2024-07-15 16:31:21 -04:00
|
|
|
class Spinner
|
|
|
|
FRAMES = [
|
2024-07-16 00:01:40 -04:00
|
|
|
"⠋",
|
|
|
|
"⠙",
|
|
|
|
"⠚",
|
|
|
|
"⠞",
|
|
|
|
"⠖",
|
|
|
|
"⠦",
|
|
|
|
"⠴",
|
|
|
|
"⠲",
|
|
|
|
"⠳",
|
|
|
|
"⠓",
|
2024-07-15 16:31:21 -04:00
|
|
|
].freeze
|
|
|
|
|
|
|
|
sig { void }
|
|
|
|
def initialize
|
|
|
|
@start = Time.now
|
|
|
|
@i = 0
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { returns(String) }
|
|
|
|
def to_s
|
|
|
|
now = Time.now
|
|
|
|
if @start + 0.1 < now
|
|
|
|
@start = now
|
|
|
|
@i = (@i + 1) % FRAMES.count
|
|
|
|
end
|
|
|
|
|
|
|
|
FRAMES.fetch(@i)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
sig { override.void }
|
|
|
|
def run
|
|
|
|
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
|
2018-10-27 23:44:32 +05:30
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
os_arch_combinations = args.os_arch_combinations
|
2023-03-25 11:56:05 +01:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
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
|
2023-04-14 15:33:40 +02:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
os_arch_combinations.each do |os, arch|
|
|
|
|
SimulateSystem.with(os:, arch:) do
|
|
|
|
formula = Formulary.factory(ref, args.HEAD? ? :head : :stable)
|
2023-04-14 15:33:40 +02:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
formula.print_tap_action verb: "Fetching"
|
2023-04-14 15:33:40 +02:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
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:)
|
|
|
|
end
|
|
|
|
|
|
|
|
bottle = formula.bottle_for_tag(bottle_tag)
|
|
|
|
|
|
|
|
if bottle.nil?
|
|
|
|
opoo "Bottle for tag #{bottle_tag.to_sym.inspect} is unavailable."
|
|
|
|
next
|
|
|
|
end
|
|
|
|
|
2024-07-14 11:42:22 -04:00
|
|
|
if (manifest_resource = bottle.github_packages_manifest_resource)
|
|
|
|
fetch_downloadable(manifest_resource)
|
2024-03-29 18:24:18 -07:00
|
|
|
end
|
2024-07-14 11:42:22 -04:00
|
|
|
fetch_downloadable(bottle)
|
2024-03-29 18:24:18 -07:00
|
|
|
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
|
2023-10-13 21:14:07 +01:00
|
|
|
end
|
2023-04-14 15:33:40 +02:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
next if fetched_bottle
|
2023-04-14 15:33:40 +02:00
|
|
|
|
2024-07-14 11:42:22 -04:00
|
|
|
fetch_downloadable(formula.resource)
|
2023-04-14 15:33:40 +02:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
formula.resources.each do |r|
|
2024-07-14 11:42:22 -04:00
|
|
|
fetch_downloadable(r)
|
|
|
|
r.patches.each { |patch| fetch_downloadable(patch.resource) if patch.external? }
|
2024-03-29 18:24:18 -07:00
|
|
|
end
|
2023-04-14 15:33:40 +02:00
|
|
|
|
2024-07-14 11:42:22 -04:00
|
|
|
formula.patchlist.each { |patch| fetch_downloadable(patch.resource) if patch.external? }
|
2024-03-29 18:24:18 -07:00
|
|
|
end
|
2023-04-14 15:33:40 +02:00
|
|
|
end
|
2024-03-29 18:24:18 -07:00
|
|
|
else
|
|
|
|
cask = formula_or_cask
|
|
|
|
ref = cask.loaded_from_api? ? cask.full_name : cask.sourcefile_path
|
2023-04-14 15:33:40 +02:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
os_arch_combinations.each do |os, arch|
|
|
|
|
next if os == :linux
|
2020-11-19 18:12:16 +01:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
SimulateSystem.with(os:, arch:) do
|
|
|
|
cask = Cask::CaskLoader.load(ref)
|
2020-11-19 18:12:16 +01:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
if cask.url.nil? || cask.sha256.nil?
|
|
|
|
opoo "Cask #{cask} is not supported on os #{os} and arch #{arch}"
|
|
|
|
next
|
|
|
|
end
|
2023-09-02 01:33:39 -07:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
quarantine = args.quarantine?
|
|
|
|
quarantine = true if quarantine.nil?
|
2023-04-14 15:33:40 +02:00
|
|
|
|
2024-03-29 18:24:18 -07:00
|
|
|
download = Cask::Download.new(cask, quarantine:)
|
2024-07-14 11:42:22 -04:00
|
|
|
fetch_downloadable(download)
|
2024-03-29 18:24:18 -07:00
|
|
|
end
|
|
|
|
end
|
2023-04-14 15:33:40 +02:00
|
|
|
end
|
|
|
|
end
|
2011-04-14 14:57:21 -07:00
|
|
|
|
2024-07-15 16:31:21 -04:00
|
|
|
if concurrency == 1
|
2024-07-16 00:01:40 -04:00
|
|
|
downloads.each do |downloadable, promise|
|
2024-07-15 16:31:21 -04:00
|
|
|
promise.wait!
|
|
|
|
rescue ChecksumMismatchError => e
|
|
|
|
opoo "#{downloadable.download_type.capitalize} reports different checksum: #{e.expected}"
|
|
|
|
Homebrew.failed = true if downloadable.is_a?(Resource::Patch)
|
2024-07-14 11:42:22 -04:00
|
|
|
end
|
2024-07-15 16:31:21 -04:00
|
|
|
else
|
|
|
|
spinner = Spinner.new
|
|
|
|
remaining_downloads = downloads.dup
|
|
|
|
previous_pending_line_count = 0
|
|
|
|
|
|
|
|
begin
|
2024-07-16 00:01:40 -04:00
|
|
|
$stdout.print Tty.hide_cursor
|
|
|
|
$stdout.flush
|
2024-07-15 16:31:21 -04:00
|
|
|
|
2024-07-16 10:18:26 -04:00
|
|
|
output_message = lambda do |downloadable, future|
|
|
|
|
status = case future.state
|
2024-07-15 16:31:21 -04:00
|
|
|
when :fulfilled
|
|
|
|
"#{Tty.green}✔︎#{Tty.reset}"
|
|
|
|
when :rejected
|
|
|
|
"#{Tty.red}✘#{Tty.reset}"
|
2024-07-16 10:18:26 -04:00
|
|
|
when :pending, :processing
|
2024-07-16 00:01:40 -04:00
|
|
|
"#{Tty.blue}#{spinner}#{Tty.reset}"
|
2024-07-15 16:31:21 -04:00
|
|
|
else
|
2024-07-16 10:18:26 -04:00
|
|
|
raise future.state.to_s
|
2024-07-15 16:31:21 -04:00
|
|
|
end
|
2024-03-30 16:31:13 -07:00
|
|
|
|
2024-07-15 16:31:21 -04:00
|
|
|
message = "#{downloadable.download_type.capitalize} #{downloadable.name}"
|
2024-07-16 00:01:40 -04:00
|
|
|
$stdout.puts "#{status} #{message}"
|
|
|
|
$stdout.flush
|
2013-08-06 19:53:43 -07:00
|
|
|
|
2024-07-16 10:18:26 -04:00
|
|
|
if future.rejected? && (e = future.reason).is_a?(ChecksumMismatchError)
|
2024-07-15 16:31:21 -04:00
|
|
|
opoo "#{downloadable.download_type.capitalize} reports different checksum: #{e.expected}"
|
|
|
|
Homebrew.failed = true if downloadable.is_a?(Resource::Patch)
|
|
|
|
next 2
|
|
|
|
end
|
|
|
|
|
|
|
|
1
|
|
|
|
end
|
|
|
|
|
|
|
|
until remaining_downloads.empty?
|
|
|
|
begin
|
|
|
|
finished_states = [:fulfilled, :rejected]
|
|
|
|
|
2024-07-16 10:18:26 -04:00
|
|
|
finished_downloads, remaining_downloads = remaining_downloads.partition do |_, future|
|
|
|
|
finished_states.include?(future.state)
|
2024-07-15 21:17:45 -04:00
|
|
|
end
|
2024-07-15 16:31:21 -04:00
|
|
|
|
2024-07-16 10:18:26 -04:00
|
|
|
finished_downloads.each do |downloadable, future|
|
2024-07-15 16:31:21 -04:00
|
|
|
previous_pending_line_count -= 1
|
2024-08-14 21:41:01 +02:00
|
|
|
$stdout.print Tty.clear_to_end
|
2024-07-16 00:01:40 -04:00
|
|
|
$stdout.flush
|
2024-07-16 10:18:26 -04:00
|
|
|
output_message.call(downloadable, future)
|
2024-07-15 16:31:21 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
previous_pending_line_count = 0
|
2024-07-16 10:18:26 -04:00
|
|
|
remaining_downloads.each do |downloadable, future|
|
2024-09-04 22:40:21 +02:00
|
|
|
# FIXME: Allow printing full terminal height.
|
2024-07-15 17:00:01 -04:00
|
|
|
break if previous_pending_line_count >= [concurrency, (Tty.height - 1)].min
|
2024-07-15 16:31:21 -04:00
|
|
|
|
2024-08-14 21:41:01 +02:00
|
|
|
$stdout.print Tty.clear_to_end
|
2024-07-16 00:01:40 -04:00
|
|
|
$stdout.flush
|
2024-07-16 10:18:26 -04:00
|
|
|
previous_pending_line_count += output_message.call(downloadable, future)
|
2024-07-15 16:31:21 -04:00
|
|
|
end
|
2013-08-06 19:53:43 -07:00
|
|
|
|
2024-07-15 16:31:21 -04:00
|
|
|
if previous_pending_line_count.positive?
|
2024-08-21 23:01:48 +02:00
|
|
|
$stdout.print Tty.move_cursor_up_beginning(previous_pending_line_count)
|
2024-07-15 16:31:21 -04:00
|
|
|
$stdout.flush
|
|
|
|
end
|
2024-07-15 17:00:01 -04:00
|
|
|
|
|
|
|
sleep 0.05
|
2024-07-15 16:31:21 -04:00
|
|
|
rescue Interrupt
|
2024-08-14 21:59:04 +02:00
|
|
|
remaining_downloads.each do |_, future|
|
|
|
|
# FIXME: Implement cancellation of running downloads.
|
|
|
|
end
|
|
|
|
|
2024-08-14 21:41:01 +02:00
|
|
|
if previous_pending_line_count.positive?
|
|
|
|
$stdout.print Tty.move_cursor_down(previous_pending_line_count - 1)
|
|
|
|
$stdout.flush
|
|
|
|
end
|
|
|
|
|
2024-07-15 16:31:21 -04:00
|
|
|
raise
|
|
|
|
end
|
|
|
|
end
|
|
|
|
ensure
|
2024-07-16 00:01:40 -04:00
|
|
|
$stdout.print Tty.show_cursor
|
|
|
|
$stdout.flush
|
2024-07-15 16:31:21 -04:00
|
|
|
end
|
2024-07-14 11:42:22 -04:00
|
|
|
end
|
2020-11-19 18:12:16 +01:00
|
|
|
|
2024-07-14 11:42:22 -04:00
|
|
|
download_queue.shutdown
|
2024-03-29 18:24:18 -07:00
|
|
|
end
|
2014-03-13 19:51:23 -05:00
|
|
|
|
2024-07-14 11:42:22 -04:00
|
|
|
private
|
2013-10-31 14:28:49 -05:00
|
|
|
|
2024-07-14 11:42:22 -04:00
|
|
|
def downloads
|
|
|
|
@downloads ||= {}
|
2024-03-29 18:24:18 -07:00
|
|
|
end
|
2014-08-16 08:48:28 +01:00
|
|
|
|
2024-07-14 11:42:22 -04:00
|
|
|
def fetch_downloadable(downloadable)
|
2024-07-16 10:18:26 -04:00
|
|
|
downloads[downloadable] ||= download_queue.enqueue(RetryableDownload.new(downloadable), force: args.force?)
|
2024-03-29 18:24:18 -07:00
|
|
|
end
|
|
|
|
end
|
2011-03-12 09:40:10 -08:00
|
|
|
end
|
|
|
|
end
|