801 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			801 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: false
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| require "shellwords"
 | |
| require "utils"
 | |
| 
 | |
| # Raised when a command is used wrong.
 | |
| class UsageError < RuntimeError
 | |
|   extend T::Sig
 | |
| 
 | |
|   attr_reader :reason
 | |
| 
 | |
|   def initialize(reason = nil)
 | |
|     super
 | |
| 
 | |
|     @reason = reason
 | |
|   end
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def to_s
 | |
|     s = "Invalid usage"
 | |
|     s += ": #{reason}" if reason
 | |
|     s
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a command expects a formula and none was specified.
 | |
| class FormulaUnspecifiedError < UsageError
 | |
|   def initialize
 | |
|     super "this command requires a formula argument"
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a command expects a formula or cask and none was specified.
 | |
| class FormulaOrCaskUnspecifiedError < UsageError
 | |
|   def initialize
 | |
|     super "this command requires a formula or cask argument"
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a command expects a keg and none was specified.
 | |
| class KegUnspecifiedError < UsageError
 | |
|   def initialize
 | |
|     super "this command requires a keg argument"
 | |
|   end
 | |
| end
 | |
| 
 | |
| class MultipleVersionsInstalledError < RuntimeError; end
 | |
| 
 | |
| class NotAKegError < RuntimeError; end
 | |
| 
 | |
| # Raised when a keg doesn't exist.
 | |
| class NoSuchKegError < RuntimeError
 | |
|   attr_reader :name
 | |
| 
 | |
|   def initialize(name)
 | |
|     @name = name
 | |
|     super "No such keg: #{HOMEBREW_CELLAR}/#{name}"
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when an invalid attribute is used in a formula.
 | |
| class FormulaValidationError < StandardError
 | |
|   attr_reader :attr, :formula
 | |
| 
 | |
|   def initialize(formula, attr, value)
 | |
|     @attr = attr
 | |
|     @formula = formula
 | |
|     super "invalid attribute for formula '#{formula}': #{attr} (#{value.inspect})"
 | |
|   end
 | |
| end
 | |
| 
 | |
| class FormulaSpecificationError < StandardError; end
 | |
| 
 | |
| # Raised when a deprecated method is used.
 | |
| #
 | |
| # @api private
 | |
| class MethodDeprecatedError < StandardError
 | |
|   attr_accessor :issues_url
 | |
| end
 | |
| 
 | |
| # Raised when neither a formula nor a cask with the given name is available.
 | |
| class FormulaOrCaskUnavailableError < RuntimeError
 | |
|   extend T::Sig
 | |
| 
 | |
|   attr_reader :name
 | |
| 
 | |
|   def initialize(name)
 | |
|     super()
 | |
| 
 | |
|     @name = name
 | |
|   end
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def did_you_mean
 | |
|     similar_formula_names = Formula.fuzzy_search(name)
 | |
|     return "" if similar_formula_names.blank?
 | |
| 
 | |
|     "Did you mean #{similar_formula_names.to_sentence two_words_connector: " or ", last_word_connector: " or "}?"
 | |
|   end
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def to_s
 | |
|     "No available formula or cask with the name \"#{name}\". #{did_you_mean}".strip
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula or cask in a specific tap is not available.
 | |
| class TapFormulaOrCaskUnavailableError < FormulaOrCaskUnavailableError
 | |
|   extend T::Sig
 | |
| 
 | |
|   attr_reader :tap
 | |
| 
 | |
|   def initialize(tap, name)
 | |
|     super "#{tap}/#{name}"
 | |
|     @tap = tap
 | |
|   end
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def to_s
 | |
|     s = super
 | |
|     s += "\nPlease tap it and then try again: brew tap #{tap}" unless tap.installed?
 | |
|     s
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula is not available.
 | |
| class FormulaUnavailableError < FormulaOrCaskUnavailableError
 | |
|   extend T::Sig
 | |
| 
 | |
|   attr_accessor :dependent
 | |
| 
 | |
|   sig { returns(T.nilable(String)) }
 | |
|   def dependent_s
 | |
|     " (dependency of #{dependent})" if dependent && dependent != name
 | |
|   end
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def to_s
 | |
|     "No available formula with the name \"#{name}\"#{dependent_s}. #{did_you_mean}".strip
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Shared methods for formula class errors.
 | |
| #
 | |
| # @api private
 | |
| module FormulaClassUnavailableErrorModule
 | |
|   extend T::Sig
 | |
| 
 | |
|   attr_reader :path, :class_name, :class_list
 | |
| 
 | |
|   def to_s
 | |
|     s = super
 | |
|     s += "\nIn formula file: #{path}"
 | |
|     s += "\nExpected to find class #{class_name}, but #{class_list_s}."
 | |
|     s
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def class_list_s
 | |
|     formula_class_list = class_list.select { |klass| klass < Formula }
 | |
|     if class_list.empty?
 | |
|       "found no classes"
 | |
|     elsif formula_class_list.empty?
 | |
|       "only found: #{format_list(class_list)} (not derived from Formula!)"
 | |
|     else
 | |
|       "only found: #{format_list(formula_class_list)}"
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def format_list(class_list)
 | |
|     class_list.map { |klass| klass.name.split("::").last }.join(", ")
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula does not contain a formula class.
 | |
| class FormulaClassUnavailableError < FormulaUnavailableError
 | |
|   include FormulaClassUnavailableErrorModule
 | |
| 
 | |
|   def initialize(name, path, class_name, class_list)
 | |
|     @path = path
 | |
|     @class_name = class_name
 | |
|     @class_list = class_list
 | |
|     super name
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Shared methods for formula unreadable errors.
 | |
| #
 | |
| # @api private
 | |
| module FormulaUnreadableErrorModule
 | |
|   extend T::Sig
 | |
| 
 | |
|   attr_reader :formula_error
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def to_s
 | |
|     "#{name}: " + formula_error.to_s
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula is unreadable.
 | |
| class FormulaUnreadableError < FormulaUnavailableError
 | |
|   include FormulaUnreadableErrorModule
 | |
| 
 | |
|   def initialize(name, error)
 | |
|     super(name)
 | |
|     @formula_error = error
 | |
|     set_backtrace(error.backtrace)
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula in a specific tap is unavailable.
 | |
| class TapFormulaUnavailableError < FormulaUnavailableError
 | |
|   attr_reader :tap, :user, :repo
 | |
| 
 | |
|   def initialize(tap, name)
 | |
|     @tap = tap
 | |
|     @user = tap.user
 | |
|     @repo = tap.repo
 | |
|     super "#{tap}/#{name}"
 | |
|   end
 | |
| 
 | |
|   def to_s
 | |
|     s = super
 | |
|     s += "\nPlease tap it and then try again: brew tap #{tap}" unless tap.installed?
 | |
|     s
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula in a the core tap is unavailable.
 | |
| class CoreTapFormulaUnavailableError < TapFormulaUnavailableError
 | |
|   def initialize(name)
 | |
|     super CoreTap.instance, name
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula in a specific tap does not contain a formula class.
 | |
| class TapFormulaClassUnavailableError < TapFormulaUnavailableError
 | |
|   include FormulaClassUnavailableErrorModule
 | |
| 
 | |
|   attr_reader :tap
 | |
| 
 | |
|   def initialize(tap, name, path, class_name, class_list)
 | |
|     @path = path
 | |
|     @class_name = class_name
 | |
|     @class_list = class_list
 | |
|     super tap, name
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula in a specific tap is unreadable.
 | |
| class TapFormulaUnreadableError < TapFormulaUnavailableError
 | |
|   include FormulaUnreadableErrorModule
 | |
| 
 | |
|   def initialize(tap, name, error)
 | |
|     super(tap, name)
 | |
|     @formula_error = error
 | |
|     set_backtrace(error.backtrace)
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula with the same name is found in multiple taps.
 | |
| class TapFormulaAmbiguityError < RuntimeError
 | |
|   attr_reader :name, :paths, :formulae
 | |
| 
 | |
|   def initialize(name, paths)
 | |
|     @name = name
 | |
|     @paths = paths
 | |
|     @formulae = paths.map do |path|
 | |
|       "#{Tap.from_path(path).name}/#{path.basename(".rb")}"
 | |
|     end
 | |
| 
 | |
|     super <<~EOS
 | |
|       Formulae found in multiple taps: #{formulae.map { |f| "\n       * #{f}" }.join}
 | |
| 
 | |
|       Please use the fully-qualified name (e.g. #{formulae.first}) to refer to the formula.
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula's old name in a specific tap is found in multiple taps.
 | |
| class TapFormulaWithOldnameAmbiguityError < RuntimeError
 | |
|   attr_reader :name, :possible_tap_newname_formulae, :taps
 | |
| 
 | |
|   def initialize(name, possible_tap_newname_formulae)
 | |
|     @name = name
 | |
|     @possible_tap_newname_formulae = possible_tap_newname_formulae
 | |
| 
 | |
|     @taps = possible_tap_newname_formulae.map do |newname|
 | |
|       newname =~ HOMEBREW_TAP_FORMULA_REGEX
 | |
|       "#{Regexp.last_match(1)}/#{Regexp.last_match(2)}"
 | |
|     end
 | |
| 
 | |
|     super <<~EOS
 | |
|       Formulae with '#{name}' old name found in multiple taps: #{taps.map { |t| "\n       * #{t}" }.join}
 | |
| 
 | |
|       Please use the fully-qualified name (e.g. #{taps.first}/#{name}) to refer to the formula or use its new name.
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a tap is unavailable.
 | |
| class TapUnavailableError < RuntimeError
 | |
|   attr_reader :name
 | |
| 
 | |
|   def initialize(name)
 | |
|     @name = name
 | |
| 
 | |
|     super <<~EOS
 | |
|       No available tap #{name}.
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a tap's remote does not match the actual remote.
 | |
| class TapRemoteMismatchError < RuntimeError
 | |
|   attr_reader :name, :expected_remote, :actual_remote
 | |
| 
 | |
|   def initialize(name, expected_remote, actual_remote)
 | |
|     @name = name
 | |
|     @expected_remote = expected_remote
 | |
|     @actual_remote = actual_remote
 | |
| 
 | |
|     super message
 | |
|   end
 | |
| 
 | |
|   def message
 | |
|     <<~EOS
 | |
|       Tap #{name} remote mismatch.
 | |
|       #{expected_remote} != #{actual_remote}
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when the remote of Homebrew/core does not match HOMEBREW_CORE_GIT_REMOTE.
 | |
| class TapCoreRemoteMismatchError < TapRemoteMismatchError
 | |
|   def message
 | |
|     <<~EOS
 | |
|       Tap #{name} remote does mot match HOMEBREW_CORE_GIT_REMOTE.
 | |
|       #{expected_remote} != #{actual_remote}
 | |
|       Please set HOMEBREW_CORE_GIT_REMOTE="#{actual_remote}" and run `brew update` instead.
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a tap is already installed.
 | |
| class TapAlreadyTappedError < RuntimeError
 | |
|   attr_reader :name
 | |
| 
 | |
|   def initialize(name)
 | |
|     @name = name
 | |
| 
 | |
|     super <<~EOS
 | |
|       Tap #{name} already tapped.
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when run `brew tap --custom-remote` without a remote URL.
 | |
| class TapNoCustomRemoteError < RuntimeError
 | |
|   attr_reader :name
 | |
| 
 | |
|   def initialize(name)
 | |
|     @name = name
 | |
| 
 | |
|     super <<~EOS
 | |
|       Tap #{name} with option `--custom-remote` but without a remote URL.
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when another Homebrew operation is already in progress.
 | |
| class OperationInProgressError < RuntimeError
 | |
|   def initialize(name)
 | |
|     message = <<~EOS
 | |
|       Operation already in progress for #{name}
 | |
|       Another active Homebrew process is already using #{name}.
 | |
|       Please wait for it to finish or terminate it to continue.
 | |
|     EOS
 | |
| 
 | |
|     super message
 | |
|   end
 | |
| end
 | |
| 
 | |
| class CannotInstallFormulaError < RuntimeError; end
 | |
| 
 | |
| # Raised when a formula installation was already attempted.
 | |
| class FormulaInstallationAlreadyAttemptedError < RuntimeError
 | |
|   def initialize(formula)
 | |
|     super "Formula installation already attempted: #{formula.full_name}"
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when there are unsatisfied requirements.
 | |
| class UnsatisfiedRequirements < RuntimeError
 | |
|   def initialize(reqs)
 | |
|     if reqs.length == 1
 | |
|       super "An unsatisfied requirement failed this build."
 | |
|     else
 | |
|       super "Unsatisfied requirements failed this build."
 | |
|     end
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a formula conflicts with another one.
 | |
| class FormulaConflictError < RuntimeError
 | |
|   extend T::Sig
 | |
| 
 | |
|   attr_reader :formula, :conflicts
 | |
| 
 | |
|   def initialize(formula, conflicts)
 | |
|     @formula = formula
 | |
|     @conflicts = conflicts
 | |
|     super message
 | |
|   end
 | |
| 
 | |
|   def conflict_message(conflict)
 | |
|     message = []
 | |
|     message << "  #{conflict.name}"
 | |
|     message << ": because #{conflict.reason}" if conflict.reason
 | |
|     message.join
 | |
|   end
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def message
 | |
|     message = []
 | |
|     message << "Cannot install #{formula.full_name} because conflicting formulae are installed."
 | |
|     message.concat conflicts.map { |c| conflict_message(c) } << ""
 | |
|     message << <<~EOS
 | |
|       Please `brew unlink #{conflicts.map(&:name) * " "}` before continuing.
 | |
| 
 | |
|       Unlinking removes a formula's symlinks from #{HOMEBREW_PREFIX}. You can
 | |
|       link the formula again after the install finishes. You can --force this
 | |
|       install, but the build may fail or cause obscure side effects in the
 | |
|       resulting software.
 | |
|     EOS
 | |
|     message.join("\n")
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raise when the Python version cannot be detected automatically.
 | |
| class FormulaUnknownPythonError < RuntimeError
 | |
|   def initialize(formula)
 | |
|     super <<~EOS
 | |
|       The version of Python to use with the virtualenv in the `#{formula.full_name}` formula
 | |
|       cannot be guessed automatically because a recognised Python dependency could not be found.
 | |
| 
 | |
|       If you are using a non-standard Python dependency, please add `:using => "python@x.y"`
 | |
|       to 'virtualenv_install_with_resources' to resolve the issue manually.
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raise when two Python versions are detected simultaneously.
 | |
| class FormulaAmbiguousPythonError < RuntimeError
 | |
|   def initialize(formula)
 | |
|     super <<~EOS
 | |
|       The version of Python to use with the virtualenv in the `#{formula.full_name}` formula
 | |
|       cannot be guessed automatically.
 | |
| 
 | |
|       If the simultaneous use of multiple Pythons is intentional, please add `:using => "python@x.y"`
 | |
|       to 'virtualenv_install_with_resources' to resolve the ambiguity manually.
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when an error occurs during a formula build.
 | |
| class BuildError < RuntimeError
 | |
|   attr_reader :cmd, :args, :env
 | |
|   attr_accessor :formula, :options
 | |
| 
 | |
|   def initialize(formula, cmd, args, env)
 | |
|     @formula = formula
 | |
|     @cmd = cmd
 | |
|     @args = args
 | |
|     @env = env
 | |
|     pretty_args = Array(args).map { |arg| arg.to_s.gsub " ", "\\ " }.join(" ")
 | |
|     super "Failed executing: #{cmd} #{pretty_args}".strip
 | |
|   end
 | |
| 
 | |
|   def issues
 | |
|     @issues ||= fetch_issues
 | |
|   end
 | |
| 
 | |
|   def fetch_issues
 | |
|     GitHub.issues_for_formula(formula.name, tap: formula.tap, state: "open")
 | |
|   rescue GitHub::API::RateLimitExceededError => e
 | |
|     opoo e.message
 | |
|     []
 | |
|   end
 | |
| 
 | |
|   def dump(verbose: false)
 | |
|     puts
 | |
| 
 | |
|     if verbose
 | |
|       require "system_config"
 | |
|       require "build_environment"
 | |
| 
 | |
|       ohai "Formula"
 | |
|       puts "Tap: #{formula.tap}" if formula.tap?
 | |
|       puts "Path: #{formula.path}"
 | |
|       ohai "Configuration"
 | |
|       SystemConfig.dump_verbose_config
 | |
|       ohai "ENV"
 | |
|       BuildEnvironment.dump env
 | |
|       puts
 | |
|       onoe "#{formula.full_name} #{formula.version} did not build"
 | |
|       unless (logs = Dir["#{formula.logs}/*"]).empty?
 | |
|         puts "Logs:"
 | |
|         puts logs.map { |fn| "     #{fn}" }.join("\n")
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     if formula.tap && defined?(OS::ISSUES_URL)
 | |
|       if formula.tap.official?
 | |
|         puts Formatter.error(Formatter.url(OS::ISSUES_URL), label: "READ THIS")
 | |
|       elsif (issues_url = formula.tap.issues_url)
 | |
|         puts <<~EOS
 | |
|           If reporting this issue please do so at (not Homebrew/brew or Homebrew/core):
 | |
|             #{Formatter.url(issues_url)}
 | |
|         EOS
 | |
|       else
 | |
|         puts <<~EOS
 | |
|           If reporting this issue please do so to (not Homebrew/brew or Homebrew/core):
 | |
|             #{formula.tap}
 | |
|         EOS
 | |
|       end
 | |
|     else
 | |
|       puts <<~EOS
 | |
|         Do not report this issue to Homebrew/brew or Homebrew/core!
 | |
|       EOS
 | |
|     end
 | |
| 
 | |
|     puts
 | |
| 
 | |
|     if issues.present?
 | |
|       puts "These open issues may also help:"
 | |
|       puts issues.map { |i| "#{i["title"]} #{i["html_url"]}" }.join("\n")
 | |
|     end
 | |
| 
 | |
|     require "diagnostic"
 | |
|     checks = Homebrew::Diagnostic::Checks.new
 | |
|     checks.build_error_checks.each do |check|
 | |
|       out = checks.send(check)
 | |
|       next if out.nil?
 | |
| 
 | |
|       puts
 | |
|       ofail out
 | |
|     end
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised if the formula or its dependencies are not bottled and are being
 | |
| # installed in a situation where a bottle is required.
 | |
| class UnbottledError < RuntimeError
 | |
|   def initialize(formulae)
 | |
|     msg = +<<~EOS
 | |
|       The following #{"formula".pluralize(formulae.count)} cannot be installed from #{"bottle".pluralize(formulae.count)} and must be
 | |
|       built from source.
 | |
|         #{formulae.to_sentence}
 | |
|     EOS
 | |
|     msg += "#{DevelopmentTools.installation_instructions}\n" unless DevelopmentTools.installed?
 | |
|     msg.freeze
 | |
|     super(msg)
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised by Homebrew.install, Homebrew.reinstall, and Homebrew.upgrade
 | |
| # if the user passes any flags/environment that would case a bottle-only
 | |
| # installation on a system without build tools to fail.
 | |
| class BuildFlagsError < RuntimeError
 | |
|   def initialize(flags, bottled: true)
 | |
|     if flags.length > 1
 | |
|       flag_text = "flags"
 | |
|       require_text = "require"
 | |
|     else
 | |
|       flag_text = "flag"
 | |
|       require_text = "requires"
 | |
|     end
 | |
| 
 | |
|     bottle_text = if bottled
 | |
|       <<~EOS
 | |
|         Alternatively, remove the #{flag_text} to attempt bottle installation.
 | |
|       EOS
 | |
|     end
 | |
| 
 | |
|     message = <<~EOS
 | |
|       The following #{flag_text}:
 | |
|         #{flags.join(", ")}
 | |
|       #{require_text} building tools, but none are installed.
 | |
|       #{DevelopmentTools.installation_instructions} #{bottle_text}
 | |
|     EOS
 | |
| 
 | |
|     super message
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised by {CompilerSelector} if the formula fails with all of
 | |
| # the compilers available on the user's system.
 | |
| class CompilerSelectionError < RuntimeError
 | |
|   def initialize(formula)
 | |
|     super <<~EOS
 | |
|       #{formula.full_name} cannot be built with any available compilers.
 | |
|       #{DevelopmentTools.custom_installation_instructions}
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised in {Resource#fetch}.
 | |
| class DownloadError < RuntimeError
 | |
|   def initialize(resource, cause)
 | |
|     super <<~EOS
 | |
|       Failed to download resource #{resource.download_name.inspect}
 | |
|       #{cause.message}
 | |
|     EOS
 | |
|     set_backtrace(cause.backtrace)
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised in {CurlDownloadStrategy#fetch}.
 | |
| class CurlDownloadStrategyError < RuntimeError
 | |
|   def initialize(url)
 | |
|     case url
 | |
|     when %r{^file://(.+)}
 | |
|       super "File does not exist: #{Regexp.last_match(1)}"
 | |
|     else
 | |
|       super "Download failed: #{url}"
 | |
|     end
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised in {HomebrewCurlDownloadStrategy#fetch}.
 | |
| class HomebrewCurlDownloadStrategyError < CurlDownloadStrategyError
 | |
|   def initialize(url)
 | |
|     super "Homebrew-installed `curl` is not installed for: #{url}"
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised by {Kernel#safe_system} in `utils.rb`.
 | |
| class ErrorDuringExecution < RuntimeError
 | |
|   extend T::Sig
 | |
| 
 | |
|   attr_reader :cmd, :status, :output
 | |
| 
 | |
|   def initialize(cmd, status:, output: nil, secrets: [])
 | |
|     @cmd = cmd
 | |
|     @status = status
 | |
|     @output = output
 | |
| 
 | |
|     raise ArgumentError, "Status cannot be nil." if status.nil?
 | |
| 
 | |
|     exitstatus = case status
 | |
|     when Integer
 | |
|       status
 | |
|     when Hash
 | |
|       status["exitstatus"]
 | |
|     else
 | |
|       status.exitstatus
 | |
|     end
 | |
| 
 | |
|     termsig = case status
 | |
|     when Integer
 | |
|       nil
 | |
|     when Hash
 | |
|       status["termsig"]
 | |
|     else
 | |
|       status.termsig
 | |
|     end
 | |
| 
 | |
|     redacted_cmd = redact_secrets(cmd.shelljoin.gsub('\=', "="), secrets)
 | |
| 
 | |
|     reason = if exitstatus
 | |
|       "exited with #{exitstatus}"
 | |
|     elsif termsig
 | |
|       "was terminated by uncaught signal #{Signal.signame(termsig)}"
 | |
|     else
 | |
|       raise ArgumentError, "Status neither has `exitstatus` nor `termsig`."
 | |
|     end
 | |
| 
 | |
|     s = +"Failure while executing; `#{redacted_cmd}` #{reason}."
 | |
| 
 | |
|     if Array(output).present?
 | |
|       format_output_line = lambda do |type_line|
 | |
|         type, line = *type_line
 | |
|         if type == :stderr
 | |
|           Formatter.error(line)
 | |
|         else
 | |
|           line
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       s << " Here's the output:\n"
 | |
|       s << output.map(&format_output_line).join
 | |
|       s << "\n" unless s.end_with?("\n")
 | |
|     end
 | |
| 
 | |
|     super s.freeze
 | |
|   end
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def stderr
 | |
|     Array(output).select { |type,| type == :stderr }.map(&:last).join
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised by {Pathname#verify_checksum} when "expected" is nil or empty.
 | |
| class ChecksumMissingError < ArgumentError; end
 | |
| 
 | |
| # Raised by {Pathname#verify_checksum} when verification fails.
 | |
| class ChecksumMismatchError < RuntimeError
 | |
|   attr_reader :expected
 | |
| 
 | |
|   def initialize(path, expected, actual)
 | |
|     @expected = expected
 | |
| 
 | |
|     super <<~EOS
 | |
|       SHA256 mismatch
 | |
|       Expected: #{Formatter.success(expected.to_s)}
 | |
|         Actual: #{Formatter.error(actual.to_s)}
 | |
|           File: #{path}
 | |
|       To retry an incomplete download, remove the file above.
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a resource is missing.
 | |
| class ResourceMissingError < ArgumentError
 | |
|   def initialize(formula, resource)
 | |
|     super "#{formula.full_name} does not define resource #{resource.inspect}"
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a resource is specified multiple times.
 | |
| class DuplicateResourceError < ArgumentError
 | |
|   def initialize(resource)
 | |
|     super "Resource #{resource.inspect} is defined more than once"
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a single patch file is not found and apply hasn't been specified.
 | |
| class MissingApplyError < RuntimeError; end
 | |
| 
 | |
| # Raised when a bottle does not contain a formula file.
 | |
| class BottleFormulaUnavailableError < RuntimeError
 | |
|   def initialize(bottle_path, formula_path)
 | |
|     super <<~EOS
 | |
|       This bottle does not contain the formula file:
 | |
|         #{bottle_path}
 | |
|         #{formula_path}
 | |
|     EOS
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a child process sends us an exception over its error pipe.
 | |
| class ChildProcessError < RuntimeError
 | |
|   attr_reader :inner, :inner_class
 | |
| 
 | |
|   def initialize(inner)
 | |
|     @inner = inner
 | |
|     @inner_class = Object.const_get inner["json_class"]
 | |
| 
 | |
|     super <<~EOS
 | |
|       An exception occurred within a child process:
 | |
|         #{inner_class}: #{inner["m"]}
 | |
|     EOS
 | |
| 
 | |
|     # Clobber our real (but irrelevant) backtrace with that of the inner exception.
 | |
|     set_backtrace inner["b"]
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when a macOS version is unsupported.
 | |
| class MacOSVersionError < RuntimeError
 | |
|   attr_reader :version
 | |
| 
 | |
|   def initialize(version)
 | |
|     @version = version
 | |
|     super "unknown or unsupported macOS version: #{version.inspect}"
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when `detected_perl_shebang` etc cannot detect the shebang.
 | |
| class ShebangDetectionError < RuntimeError
 | |
|   def initialize(type, reason)
 | |
|     super "Cannot detect #{type} shebang: #{reason}."
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Raised when one or more formulae have cyclic dependencies.
 | |
| class CyclicDependencyError < RuntimeError
 | |
|   def initialize(strongly_connected_components)
 | |
|     super <<~EOS
 | |
|       The following packages contain cyclic dependencies:
 | |
|         #{strongly_connected_components.select { |packages| packages.count > 1 }.map(&:to_sentence).join("\n  ")}
 | |
|     EOS
 | |
|   end
 | |
| end
 | 
