150 lines
4.3 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
require "formula"
require "fetch"
2019-04-17 18:25:08 +09:00
require "cli/parser"
2011-03-12 09:40:10 -08:00
module Homebrew
extend Fetch
2016-09-26 01:44:51 +02:00
module_function
2018-10-27 23:44:32 +05:30
def fetch_args
Homebrew::CLI::Parser.new do
usage_banner <<~EOS
`fetch` [<options>] <formula>
2018-10-27 23:44:32 +05:30
Download a bottle (if available) or source packages for <formula>.
2018-10-27 23:44:32 +05:30
For tarballs, also print SHA-256 checksums.
EOS
switch "--HEAD",
2019-04-30 08:44:35 +01:00
description: "Fetch HEAD version instead of stable version."
2018-10-27 23:44:32 +05:30
switch "--devel",
2019-04-30 08:44:35 +01:00
description: "Fetch development version instead of stable version."
switch "-f", "--force",
description: "Remove a previously cached version and re-fetch."
2020-07-30 18:40:10 +02:00
switch "-v", "--verbose",
2019-04-30 08:44:35 +01:00
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."
2018-10-27 23:44:32 +05:30
switch "--retry",
description: "Retry if downloading fails or re-download if the checksum of a previously cached "\
2019-08-06 13:23:19 -04:00
"version no longer matches."
2018-10-27 23:44:32 +05:30
switch "--deps",
description: "Also download dependencies for any listed <formula>."
2018-10-27 23:44:32 +05:30
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."
2018-10-27 23:44:32 +05:30
switch "--force-bottle",
2019-04-30 08:44:35 +01:00
description: "Download a bottle if it exists for the current or newest version of macOS, "\
"even if it would not be used during installation."
2020-07-30 18:40:10 +02:00
conflicts "--devel", "--HEAD"
conflicts "--build-from-source", "--build-bottle", "--force-bottle"
min_named :formula
2018-10-27 23:44:32 +05:30
end
end
2011-03-12 09:40:10 -08:00
def fetch
args = fetch_args.parse
2018-10-27 23:44:32 +05:30
if args.deps?
bucket = []
args.named.to_formulae.each do |f|
bucket << f
bucket.concat f.recursive_dependencies.map(&:to_formula)
end
bucket.uniq!
else
bucket = args.named.to_formulae
end
puts "Fetching: #{bucket * ", "}" if bucket.size > 1
bucket.each do |f|
f.print_tap_action verb: "Fetching"
fetched_bottle = false
if fetch_bottle?(f, args: args)
begin
2020-07-31 19:14:25 +02:00
fetch_formula(f.bottle, args: args)
rescue Interrupt
raise
rescue => e
2020-04-05 15:44:50 +01:00
raise if Homebrew::EnvConfig.developer?
2018-09-17 02:45:00 +02:00
fetched_bottle = false
onoe e.message
opoo "Bottle fetch failed: fetching the source."
else
fetched_bottle = true
end
end
2016-09-10 10:24:56 +01:00
next if fetched_bottle
2018-09-17 02:45:00 +02:00
2020-07-31 19:14:25 +02:00
fetch_formula(f, args: args)
2018-01-21 23:41:54 -08:00
f.resources.each do |r|
2020-07-31 19:14:25 +02:00
fetch_resource(r, args: args)
r.patches.each { |p| fetch_patch(p, args: args) if p.external? }
2018-01-21 23:41:54 -08:00
end
2020-07-31 19:14:25 +02:00
f.patchlist.each { |p| fetch_patch(p, args: args) if p.external? }
end
end
2020-07-31 19:14:25 +02:00
def fetch_resource(r, args:)
puts "Resource: #{r.name}"
2020-07-31 19:14:25 +02:00
fetch_fetchable r, args: args
rescue ChecksumMismatchError => e
2020-07-31 19:14:25 +02:00
retry if retry_fetch?(r, args: args)
opoo "Resource #{r.name} reports different #{e.hash_type}: #{e.expected}"
end
2020-07-31 19:14:25 +02:00
def fetch_formula(f, args:)
fetch_fetchable f, args: args
rescue ChecksumMismatchError => e
2020-07-31 19:14:25 +02:00
retry if retry_fetch?(f, args: args)
opoo "Formula reports different #{e.hash_type}: #{e.expected}"
end
2020-07-31 19:14:25 +02:00
def fetch_patch(p, args:)
fetch_fetchable p, args: args
2014-03-13 19:51:23 -05:00
rescue ChecksumMismatchError => e
Homebrew.failed = true
opoo "Patch reports different #{e.hash_type}: #{e.expected}"
end
2020-07-31 19:14:25 +02:00
def retry_fetch?(f, args:)
2014-08-22 22:55:10 -05:00
@fetch_failed ||= Set.new
2018-10-27 23:44:32 +05:30
if args.retry? && @fetch_failed.add?(f)
2014-08-22 22:55:10 -05:00
ohai "Retrying download"
f.clear_cache
true
else
Homebrew.failed = true
2014-08-22 22:55:10 -05:00
false
end
end
2020-07-31 19:14:25 +02:00
def fetch_fetchable(f, args:)
2018-10-27 23:44:32 +05:30
f.clear_cache if args.force?
already_fetched = f.cached_download.exist?
begin
download = f.fetch(verify_download_integrity: false)
2014-08-22 22:55:10 -05:00
rescue DownloadError
2020-07-31 19:14:25 +02:00
retry if retry_fetch?(f, args: args)
2014-08-22 22:55:10 -05:00
raise
end
2011-03-12 09:40:10 -08:00
return unless download.file?
2011-03-12 09:40:10 -08:00
puts "Downloaded to: #{download}" unless already_fetched
puts Checksum::TYPES.map { |t| "#{t.to_s.upcase}: #{download.send(t)}" }
2013-05-16 14:06:26 -05:00
f.verify_download_integrity(download)
2011-03-12 09:40:10 -08:00
end
end