Skip to content
Snippets Groups Projects
formulary.rb 13.4 KiB
Newer Older
require "tap"
# The Formulary is responsible for creating instances of Formula.
# It is not meant to be used directly from formulae.
Adam Vandenberg's avatar
Adam Vandenberg committed

module Formulary
  FORMULAE = {}
  def self.formula_class_defined?(path)
    FORMULAE.key?(path)
Adam Vandenberg's avatar
Adam Vandenberg committed
  end
  def self.formula_class_get(path)
    FORMULAE.fetch(path)
Adam Vandenberg's avatar
Adam Vandenberg committed
  end
  def self.load_formula(name, path, contents, namespace)
    if ENV["HOMEBREW_DISABLE_LOAD_FORMULA"]
      raise "Formula loading disabled by HOMEBREW_DISABLE_LOAD_FORMULA!"
    end

    mod = Module.new
    const_set(namespace, mod)
    begin
      mod.module_eval(contents, path)
    rescue ScriptError => e
      raise FormulaUnreadableError.new(name, e)
    end
    class_name = class_s(name)

    begin
      mod.const_get(class_name)
    rescue NameError => original_exception
      class_list = mod.constants
                      .map { |const_name| mod.const_get(const_name) }
                      .select { |const| const.is_a?(Class) }
      e = FormulaClassUnavailableError.new(name, path, class_name, class_list)
      raise e, "", original_exception.backtrace
  def self.load_formula_from_path(name, path)
    contents = path.open("r") { |f| ensure_utf8_encoding(f).read }
    namespace = "FormulaNamespace#{Digest::MD5.hexdigest(path.to_s)}"
    klass = load_formula(name, path, contents, namespace)
    FORMULAE[path] = klass
  end

  if IO.method_defined?(:set_encoding)
    def self.ensure_utf8_encoding(io)
      io.set_encoding(Encoding::UTF_8)
    end
  else
    def self.ensure_utf8_encoding(io)
BrewTestBot's avatar
BrewTestBot committed
  def self.class_s(name)
    class_name = name.capitalize
    class_name.gsub!(/[-_.\s]([a-zA-Z0-9])/) { $1.upcase }
    class_name.tr!("+", "x")
    class_name.sub!(/(.)@(\d)/, "\\1AT\\2")
Adam Vandenberg's avatar
Adam Vandenberg committed
  # A FormulaLoader returns instances of formulae.
  # Subclasses implement loaders for particular sources of formulae.
  class FormulaLoader
    # The formula's name
    attr_reader :name
    # The formula's ruby file's path or filename
    attr_reader :path
    # The name used to install the formula
    attr_reader :alias_path

    def initialize(name, path)
      @name = name
      @path = path.resolved_path
Adam Vandenberg's avatar
Adam Vandenberg committed

    # Gets the formula instance.
    #
    # `alias_path` can be overridden here in case an alias was used to refer to
    # a formula that was loaded in another way.
    def get_formula(spec, alias_path: nil)
      klass.new(name, path, spec, alias_path: alias_path || self.alias_path)
Adam Vandenberg's avatar
Adam Vandenberg committed

    def klass
      load_file unless Formulary.formula_class_defined?(path)
      Formulary.formula_class_get(path)
Adam Vandenberg's avatar
Adam Vandenberg committed
    end
Jack Nagel's avatar
Jack Nagel committed

    private

    def load_file
      $stderr.puts "#{$0} (#{self.class.name}): loading #{path}" if ARGV.debug?
      raise FormulaUnavailableError, name unless path.file?
      Formulary.load_formula_from_path(name, path)
Jack Nagel's avatar
Jack Nagel committed
    end
Adam Vandenberg's avatar
Adam Vandenberg committed
  end
Adam Vandenberg's avatar
Adam Vandenberg committed
  # Loads formulae from bottles.
  class BottleLoader < FormulaLoader
BrewTestBot's avatar
BrewTestBot committed
    def initialize(bottle_name)
Adam Vandenberg's avatar
Adam Vandenberg committed
      @bottle_filename = Pathname(bottle_name).realpath
      name, full_name = Utils::Bottles.resolve_formula_names @bottle_filename
      super name, Formulary.path(full_name)
Adam Vandenberg's avatar
Adam Vandenberg committed
    end
    def get_formula(spec, alias_path: nil)
      formula.local_bottle_path = @bottle_filename
      formula_version = formula.pkg_version
      bottle_version =  Utils::Bottles.resolve_version(@bottle_filename)
      unless formula_version == bottle_version
        raise BottleVersionMismatchError.new(@bottle_filename, bottle_version, formula, formula_version)
      end
Adam Vandenberg's avatar
Adam Vandenberg committed
    end
  end
  class AliasLoader < FormulaLoader
BrewTestBot's avatar
BrewTestBot committed
    def initialize(alias_path)
      path = alias_path.resolved_path
      name = path.basename(".rb").to_s
      super name, path
      @alias_path = alias_path.to_s
Adam Vandenberg's avatar
Adam Vandenberg committed
  # Loads formulae from disk using a path
  class FromPathLoader < FormulaLoader
BrewTestBot's avatar
BrewTestBot committed
    def initialize(path)
      path = Pathname.new(path).expand_path
      super path.basename(".rb").to_s, path
  # Loads formulae from URLs
Adam Vandenberg's avatar
Adam Vandenberg committed
  class FromUrlLoader < FormulaLoader
    attr_reader :url

BrewTestBot's avatar
BrewTestBot committed
    def initialize(url)
Adam Vandenberg's avatar
Adam Vandenberg committed
      @url = url
      uri = URI(url)
      formula = File.basename(uri.path, ".rb")
      super formula, HOMEBREW_CACHE_FORMULA/File.basename(uri.path)
    def load_file
      HOMEBREW_CACHE_FORMULA.mkpath
      FileUtils.rm_f(path)
      curl url, "-o", path
    rescue MethodDeprecatedError => e
      if url =~ %r{github.com/([\w-]+)/homebrew-([\w-]+)/}
        e.issues_url = "https://github.com/#{$1}/homebrew-#{$2}/issues/new"
      end
      raise
Adam Vandenberg's avatar
Adam Vandenberg committed
  end
Adam Vandenberg's avatar
Adam Vandenberg committed
  # Loads tapped formulae.
  class TapLoader < FormulaLoader
    attr_reader :tap
    def initialize(tapped_name, from: nil)
      warn = ![:keg, :rack].include?(from)
      name, path = formula_name_path(tapped_name, warn: warn)
      super name, path
    end

    def formula_name_path(tapped_name, warn: true)
Jack Nagel's avatar
Jack Nagel committed
      user, repo, name = tapped_name.split("/", 3).map(&:downcase)
      @tap = Tap.fetch user, repo
Xu Cheng's avatar
Xu Cheng committed
      formula_dir = @tap.formula_dir || @tap.path
      path = formula_dir/"#{name}.rb"
Xu Cheng's avatar
Xu Cheng committed

Xu Cheng's avatar
Xu Cheng committed
      unless path.file?
Xu Cheng's avatar
Xu Cheng committed
        if (possible_alias = @tap.alias_dir/name).file?
Xu Cheng's avatar
Xu Cheng committed
          path = possible_alias.resolved_path
          name = path.basename(".rb").to_s
Xu Cheng's avatar
Xu Cheng committed
        elsif (new_name = @tap.formula_renames[name]) &&
              (new_path = formula_dir/"#{new_name}.rb").file?
Xu Cheng's avatar
Xu Cheng committed
          path = new_path
          name = new_name
          new_name = @tap.core_tap? ? name : "#{@tap}/#{name}"
        elsif (new_tap_name = @tap.tap_migrations[name])
          new_tap_user, new_tap_repo, = new_tap_name.split("/")
          new_tap_name = "#{new_tap_user}/#{new_tap_repo}"
          new_tap = Tap.fetch new_tap_name
          new_tap.install unless new_tap.installed?
          new_tapped_name = "#{new_tap_name}/#{name}"
          name, path = formula_name_path(new_tapped_name, warn: false)
          old_name = tapped_name
          new_name = new_tap.core_tap? ? name : new_tapped_name
        end

        if warn && old_name && new_name
          opoo "Use #{new_name} instead of deprecated #{old_name}"
Xu Cheng's avatar
Xu Cheng committed
        end
      end
    def get_formula(spec, alias_path: nil)
    rescue FormulaUnreadableError => e
      raise TapFormulaUnreadableError.new(tap, name, e.formula_error), "", e.backtrace
    rescue FormulaClassUnavailableError => e
      raise TapFormulaClassUnavailableError.new(tap, name, e.path, e.class_name, e.class_list), "", e.backtrace
    rescue FormulaUnavailableError => e
      raise TapFormulaUnavailableError.new(tap, name), "", e.backtrace
Adam Vandenberg's avatar
Adam Vandenberg committed
    end

    def load_file
      super
    rescue MethodDeprecatedError => e
      e.issues_url = tap.issues_url || tap.to_s
  class NullLoader < FormulaLoader
    def initialize(name)
      super name, Formulary.core_path(name)
    def get_formula(*)
      raise FormulaUnavailableError, name
Xu Cheng's avatar
Xu Cheng committed
  # Load formulae directly from their contents
  class FormulaContentsLoader < FormulaLoader
    # The formula's contents
    attr_reader :contents

    def initialize(name, path, contents)
      @contents = contents
      super name, path
    end

    def klass
      $stderr.puts "#{$0} (#{self.class.name}): loading #{path}" if ARGV.debug?
Xu Cheng's avatar
Xu Cheng committed
      namespace = "FormulaNamespace#{Digest::MD5.hexdigest(contents)}"
      Formulary.load_formula(name, path, contents, namespace)
    end
  end

Adam Vandenberg's avatar
Adam Vandenberg committed
  # Return a Formula instance for the given reference.
  # `ref` is string containing:
  # * a formula name
  # * a formula pathname
  # * a formula URL
  # * a local bottle reference
  def self.factory(ref, spec = :stable, alias_path: nil, from: nil)
    loader_for(ref, from: from).get_formula(spec, alias_path: alias_path)
Xu Cheng's avatar
Xu Cheng committed
  # Return a Formula instance for the given rack.
  # It will auto resolve formula's spec when requested spec is nil
  #
  # The :alias_path option will be used if the formula is found not to be
  # installed, and discarded if it is installed because the alias_path used
  # to install the formula will be set instead.
  def self.from_rack(rack, spec = nil, alias_path: nil)
Xu Cheng's avatar
Xu Cheng committed
    kegs = rack.directory? ? rack.subdirs.map { |d| Keg.new(d) } : []
    keg = kegs.detect(&:linked?) || kegs.detect(&:optlinked?) || kegs.max_by(&:version)

Xu Cheng's avatar
Xu Cheng committed
    if keg
      from_keg(keg, spec, alias_path: alias_path)
Xu Cheng's avatar
Xu Cheng committed
    else
      factory(rack.basename.to_s, spec || :stable, alias_path: alias_path, from: :rack)
Xu Cheng's avatar
Xu Cheng committed
    end
  end

  # Return a Formula instance for the given keg.
  # It will auto resolve formula's spec when requested spec is nil
  def self.from_keg(keg, spec = nil, alias_path: nil)
    tab = Tab.for_keg(keg)
    tap = tab.tap
    spec ||= tab.spec
      factory(keg.rack.basename.to_s, spec, alias_path: alias_path, from: :keg)
Xu Cheng's avatar
Xu Cheng committed
    else
        factory("#{tap}/#{keg.rack.basename}", spec, alias_path: alias_path, from: :keg)
      rescue FormulaUnavailableError
        # formula may be migrated to different tap. Try to search in core and all taps.
        factory(keg.rack.basename.to_s, spec, alias_path: alias_path, from: :keg)
Xu Cheng's avatar
Xu Cheng committed
    end
    f.build.used_options = Tab.remap_deprecated_options(f.deprecated_options, tab.used_options).as_flags
    f.version.update_commit(keg.version.version.commit) if f.head? && keg.version.head?
Xu Cheng's avatar
Xu Cheng committed
  # Return a Formula instance directly from contents
  def self.from_contents(name, path, contents, spec = :stable)
    FormulaContentsLoader.new(name, path, contents).get_formula(spec)
  end

    # If using a fully-scoped reference, check if the formula can be resolved.
    factory(ref) if ref.include? "/"

    # Check whether the rack with the given name exists.
Xu Cheng's avatar
Xu Cheng committed
    if (rack = HOMEBREW_CELLAR/File.basename(ref, ".rb")).directory?
      return rack.resolved_path
    # Use canonical name to locate rack.
    (HOMEBREW_CELLAR/canonical_name(ref)).resolved_path
  def self.canonical_name(ref)
    loader_for(ref).name
  rescue TapFormulaAmbiguityError
    # If there are multiple tap formulae with the name of ref,
    # then ref is the canonical name
    ref.downcase
Xu Cheng's avatar
Xu Cheng committed
  def self.path(ref)
    loader_for(ref).path
  end

  def self.loader_for(ref, from: nil)
    when %r{(https?|ftp|file)://}
      return FromUrlLoader.new(ref)
    when Pathname::BOTTLE_EXTNAME_RX
      return BottleLoader.new(ref)
Jack Nagel's avatar
Jack Nagel committed
    when HOMEBREW_TAP_FORMULA_REGEX
      return TapLoader.new(ref, from: from)
    if File.extname(ref) == ".rb" && Pathname.new(ref).expand_path.exist?
      return FromPathLoader.new(ref)
    end
    formula_with_that_name = core_path(ref)
    if formula_with_that_name.file?
      return FormulaLoader.new(ref, formula_with_that_name)
    possible_alias = CoreTap.instance.alias_dir/ref
    return AliasLoader.new(possible_alias) if possible_alias.file?
Xu Cheng's avatar
Xu Cheng committed
    possible_tap_formulae = tap_paths(ref)
    if possible_tap_formulae.size > 1
      raise TapFormulaAmbiguityError.new(ref, possible_tap_formulae)
Markus Reiter's avatar
Markus Reiter committed
    end

    if possible_tap_formulae.size == 1
Xu Cheng's avatar
Xu Cheng committed
      path = possible_tap_formulae.first.resolved_path
      name = path.basename(".rb").to_s
      return FormulaLoader.new(name, path)
    if newref = CoreTap.instance.formula_renames[ref]
      formula_with_that_oldname = core_path(newref)
      if formula_with_that_oldname.file?
        return FormulaLoader.new(newref, formula_with_that_oldname)
      end
    end

    possible_tap_newname_formulae = []
    Tap.each do |tap|
      if newref = tap.formula_renames[ref]
        possible_tap_newname_formulae << "#{tap.name}/#{newref}"
      end
    end

    if possible_tap_newname_formulae.size > 1
      raise TapFormulaWithOldnameAmbiguityError.new(ref, possible_tap_newname_formulae)
Markus Reiter's avatar
Markus Reiter committed
    end

    unless possible_tap_newname_formulae.empty?
      return TapLoader.new(possible_tap_newname_formulae.first, from: from)
    possible_keg_formula = Pathname.new("#{HOMEBREW_PREFIX}/opt/#{ref}/.brew/#{ref}.rb")
    if possible_keg_formula.file?
      return FormulaLoader.new(ref, possible_keg_formula)
    end

    possible_cached_formula = Pathname.new("#{HOMEBREW_CACHE_FORMULA}/#{ref}.rb")
    if possible_cached_formula.file?
      return FormulaLoader.new(ref, possible_cached_formula)
BrewTestBot's avatar
BrewTestBot committed
    NullLoader.new(ref)
    CoreTap.instance.formula_dir/"#{name.to_s.downcase}.rb"
Xu Cheng's avatar
Xu Cheng committed
  def self.tap_paths(name, taps = Dir["#{HOMEBREW_LIBRARY}/Taps/*/*/"])
    taps.map do |tap|
                      "#{tap}Formula/#{name}.rb",
                      "#{tap}HomebrewFormula/#{name}.rb",
                      "#{tap}#{name}.rb",
                      "#{tap}Aliases/#{name}",
                    ]).detect(&:file?)
Xu Cheng's avatar
Xu Cheng committed
  def self.find_with_priority(ref, spec = :stable)
    possible_pinned_tap_formulae = tap_paths(ref, Dir["#{HOMEBREW_LIBRARY}/PinnedTaps/*/*/"]).map(&:realpath)
    if possible_pinned_tap_formulae.size > 1
      raise TapFormulaAmbiguityError.new(ref, possible_pinned_tap_formulae)
Markus Reiter's avatar
Markus Reiter committed
    end

    if possible_pinned_tap_formulae.size == 1
      selected_formula = factory(possible_pinned_tap_formulae.first, spec)
      if core_path(ref).file?
        opoo <<-EOS.undent
          #{ref} is provided by core, but is now shadowed by #{selected_formula.full_name}.
          To refer to the core formula, use Homebrew/core/#{ref} instead.
        EOS
      end
      selected_formula
    else
      factory(ref, spec)
    end
  end