Enable strict typing in NamedArgs
This commit is contained in:
		
							parent
							
								
									58e72b0a7f
								
							
						
					
					
						commit
						e1fdd2eda4
					
				@ -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
 | 
				
			||||||
 | 
				
			|||||||
@ -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
 | 
				
			||||||
 | 
				
			|||||||
@ -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?
 | 
				
			||||||
 | 
				
			|||||||
@ -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?
 | 
				
			||||||
 | 
				
			|||||||
@ -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?,
 | 
				
			||||||
 | 
				
			|||||||
@ -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?
 | 
				
			||||||
 | 
				
			|||||||
@ -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:,
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
				
			|||||||
@ -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?,
 | 
				
			||||||
 | 
				
			|||||||
@ -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?,
 | 
				
			||||||
 | 
				
			|||||||
@ -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?
 | 
				
			||||||
 | 
				
			|||||||
@ -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?
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user