Enable strict typing in NamedArgs

This commit is contained in:
Douglas Eichelberger 2024-12-03 17:43:22 -08:00
parent 58e72b0a7f
commit e1fdd2eda4
11 changed files with 122 additions and 72 deletions

View File

@ -1,4 +1,4 @@
# typed: true # rubocop:todo Sorbet/StrictSigil # typed: strict
# frozen_string_literal: true # frozen_string_literal: true
require "delegate" require "delegate"
@ -8,6 +8,10 @@ module Homebrew
module CLI module CLI
# Helper class for loading formulae/casks from named arguments. # Helper class for loading formulae/casks from named arguments.
class NamedArgs < Array class NamedArgs < Array
extend T::Generic
Elem = type_member(:out) { { fixed: String } }
sig { sig {
params( params(
args: String, args: String,
@ -39,14 +43,23 @@ module Homebrew
super(@args) super(@args)
end end
sig { returns(Args) }
attr_reader :parent attr_reader :parent
sig { returns(T::Array[Cask::Cask]) }
def to_casks def to_casks
@to_casks ||= to_formulae_and_casks(only: :cask).freeze @to_casks ||= T.let(
to_formulae_and_casks(only: :cask).freeze, T.nilable(T::Array[T.any(Formula, Keg, Cask::Cask)])
)
T.cast(@to_casks, T::Array[Cask::Cask])
end end
sig { returns(T::Array[Formula]) }
def to_formulae def to_formulae
@to_formulae ||= to_formulae_and_casks(only: :formula).freeze @to_formulae ||= T.let(
to_formulae_and_casks(only: :formula).freeze, T.nilable(T::Array[T.any(Formula, Keg, Cask::Cask)])
)
T.cast(@to_formulae, T::Array[Formula])
end end
# Convert named arguments to {Formula} or {Cask} objects. # Convert named arguments to {Formula} or {Cask} objects.
@ -62,13 +75,15 @@ module Homebrew
).returns(T::Array[T.any(Formula, Keg, Cask::Cask)]) ).returns(T::Array[T.any(Formula, Keg, Cask::Cask)])
} }
def to_formulae_and_casks( def to_formulae_and_casks(
only: parent&.only_formula_or_cask, only: parent.only_formula_or_cask,
ignore_unavailable: false, ignore_unavailable: false,
method: T.unsafe(nil), method: T.unsafe(nil),
uniq: true, uniq: true,
warn: T.unsafe(nil) warn: T.unsafe(nil)
) )
@to_formulae_and_casks ||= {} @to_formulae_and_casks ||= T.let(
{}, T.nilable(T::Hash[T.nilable(Symbol), T::Array[T.any(Formula, Keg, Cask::Cask)]])
)
@to_formulae_and_casks[only] ||= downcased_unique_named.flat_map do |name| @to_formulae_and_casks[only] ||= downcased_unique_named.flat_map do |name|
options = { warn: }.compact options = { warn: }.compact
load_formula_or_cask(name, only:, method:, **options) load_formula_or_cask(name, only:, method:, **options)
@ -83,20 +98,31 @@ module Homebrew
end.freeze end.freeze
if uniq if uniq
@to_formulae_and_casks[only].uniq.freeze @to_formulae_and_casks.fetch(only).uniq.freeze
else else
@to_formulae_and_casks[only] @to_formulae_and_casks.fetch(only)
end end
end end
def to_formulae_to_casks(only: parent&.only_formula_or_cask, method: nil) sig {
@to_formulae_to_casks ||= {} params(only: T.nilable(Symbol), method: T.nilable(Symbol))
@to_formulae_to_casks[[method, only]] = to_formulae_and_casks(only:, method:) .returns([T::Array[T.any(Formula, Keg)], T::Array[Cask::Cask]])
.partition { |o| o.is_a?(Formula) || o.is_a?(Keg) } }
.map(&:freeze).freeze def to_formulae_to_casks(only: parent.only_formula_or_cask, method: nil)
@to_formulae_to_casks ||= T.let(
{}, T.nilable(T::Hash[[T.nilable(Symbol), T.nilable(Symbol)],
[T::Array[T.any(Formula, Keg)], T::Array[Cask::Cask]]])
)
@to_formulae_to_casks[[method, only]] =
T.cast(
to_formulae_and_casks(only:, method:).partition { |o| o.is_a?(Formula) || o.is_a?(Keg) }
.map(&:freeze).freeze,
[T::Array[T.any(Formula, Keg)], T::Array[Cask::Cask]],
)
end end
# Returns formulae and casks after validating that a tap is present for each of them. # Returns formulae and casks after validating that a tap is present for each of them.
sig { returns(T::Array[T.any(Formula, Keg, Cask::Cask)]) }
def to_formulae_and_casks_with_taps def to_formulae_and_casks_with_taps
formulae_and_casks_with_taps, formulae_and_casks_without_taps = formulae_and_casks_with_taps, formulae_and_casks_without_taps =
to_formulae_and_casks.partition do |formula_or_cask| to_formulae_and_casks.partition do |formula_or_cask|
@ -118,8 +144,18 @@ module Homebrew
ERROR ERROR
end end
def to_formulae_and_casks_and_unavailable(only: parent&.only_formula_or_cask, method: nil) sig {
@to_formulae_casks_unknowns ||= {} params(only: T.nilable(Symbol), method: T.nilable(Symbol))
.returns(T::Array[T.any(Formula, Keg, Cask::Cask, T::Array[Keg], FormulaOrCaskUnavailableError)])
}
def to_formulae_and_casks_and_unavailable(only: parent.only_formula_or_cask, method: nil)
@to_formulae_casks_unknowns ||= T.let(
{},
T.nilable(
T::Hash[T.nilable(Symbol),
T::Array[T.any(Formula, Keg, Cask::Cask, T::Array[Keg], FormulaOrCaskUnavailableError)]],
),
)
@to_formulae_casks_unknowns[method] = downcased_unique_named.map do |name| @to_formulae_casks_unknowns[method] = downcased_unique_named.map do |name|
load_formula_or_cask(name, only:, method:) load_formula_or_cask(name, only:, method:)
rescue FormulaOrCaskUnavailableError => e rescue FormulaOrCaskUnavailableError => e
@ -127,6 +163,10 @@ module Homebrew
end.uniq.freeze end.uniq.freeze
end end
sig {
params(name: String, only: T.nilable(Symbol), method: T.nilable(Symbol), warn: T.nilable(T::Boolean))
.returns(T.any(Formula, Keg, Cask::Cask, T::Array[Keg]))
}
def load_formula_or_cask(name, only: nil, method: nil, warn: nil) def load_formula_or_cask(name, only: nil, method: nil, warn: nil)
Homebrew.with_no_api_env_if_needed(@without_api) do Homebrew.with_no_api_env_if_needed(@without_api) do
unreadable_error = nil unreadable_error = nil
@ -246,17 +286,16 @@ module Homebrew
user, repo, short_name = name.downcase.split("/", 3) user, repo, short_name = name.downcase.split("/", 3)
if repo.present? && short_name.present? if repo.present? && short_name.present?
tap = Tap.fetch(user, repo) tap = Tap.fetch(T.must(user), repo)
raise TapFormulaOrCaskUnavailableError.new(tap, short_name) raise TapFormulaOrCaskUnavailableError.new(tap, short_name)
end end
raise NoSuchKegError, name if resolve_formula(name) raise NoSuchKegError, name if resolve_formula(name)
raise FormulaOrCaskUnavailableError, name
end end
end end
private :load_formula_or_cask private :load_formula_or_cask
sig { params(name: String).returns(Formula) }
def resolve_formula(name) def resolve_formula(name)
Formulary.resolve(name, **{ spec: @override_spec, force_bottle: @force_bottle, flags: @flags }.compact) Formulary.resolve(name, **{ spec: @override_spec, force_bottle: @force_bottle, flags: @flags }.compact)
end end
@ -264,12 +303,16 @@ module Homebrew
sig { params(uniq: T::Boolean).returns(T::Array[Formula]) } sig { params(uniq: T::Boolean).returns(T::Array[Formula]) }
def to_resolved_formulae(uniq: true) def to_resolved_formulae(uniq: true)
@to_resolved_formulae ||= to_formulae_and_casks(only: :formula, method: :resolve, uniq:) @to_resolved_formulae ||= T.let(
.freeze to_formulae_and_casks(only: :formula, method: :resolve, uniq:).freeze,
T.nilable(T::Array[T.any(Formula, Keg, Cask::Cask)]),
)
T.cast(@to_resolved_formulae, T::Array[Formula])
end end
def to_resolved_formulae_to_casks(only: parent&.only_formula_or_cask) sig { params(only: T.nilable(Symbol)).returns([T::Array[Formula], T::Array[Cask::Cask]]) }
to_formulae_to_casks(only:, method: :resolve) def to_resolved_formulae_to_casks(only: parent.only_formula_or_cask)
T.cast(to_formulae_to_casks(only:, method: :resolve), [T::Array[Formula], T::Array[Cask::Cask]])
end end
LOCAL_PATH_REGEX = %r{^/|[.]|/$} LOCAL_PATH_REGEX = %r{^/|[.]|/$}
@ -280,8 +323,8 @@ module Homebrew
# If a cask and formula with the same name exist, includes both their paths # If a cask and formula with the same name exist, includes both their paths
# unless `only` is specified. # unless `only` is specified.
sig { params(only: T.nilable(Symbol), recurse_tap: T::Boolean).returns(T::Array[Pathname]) } sig { params(only: T.nilable(Symbol), recurse_tap: T::Boolean).returns(T::Array[Pathname]) }
def to_paths(only: parent&.only_formula_or_cask, recurse_tap: false) def to_paths(only: parent.only_formula_or_cask, recurse_tap: false)
@to_paths ||= {} @to_paths ||= T.let({}, T.nilable(T::Hash[T.nilable(Symbol), T::Array[Pathname]]))
@to_paths[only] ||= Homebrew.with_no_api_env_if_needed(@without_api) do @to_paths[only] ||= Homebrew.with_no_api_env_if_needed(@without_api) do
downcased_unique_named.flat_map do |name| downcased_unique_named.flat_map do |name|
path = Pathname(name).expand_path path = Pathname(name).expand_path
@ -328,67 +371,70 @@ module Homebrew
def to_default_kegs def to_default_kegs
require "missing_formula" require "missing_formula"
@to_default_kegs ||= begin @to_default_kegs ||= T.let(begin
to_formulae_and_casks(only: :formula, method: :default_kegs).freeze to_formulae_and_casks(only: :formula, method: :default_kegs).freeze
rescue NoSuchKegError => e rescue NoSuchKegError => e
if (reason = MissingFormula.suggest_command(e.name, "uninstall")) if (reason = MissingFormula.suggest_command(e.name, "uninstall"))
$stderr.puts reason $stderr.puts reason
end end
raise e raise e
end end, T.nilable(T::Array[T.any(Formula, Keg, Cask::Cask)]))
T.cast(@to_default_kegs, T::Array[Keg])
end end
sig { returns(T::Array[Keg]) } sig { returns(T::Array[Keg]) }
def to_latest_kegs def to_latest_kegs
require "missing_formula" require "missing_formula"
@to_latest_kegs ||= begin @to_latest_kegs ||= T.let(begin
to_formulae_and_casks(only: :formula, method: :latest_kegs).freeze to_formulae_and_casks(only: :formula, method: :latest_kegs).freeze
rescue NoSuchKegError => e rescue NoSuchKegError => e
if (reason = MissingFormula.suggest_command(e.name, "uninstall")) if (reason = MissingFormula.suggest_command(e.name, "uninstall"))
$stderr.puts reason $stderr.puts reason
end end
raise e raise e
end end, T.nilable(T::Array[T.any(Formula, Keg, Cask::Cask)]))
T.cast(@to_latest_kegs, T::Array[Keg])
end end
sig { returns(T::Array[Keg]) } sig { returns(T::Array[Keg]) }
def to_kegs def to_kegs
require "missing_formula" require "missing_formula"
@to_kegs ||= begin @to_kegs ||= T.let(begin
to_formulae_and_casks(only: :formula, method: :kegs).freeze to_formulae_and_casks(only: :formula, method: :kegs).freeze
rescue NoSuchKegError => e rescue NoSuchKegError => e
if (reason = MissingFormula.suggest_command(e.name, "uninstall")) if (reason = MissingFormula.suggest_command(e.name, "uninstall"))
$stderr.puts reason $stderr.puts reason
end end
raise e raise e
end end, T.nilable(T::Array[T.any(Formula, Keg, Cask::Cask)]))
T.cast(@to_kegs, T::Array[Keg])
end end
sig { sig {
params(only: T.nilable(Symbol), ignore_unavailable: T::Boolean, all_kegs: T.nilable(T::Boolean)) params(only: T.nilable(Symbol), ignore_unavailable: T::Boolean, all_kegs: T.nilable(T::Boolean))
.returns([T::Array[Keg], T::Array[Cask::Cask]]) .returns([T::Array[Keg], T::Array[Cask::Cask]])
} }
def to_kegs_to_casks(only: parent&.only_formula_or_cask, ignore_unavailable: false, all_kegs: nil) def to_kegs_to_casks(only: parent.only_formula_or_cask, ignore_unavailable: false, all_kegs: nil)
method = all_kegs ? :kegs : :default_kegs method = all_kegs ? :kegs : :default_kegs
@to_kegs_to_casks ||= {} @to_kegs_to_casks ||= T.let({}, T.nilable(T::Hash[T.nilable(Symbol), [T::Array[Keg], T::Array[Cask::Cask]]]))
@to_kegs_to_casks[method] ||= @to_kegs_to_casks[method] ||=
to_formulae_and_casks(only:, ignore_unavailable:, method:) T.cast(to_formulae_and_casks(only:, ignore_unavailable:, method:)
.partition { |o| o.is_a?(Keg) } .partition { |o| o.is_a?(Keg) }
.map(&:freeze).freeze .map(&:freeze).freeze, [T::Array[Keg], T::Array[Cask::Cask]])
end end
sig { returns(T::Array[Tap]) } sig { returns(T::Array[Tap]) }
def to_taps def to_taps
@to_taps ||= downcased_unique_named.map { |name| Tap.fetch name }.uniq.freeze @to_taps ||= T.let(downcased_unique_named.map { |name| Tap.fetch name }.uniq.freeze, T.nilable(T::Array[Tap]))
end end
sig { returns(T::Array[Tap]) } sig { returns(T::Array[Tap]) }
def to_installed_taps def to_installed_taps
@to_installed_taps ||= to_taps.each do |tap| @to_installed_taps ||= T.let(to_taps.each do |tap|
raise TapUnavailableError, tap.name unless tap.installed? raise TapUnavailableError, tap.name unless tap.installed?
end.uniq.freeze end.uniq.freeze, T.nilable(T::Array[Tap]))
end end
sig { returns(T::Array[String]) } sig { returns(T::Array[String]) }
@ -410,6 +456,7 @@ module Homebrew
end.uniq end.uniq
end end
sig { params(name: String).returns([Pathname, T::Array[Keg]]) }
def resolve_kegs(name) def resolve_kegs(name)
raise UsageError if name.blank? raise UsageError if name.blank?
@ -433,23 +480,26 @@ module Homebrew
[rack, kegs] [rack, kegs]
end end
sig { params(name: String).returns(Keg) }
def resolve_latest_keg(name) def resolve_latest_keg(name)
_, kegs = resolve_kegs(name) _, kegs = resolve_kegs(name)
# Return keg if it is the only installed keg # Return keg if it is the only installed keg
return kegs if kegs.length == 1 return kegs.fetch(0) if kegs.length == 1
stable_kegs = kegs.reject { |keg| keg.version.head? } stable_kegs = kegs.reject { |keg| keg.version.head? }
if stable_kegs.blank? latest_keg = if stable_kegs.empty?
return kegs.max_by do |keg| kegs.max_by do |keg|
[keg.tab.source_modified_time, keg.version.revision] [keg.tab.source_modified_time, keg.version.revision]
end end
else
stable_kegs.max_by(&:scheme_and_version)
end end
T.must(latest_keg)
stable_kegs.max_by(&:scheme_and_version)
end end
sig { params(name: String).returns(Keg) }
def resolve_default_keg(name) def resolve_default_keg(name)
rack, kegs = resolve_kegs(name) rack, kegs = resolve_kegs(name)
@ -459,7 +509,7 @@ module Homebrew
begin begin
return Keg.new(opt_prefix.resolved_path) if opt_prefix.symlink? && opt_prefix.directory? return Keg.new(opt_prefix.resolved_path) if opt_prefix.symlink? && opt_prefix.directory?
return Keg.new(linked_keg_ref.resolved_path) if linked_keg_ref.symlink? && linked_keg_ref.directory? return Keg.new(linked_keg_ref.resolved_path) if linked_keg_ref.symlink? && linked_keg_ref.directory?
return kegs.first if kegs.length == 1 return kegs.fetch(0) if kegs.length == 1
f = if name.include?("/") || File.exist?(name) f = if name.include?("/") || File.exist?(name)
Formulary.factory(name) Formulary.factory(name)
@ -484,6 +534,12 @@ module Homebrew
end end
end end
sig {
params(
ref: String, loaded_type: String,
package: T.any(T::Array[T.any(Formula, Keg)], Cask::Cask, Formula, Keg, NilClass)
).returns(String)
}
def package_conflicts_message(ref, loaded_type, package) def package_conflicts_message(ref, loaded_type, package)
message = "Treating #{ref} as a #{loaded_type}." message = "Treating #{ref} as a #{loaded_type}."
case package case package
@ -503,6 +559,7 @@ module Homebrew
message.freeze message.freeze
end end
sig { params(ref: String, loaded_type: String).void }
def warn_if_cask_conflicts(ref, loaded_type) def warn_if_cask_conflicts(ref, loaded_type)
available = true available = true
cask = begin cask = begin

View File

@ -162,12 +162,6 @@ module Homebrew
info_formula(obj) info_formula(obj)
when Cask::Cask when Cask::Cask
info_cask(obj) info_cask(obj)
when FormulaUnreadableError, FormulaClassUnavailableError,
TapFormulaUnreadableError, TapFormulaClassUnavailableError,
Cask::CaskUnreadableError
# We found the formula/cask, but failed to read it
$stderr.puts obj.backtrace if Homebrew::EnvConfig.developer?
ofail obj.message
when FormulaOrCaskUnavailableError when FormulaOrCaskUnavailableError
# The formula/cask could not be found # The formula/cask could not be found
ofail obj.message ofail obj.message

View File

@ -108,10 +108,7 @@ module Homebrew
sig { override.void } sig { override.void }
def run def run
formulae, casks = T.cast( formulae, casks = args.named.to_resolved_formulae_to_casks
args.named.to_resolved_formulae_to_casks,
[T::Array[Formula], T::Array[Cask::Cask]],
)
if args.build_from_source? if args.build_from_source?
unless DevelopmentTools.installed? unless DevelopmentTools.installed?

View File

@ -42,7 +42,7 @@ module Homebrew
end end
raise UsageError, "No marking option specified." if installed_on_request.nil? raise UsageError, "No marking option specified." if installed_on_request.nil?
formulae, casks = args.named.to_formulae_to_casks formulae, casks = T.cast(args.named.to_formulae_to_casks, [T::Array[Formula], T::Array[Cask::Cask]])
formulae_not_installed = formulae.reject(&:any_version_installed?) formulae_not_installed = formulae.reject(&:any_version_installed?)
casks_not_installed = casks.reject(&:installed?) casks_not_installed = casks.reject(&:installed?)
if formulae_not_installed.any? || casks_not_installed.any? if formulae_not_installed.any? || casks_not_installed.any?

View File

@ -58,7 +58,7 @@ module Homebrew
elsif args.no_named? elsif args.no_named?
puts Tap.installed.sort_by(&:name) puts Tap.installed.sort_by(&:name)
else else
tap = Tap.fetch(args.named.first) tap = Tap.fetch(args.named.fetch(0))
begin begin
tap.install clone_target: args.named.second, tap.install clone_target: args.named.second,
custom_remote: args.custom_remote?, custom_remote: args.custom_remote?,

View File

@ -72,7 +72,7 @@ module Homebrew
# Use the user's browser, too. # Use the user's browser, too.
ENV["BROWSER"] = EnvConfig.browser ENV["BROWSER"] = EnvConfig.browser
cask = args.named.to_casks.first cask = args.named.to_casks.fetch(0)
odie "This cask is not in a tap!" if cask.tap.blank? odie "This cask is not in a tap!" if cask.tap.blank?
odie "This cask's tap is not a Git repository!" unless cask.tap.git? odie "This cask's tap is not a Git repository!" unless cask.tap.git?

View File

@ -109,18 +109,19 @@ module Homebrew
odie "This formula is disabled!" if formula.disabled? odie "This formula is disabled!" if formula.disabled?
odie "This formula is deprecated and does not build!" if formula.deprecation_reason == :does_not_build odie "This formula is deprecated and does not build!" if formula.deprecation_reason == :does_not_build
odie "This formula is not in a tap!" if formula.tap.blank? tap = formula.tap
odie "This formula's tap is not a Git repository!" unless formula.tap.git? odie "This formula is not in a tap!" if tap.blank?
odie "This formula's tap is not a Git repository!" unless tap.git?
odie <<~EOS unless formula.tap.allow_bump?(formula.name) odie <<~EOS unless tap.allow_bump?(formula.name)
Whoops, the #{formula.name} formula has its version update Whoops, the #{formula.name} formula has its version update
pull requests automatically opened by BrewTestBot every ~3 hours! pull requests automatically opened by BrewTestBot every ~3 hours!
We'd still love your contributions, though, so try another one We'd still love your contributions, though, so try another one
that's not in the autobump list: that's not in the autobump list:
#{Formatter.url("#{formula.tap.remote}/blob/master/.github/autobump.txt")} #{Formatter.url("#{tap.remote}/blob/master/.github/autobump.txt")}
EOS EOS
odie "You have too many PRs open: close or merge some first!" if GitHub.too_many_open_prs?(formula.tap) odie "You have too many PRs open: close or merge some first!" if GitHub.too_many_open_prs?(tap)
formula_spec = formula.stable formula_spec = formula.stable
odie "#{formula}: no stable specification found!" if formula_spec.blank? odie "#{formula}: no stable specification found!" if formula_spec.blank?
@ -129,9 +130,9 @@ module Homebrew
# spamming during normal output. # spamming during normal output.
Homebrew.install_bundler_gems!(groups: ["audit", "style"]) unless args.no_audit? Homebrew.install_bundler_gems!(groups: ["audit", "style"]) unless args.no_audit?
tap_remote_repo = formula.tap.remote_repository tap_remote_repo = T.must(tap.remote_repository)
remote = "origin" remote = "origin"
remote_branch = formula.tap.git_repository.origin_branch_name remote_branch = tap.git_repository.origin_branch_name
previous_branch = "-" previous_branch = "-"
check_pull_requests(formula, tap_remote_repo, state: "open") check_pull_requests(formula, tap_remote_repo, state: "open")
@ -333,7 +334,7 @@ module Homebrew
alias_rename = alias_update_pair(formula, new_formula_version) alias_rename = alias_update_pair(formula, new_formula_version)
if alias_rename.present? if alias_rename.present?
ohai "Renaming alias #{alias_rename.first} to #{alias_rename.last}" ohai "Renaming alias #{alias_rename.first} to #{alias_rename.last}"
alias_rename.map! { |a| formula.tap.alias_dir/a } alias_rename.map! { |a| tap.alias_dir/a }
end end
unless args.dry_run? unless args.dry_run?
@ -389,7 +390,7 @@ module Homebrew
branch_name: "bump-#{formula.name}-#{new_formula_version}", branch_name: "bump-#{formula.name}-#{new_formula_version}",
commit_message: "#{formula.name} #{new_formula_version}", commit_message: "#{formula.name} #{new_formula_version}",
previous_branch:, previous_branch:,
tap: formula.tap, tap: tap,
tap_remote_repo:, tap_remote_repo:,
pr_message:, pr_message:,
} }

View File

@ -81,7 +81,7 @@ module Homebrew
sig { returns(Pathname) } sig { returns(Pathname) }
def create_cask def create_cask
url = args.named.first url = args.named.fetch(0)
name = if args.set_name.blank? name = if args.set_name.blank?
stem = Pathname.new(url).stem.rpartition("=").last stem = Pathname.new(url).stem.rpartition("=").last
print "Cask name [#{stem}]: " print "Cask name [#{stem}]: "
@ -179,7 +179,7 @@ module Homebrew
args.set_name, args.set_name,
args.set_version, args.set_version,
tap: args.tap, tap: args.tap,
url: args.named.first, url: args.named.fetch(0),
mode:, mode:,
license: args.set_license, license: args.set_license,
fetch: !args.no_fetch?, fetch: !args.no_fetch?,

View File

@ -39,9 +39,10 @@ module Homebrew
raise UsageError, "`--all-supported` is mutually exclusive to other arguments." raise UsageError, "`--all-supported` is mutually exclusive to other arguments."
end end
testing_formulae = args.named.first&.split(",").to_a testing_formulae = args.named.first&.split(",").to_a.map do |name|
testing_formulae.map! { |name| TestRunnerFormula.new(Formulary.factory(name), eval_all: args.eval_all?) } TestRunnerFormula.new(Formulary.factory(name), eval_all: args.eval_all?)
.freeze end
.freeze
deleted_formulae = args.named.second&.split(",").to_a.freeze deleted_formulae = args.named.second&.split(",").to_a.freeze
runner_matrix = GitHubRunnerMatrix.new(testing_formulae, deleted_formulae, runner_matrix = GitHubRunnerMatrix.new(testing_formulae, deleted_formulae,
all_supported: args.all_supported?, all_supported: args.all_supported?,

View File

@ -37,12 +37,12 @@ module Homebrew
if (tap_with_name = args.named.first&.then { Tap.with_formula_name(_1) }) if (tap_with_name = args.named.first&.then { Tap.with_formula_name(_1) })
source_tap, name = tap_with_name source_tap, name = tap_with_name
else else
name = args.named.first.downcase name = args.named.fetch(0).downcase
source_tap = CoreTap.instance source_tap = CoreTap.instance
end end
raise TapFormulaUnavailableError.new(source_tap, name) unless source_tap.installed? raise TapFormulaUnavailableError.new(source_tap, name) unless source_tap.installed?
destination_tap = Tap.fetch(args.named.second) destination_tap = Tap.fetch(args.named.fetch(1))
unless Homebrew::EnvConfig.developer? unless Homebrew::EnvConfig.developer?
odie "Cannot extract formula to homebrew/core!" if destination_tap.core_tap? odie "Cannot extract formula to homebrew/core!" if destination_tap.core_tap?
odie "Cannot extract formula to homebrew/cask!" if destination_tap.core_cask_tap? odie "Cannot extract formula to homebrew/cask!" if destination_tap.core_cask_tap?

View File

@ -93,7 +93,7 @@ module Homebrew
arg = "#{tap.default_remote}/pull/#{arg}" if arg.to_i.positive? arg = "#{tap.default_remote}/pull/#{arg}" if arg.to_i.positive?
url_match = arg.match HOMEBREW_PULL_OR_COMMIT_URL_REGEX url_match = arg.match HOMEBREW_PULL_OR_COMMIT_URL_REGEX
_, user, repo, pr = *url_match _, user, repo, pr = *url_match
odie "Not a GitHub pull request: #{arg}" unless pr odie "Not a GitHub pull request: #{arg}" if !user || !repo || !pr
git_repo = tap.git_repository git_repo = tap.git_repository
if !git_repo.default_origin_branch? && !args.branch_okay? && !args.no_commit? && !args.no_cherry_pick? if !git_repo.default_origin_branch? && !args.branch_okay? && !args.no_commit? && !args.no_cherry_pick?