355 lines
12 KiB
Ruby
Raw Normal View History

2025-06-17 16:33:16 +01:00
# typed: strict
# frozen_string_literal: true
2024-03-29 18:24:18 -07:00
require "abstract_command"
require "formula"
require "fetch"
2020-11-19 18:12:16 +01:00
require "cask/download"
2024-07-14 11:42:22 -04:00
require "retryable_download"
2025-06-17 16:33:16 +01:00
require "download_queue"
2011-03-12 09:40:10 -08: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
2025-06-17 16:33:16 +01:00
sig { returns(Integer) }
2024-07-14 11:42:22 -04:00
def concurrency
2025-06-17 16:33:16 +01:00
@concurrency ||= T.let(args.concurrency&.to_i || 1, T.nilable(Integer))
2024-07-14 11:42:22 -04:00
end
2025-06-17 16:33:16 +01:00
sig { returns(DownloadQueue) }
2024-07-14 11:42:22 -04:00
def download_queue
2025-06-17 16:33:16 +01:00
@download_queue ||= T.let(begin
2024-07-14 11:42:22 -04:00
DownloadQueue.new(concurrency)
2025-06-17 16:33:16 +01:00
end, T.nilable(DownloadQueue))
2024-07-14 11:42:22 -04:00
end
class Spinner
FRAMES = [
2024-07-16 00:01:40 -04:00
"",
"",
"",
"",
"",
"",
"",
"",
"",
"",
].freeze
sig { void }
def initialize
2025-06-17 16:33:16 +01:00
@start = T.let(Time.now, Time)
@i = T.let(0, Integer)
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
2025-06-17 16:33:16 +01:00
formula = formula_or_cask
2024-03-29 18:24:18 -07:00
ref = formula.loaded_from_api? ? formula.full_name : formula.path
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)
2024-03-29 18:24:18 -07:00
formula.print_tap_action verb: "Fetching"
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
end
2024-03-29 18:24:18 -07:00
next if fetched_bottle
2025-06-17 16:33:16 +01:00
if (resource = formula.resource)
fetch_downloadable(resource)
end
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
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
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
2024-03-29 18:24:18 -07:00
os_arch_combinations.each do |os, arch|
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
2024-03-29 18:24:18 -07:00
quarantine = args.quarantine?
quarantine = true if quarantine.nil?
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
end
end
if concurrency == 1
2024-07-16 00:01:40 -04:00
downloads.each do |downloadable, promise|
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
else
spinner = Spinner.new
2025-06-17 16:33:16 +01:00
remaining_downloads = downloads.dup.to_a
previous_pending_line_count = 0
begin
2024-07-16 00:01:40 -04:00
$stdout.print Tty.hide_cursor
$stdout.flush
output_message = lambda do |downloadable, future, last|
2024-07-16 10:18:26 -04:00
status = case future.state
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}"
else
2024-07-16 10:18:26 -04:00
raise future.state.to_s
end
2024-03-30 16:31:13 -07:00
message = "#{downloadable.download_type.capitalize} #{downloadable.name}"
$stdout.print "#{status} #{message}#{"\n" unless last}"
2024-07-16 00:01:40 -04:00
$stdout.flush
2025-02-03 14:40:48 +01:00
if future.rejected?
if (e = future.reason).is_a?(ChecksumMismatchError)
opoo "#{downloadable.download_type.capitalize} reports different checksum: #{e.expected}"
Homebrew.failed = true if downloadable.is_a?(Resource::Patch)
next 2
else
message = future.reason.to_s
onoe message
Homebrew.failed = true
next message.count("\n")
end
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-16 10:18:26 -04:00
finished_downloads.each do |downloadable, future|
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
output_message.call(downloadable, future, false)
end
previous_pending_line_count = 0
max_lines = [concurrency, Tty.height].min
remaining_downloads.each_with_index do |(downloadable, future), i|
break if previous_pending_line_count >= max_lines
2024-08-14 21:41:01 +02:00
$stdout.print Tty.clear_to_end
2024-07-16 00:01:40 -04:00
$stdout.flush
last = i == max_lines - 1 || i == remaining_downloads.count - 1
previous_pending_line_count += output_message.call(downloadable, future, last)
end
if previous_pending_line_count.positive?
if (previous_pending_line_count - 1).zero?
$stdout.print Tty.move_cursor_beginning
else
$stdout.print Tty.move_cursor_up_beginning(previous_pending_line_count - 1)
end
$stdout.flush
end
2024-07-15 17:00:01 -04:00
sleep 0.05
rescue Interrupt
remaining_downloads.each do |_, future|
# FIXME: Implement cancellation of running downloads.
end
2024-09-07 14:45:30 +02:00
download_queue.cancel
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
raise
end
end
ensure
2024-07-16 00:01:40 -04:00
$stdout.print Tty.show_cursor
$stdout.flush
end
2024-07-14 11:42:22 -04:00
end
2024-09-07 14:45:30 +02:00
ensure
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
2025-06-17 16:33:16 +01:00
sig { returns(T::Hash[T.any(Resource, Bottle, Cask::Download), Concurrent::Promises::Future]) }
2024-07-14 11:42:22 -04:00
def downloads
2025-06-17 16:33:16 +01:00
@downloads ||= T.let({}, T.nilable(T::Hash[T.any(Resource, Bottle, Cask::Download),
Concurrent::Promises::Future]))
2024-03-29 18:24:18 -07:00
end
2025-06-17 16:33:16 +01:00
sig { params(downloadable: T.any(Resource, Bottle, Cask::Download)).void }
2024-07-14 11:42:22 -04:00
def fetch_downloadable(downloadable)
2024-09-05 18:11:49 +02:00
downloads[downloadable] ||= begin
tries = args.retry? ? {} : { tries: 1 }
download_queue.enqueue(RetryableDownload.new(downloadable, **tries), force: args.force?)
end
2024-03-29 18:24:18 -07:00
end
end
2011-03-12 09:40:10 -08:00
end
end