require "shellwords" class UsageError < RuntimeError attr_reader :reason def initialize(reason = nil) @reason = reason end def to_s s = "Invalid usage" s += ": #{reason}" if reason s end end class FormulaUnspecifiedError < UsageError def initialize super "This command requires a formula argument" end end class KegUnspecifiedError < UsageError def initialize super "This command requires a keg argument" end end class MultipleVersionsInstalledError < RuntimeError attr_reader :name def initialize(name) @name = name super "#{name} has multiple installed versions" end end class NotAKegError < RuntimeError; end class NoSuchKegError < RuntimeError attr_reader :name def initialize(name) @name = name super "No such keg: #{HOMEBREW_CELLAR}/#{name}" end end 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 class MethodDeprecatedError < StandardError attr_accessor :issues_url end class FormulaUnavailableError < RuntimeError attr_reader :name attr_accessor :dependent def initialize(name) @name = name end def dependent_s "(dependency of #{dependent})" if dependent && dependent != name end def to_s "No available formula with the name \"#{name}\" #{dependent_s}" end end module FormulaClassUnavailableErrorModule attr_reader :path attr_reader :class_name attr_reader :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 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 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 module FormulaUnreadableErrorModule attr_reader :formula_error def to_s "#{name}: " + formula_error.to_s end end class FormulaUnreadableError < FormulaUnavailableError include FormulaUnreadableErrorModule def initialize(name, error) super(name) @formula_error = error end end 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 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 class TapFormulaUnreadableError < TapFormulaUnavailableError include FormulaUnreadableErrorModule def initialize(tap, name, error) super(tap, name) @formula_error = error end end 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 the formula. EOS end end 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 the formula or use its new name. EOS end end class TapUnavailableError < RuntimeError attr_reader :name def initialize(name) @name = name super <<~EOS No available tap #{name}. EOS end end class TapRemoteMismatchError < RuntimeError attr_reader :name attr_reader :expected_remote attr_reader :actual_remote def initialize(name, expected_remote, actual_remote) @name = name @expected_remote = expected_remote @actual_remote = actual_remote super <<~EOS Tap #{name} remote mismatch. #{expected_remote} != #{actual_remote} EOS end end class TapAlreadyTappedError < RuntimeError attr_reader :name def initialize(name) @name = name super <<~EOS Tap #{name} already tapped. EOS end end class TapAlreadyUnshallowError < RuntimeError attr_reader :name def initialize(name) @name = name super <<~EOS Tap #{name} already a full clone. EOS end end class TapPinStatusError < RuntimeError attr_reader :name, :pinned def initialize(name, pinned) @name = name @pinned = pinned super pinned ? "#{name} is already pinned." : "#{name} is already unpinned." end end 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 class FormulaInstallationAlreadyAttemptedError < RuntimeError def initialize(formula) super "Formula installation already attempted: #{formula.full_name}" end end 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 class FormulaConflictError < RuntimeError 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 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 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 python and python@2 is intentional, please add `:using => "python"` or `:using => "python@2"` to `virtualenv_install_with_resources` to resolve the ambiguity manually. EOS end end 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) rescue GitHub::RateLimitExceededError => e opoo e.message [] end def dump puts if ARGV.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" Homebrew.dump_build_env(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 unless issues&.empty? 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 by {FormulaInstaller#check_dependencies_bottled} and # {FormulaInstaller#install} if the formula or its dependencies are not bottled # and are being installed on a system without necessary build tools. class BuildToolsError < RuntimeError def initialize(formulae) super <<~EOS The following #{"formula".pluralize(formulae.count)} #{formulae.to_sentence} cannot be installed as #{"binary package".pluralize(formulae.count)} and must be built from source. #{DevelopmentTools.installation_instructions} EOS 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 message = <<~EOS The following #{flag_text}: #{flags.join(", ")} #{require_text} building tools, but none are installed. #{DevelopmentTools.installation_instructions} EOS message << <<~EOS if bottled Alternatively, remove the #{flag_text} to attempt bottle installation. 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 {ScpDownloadStrategy#fetch}. class ScpDownloadStrategyError < RuntimeError def initialize(cause) super "Download failed: #{cause}" end end # Raised by {#safe_system} in `utils.rb`. class ErrorDuringExecution < RuntimeError attr_reader :cmd attr_reader :status attr_reader :output def initialize(cmd, status:, output: nil) @cmd = cmd @status = status @output = output exitstatus = if status.respond_to?(:exitstatus) status.exitstatus else status end s = "Failure while executing; `#{cmd.shelljoin.gsub(/\\=/, "=")}` exited with #{exitstatus}." unless [*output].empty? 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 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, :hash_type def initialize(fn, expected, actual) @expected = expected @hash_type = expected.hash_type.to_s.upcase super <<~EOS #{@hash_type} mismatch Expected: #{expected} Actual: #{actual} Archive: #{fn} To retry an incomplete download, remove the file above. EOS end end class ResourceMissingError < ArgumentError def initialize(formula, resource) super "#{formula.full_name} does not define resource #{resource.inspect}" end end 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 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 attr_reader :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