 d8adae0f92
			
		
	
	
		d8adae0f92
		
	
	
	
	
		
			
			We want to prefer the newer/versioned formulae when possible but this is preferable to a random cached formula or, worse, no formula. This means `brew info foo` will never complain that `foo` is a missing formula.
		
			
				
	
	
		
			432 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			432 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| require "digest/md5"
 | |
| require "tap"
 | |
| 
 | |
| # The Formulary is responsible for creating instances of Formula.
 | |
| # It is not meant to be used directly from formulae.
 | |
| 
 | |
| class Formulary
 | |
|   FORMULAE = {}
 | |
| 
 | |
|   def self.formula_class_defined?(path)
 | |
|     FORMULAE.key?(path)
 | |
|   end
 | |
| 
 | |
|   def self.formula_class_get(path)
 | |
|     FORMULAE.fetch(path)
 | |
|   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)
 | |
|     mod.module_eval(contents, path)
 | |
|     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
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   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)
 | |
|       io
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   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")
 | |
|     class_name
 | |
|   end
 | |
| 
 | |
|   # 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
 | |
|     end
 | |
| 
 | |
|     # 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)
 | |
|     end
 | |
| 
 | |
|     def klass
 | |
|       load_file unless Formulary.formula_class_defined?(path)
 | |
|       Formulary.formula_class_get(path)
 | |
|     end
 | |
| 
 | |
|     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)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # Loads formulae from bottles.
 | |
|   class BottleLoader < FormulaLoader
 | |
|     def initialize(bottle_name)
 | |
|       @bottle_filename = Pathname(bottle_name).realpath
 | |
|       name, full_name = Utils::Bottles.resolve_formula_names @bottle_filename
 | |
|       super name, Formulary.path(full_name)
 | |
|     end
 | |
| 
 | |
|     def get_formula(spec, alias_path: nil)
 | |
|       formula = super
 | |
|       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
 | |
|       formula
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   class AliasLoader < FormulaLoader
 | |
|     def initialize(alias_path)
 | |
|       path = alias_path.resolved_path
 | |
|       name = path.basename(".rb").to_s
 | |
|       super name, path
 | |
|       @alias_path = alias_path.to_s
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # Loads formulae from disk using a path
 | |
|   class FromPathLoader < FormulaLoader
 | |
|     def initialize(path)
 | |
|       path = Pathname.new(path).expand_path
 | |
|       super path.basename(".rb").to_s, path
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # Loads formulae from URLs
 | |
|   class FromUrlLoader < FormulaLoader
 | |
|     attr_reader :url
 | |
| 
 | |
|     def initialize(url)
 | |
|       @url = url
 | |
|       uri = URI(url)
 | |
|       formula = File.basename(uri.path, ".rb")
 | |
|       super formula, HOMEBREW_CACHE_FORMULA/File.basename(uri.path)
 | |
|     end
 | |
| 
 | |
|     def load_file
 | |
|       HOMEBREW_CACHE_FORMULA.mkpath
 | |
|       FileUtils.rm_f(path)
 | |
|       curl url, "-o", path
 | |
|       super
 | |
|     rescue MethodDeprecatedError => e
 | |
|       if url =~ %r{github.com/([\w-]+)/homebrew-([\w-]+)/}
 | |
|         e.issues_url = "https://github.com/#{$1}/homebrew-#{$2}/issues/new"
 | |
|       end
 | |
|       raise
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # 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)
 | |
|       user, repo, name = tapped_name.split("/", 3).map(&:downcase)
 | |
|       @tap = Tap.fetch user, repo
 | |
|       formula_dir = @tap.formula_dir || @tap.path
 | |
|       path = formula_dir/"#{name}.rb"
 | |
| 
 | |
|       unless path.file?
 | |
|         if (possible_alias = @tap.alias_dir/name).file?
 | |
|           path = possible_alias.resolved_path
 | |
|           name = path.basename(".rb").to_s
 | |
|         elsif (new_name = @tap.formula_renames[name]) &&
 | |
|               (new_path = formula_dir/"#{new_name}.rb").file?
 | |
|           old_name = name
 | |
|           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}"
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       [name, path]
 | |
|     end
 | |
| 
 | |
|     def get_formula(spec, alias_path: nil)
 | |
|       super
 | |
|     rescue FormulaUnavailableError => e
 | |
|       raise TapFormulaUnavailableError.new(tap, name), "", e.backtrace
 | |
|     end
 | |
| 
 | |
|     def load_file
 | |
|       super
 | |
|     rescue MethodDeprecatedError => e
 | |
|       e.issues_url = tap.issues_url || tap.to_s
 | |
|       raise
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   class NullLoader < FormulaLoader
 | |
|     def initialize(name)
 | |
|       super name, Formulary.core_path(name)
 | |
|     end
 | |
| 
 | |
|     def get_formula(*)
 | |
|       raise FormulaUnavailableError, name
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # 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?
 | |
|       namespace = "FormulaNamespace#{Digest::MD5.hexdigest(contents)}"
 | |
|       Formulary.load_formula(name, path, contents, namespace)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # 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)
 | |
|   end
 | |
| 
 | |
|   # 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)
 | |
|     kegs = rack.directory? ? rack.subdirs.map { |d| Keg.new(d) } : []
 | |
|     keg = kegs.detect(&:linked?) || kegs.detect(&:optlinked?) || kegs.max_by(&:version)
 | |
| 
 | |
|     if keg
 | |
|       from_keg(keg, spec, alias_path: alias_path)
 | |
|     else
 | |
|       factory(rack.basename.to_s, spec || :stable, alias_path: alias_path, from: :rack)
 | |
|     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
 | |
| 
 | |
|     f = if tap.nil?
 | |
|       factory(keg.rack.basename.to_s, spec, alias_path: alias_path, from: :keg)
 | |
|     else
 | |
|       begin
 | |
|         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)
 | |
|       end
 | |
|     end
 | |
|     f.build = tab
 | |
|     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?
 | |
|     f
 | |
|   end
 | |
| 
 | |
|   # 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
 | |
| 
 | |
|   def self.to_rack(ref)
 | |
|     # 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.
 | |
|     if (rack = HOMEBREW_CELLAR/File.basename(ref, ".rb")).directory?
 | |
|       return rack.resolved_path
 | |
|     end
 | |
| 
 | |
|     # Use canonical name to locate rack.
 | |
|     (HOMEBREW_CELLAR/canonical_name(ref)).resolved_path
 | |
|   end
 | |
| 
 | |
|   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
 | |
|   end
 | |
| 
 | |
|   def self.path(ref)
 | |
|     loader_for(ref).path
 | |
|   end
 | |
| 
 | |
|   def self.loader_for(ref, from: nil)
 | |
|     case ref
 | |
|     when %r{(https?|ftp|file)://}
 | |
|       return FromUrlLoader.new(ref)
 | |
|     when Pathname::BOTTLE_EXTNAME_RX
 | |
|       return BottleLoader.new(ref)
 | |
|     when HOMEBREW_TAP_FORMULA_REGEX
 | |
|       return TapLoader.new(ref, from: from)
 | |
|     end
 | |
| 
 | |
|     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)
 | |
|     end
 | |
| 
 | |
|     possible_alias = CoreTap.instance.alias_dir/ref
 | |
|     return AliasLoader.new(possible_alias) if possible_alias.file?
 | |
| 
 | |
|     possible_tap_formulae = tap_paths(ref)
 | |
|     if possible_tap_formulae.size > 1
 | |
|       raise TapFormulaAmbiguityError.new(ref, possible_tap_formulae)
 | |
|     end
 | |
| 
 | |
|     if possible_tap_formulae.size == 1
 | |
|       path = possible_tap_formulae.first.resolved_path
 | |
|       name = path.basename(".rb").to_s
 | |
|       return FormulaLoader.new(name, path)
 | |
|     end
 | |
| 
 | |
|     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)
 | |
|     end
 | |
| 
 | |
|     unless possible_tap_newname_formulae.empty?
 | |
|       return TapLoader.new(possible_tap_newname_formulae.first, from: from)
 | |
|     end
 | |
| 
 | |
|     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)
 | |
|     end
 | |
| 
 | |
|     NullLoader.new(ref)
 | |
|   end
 | |
| 
 | |
|   def self.core_path(name)
 | |
|     CoreTap.instance.formula_dir/"#{name.to_s.downcase}.rb"
 | |
|   end
 | |
| 
 | |
|   def self.tap_paths(name, taps = Dir["#{HOMEBREW_LIBRARY}/Taps/*/*/"])
 | |
|     name = name.downcase
 | |
|     taps.map do |tap|
 | |
|       Pathname.glob([
 | |
|                       "#{tap}Formula/#{name}.rb",
 | |
|                       "#{tap}HomebrewFormula/#{name}.rb",
 | |
|                       "#{tap}#{name}.rb",
 | |
|                       "#{tap}Aliases/#{name}",
 | |
|                     ]).detect(&:file?)
 | |
|     end.compact
 | |
|   end
 | |
| 
 | |
|   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)
 | |
|     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
 | |
| end
 |