mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
334178722b
Signed-off-by: David Heinemeier Hansson <david@loudthinking.com>
140 lines
5.1 KiB
Ruby
140 lines
5.1 KiB
Ruby
require 'rubygems'
|
|
require 'yaml'
|
|
|
|
module Rails
|
|
|
|
class VendorGemSourceIndex
|
|
# VendorGemSourceIndex acts as a proxy for the Gem source index, allowing
|
|
# gems to be loaded from vendor/gems. Rather than the standard gem repository format,
|
|
# vendor/gems contains unpacked gems, with YAML specifications in .specification in
|
|
# each gem directory.
|
|
include Enumerable
|
|
|
|
attr_reader :installed_source_index
|
|
attr_reader :vendor_source_index
|
|
|
|
@@silence_spec_warnings = false
|
|
|
|
def self.silence_spec_warnings
|
|
@@silence_spec_warnings
|
|
end
|
|
|
|
def self.silence_spec_warnings=(v)
|
|
@@silence_spec_warnings = v
|
|
end
|
|
|
|
def initialize(installed_index, vendor_dir=Rails::GemDependency.unpacked_path)
|
|
@installed_source_index = installed_index
|
|
@vendor_dir = vendor_dir
|
|
refresh!
|
|
end
|
|
|
|
def refresh!
|
|
# reload the installed gems
|
|
@installed_source_index.refresh!
|
|
vendor_gems = {}
|
|
|
|
# handle vendor Rails gems - they are identified by having loaded_from set to ""
|
|
# we add them manually to the list, so that other gems can find them via dependencies
|
|
Gem.loaded_specs.each do |n, s|
|
|
next unless s.loaded_from.empty?
|
|
vendor_gems[s.full_name] = s
|
|
end
|
|
|
|
# load specifications from vendor/gems
|
|
Dir[File.join(Rails::GemDependency.unpacked_path, '*')].each do |d|
|
|
dir_name = File.basename(d)
|
|
dir_version = version_for_dir(dir_name)
|
|
spec = load_specification(d)
|
|
if spec
|
|
if spec.full_name != dir_name
|
|
# mismatched directory name and gem spec - produced by 2.1.0-era unpack code
|
|
if dir_version
|
|
# fix the spec version - this is not optimal (spec.files may be wrong)
|
|
# but it's better than breaking apps. Complain to remind users to get correct specs.
|
|
# use ActiveSupport::Deprecation.warn, as the logger is not set yet
|
|
$stderr.puts("config.gem: Unpacked gem #{dir_name} in vendor/gems has a mismatched specification file."+
|
|
" Run 'rake gems:refresh_specs' to fix this.") unless @@silence_spec_warnings
|
|
spec.version = dir_version
|
|
else
|
|
$stderr.puts("config.gem: Unpacked gem #{dir_name} in vendor/gems is not in a versioned directory"+
|
|
"(should be #{spec.full_name}).") unless @@silence_spec_warnings
|
|
# continue, assume everything is OK
|
|
end
|
|
end
|
|
else
|
|
# no spec - produced by early-2008 unpack code
|
|
# emulate old behavior, and complain.
|
|
$stderr.puts("config.gem: Unpacked gem #{dir_name} in vendor/gems has no specification file."+
|
|
" Run 'rake gems:refresh_specs' to fix this.") unless @@silence_spec_warnings
|
|
if dir_version
|
|
spec = Gem::Specification.new
|
|
spec.version = dir_version
|
|
spec.require_paths = ['lib']
|
|
ext_path = File.join(d, 'ext')
|
|
spec.require_paths << 'ext' if File.exist?(ext_path)
|
|
spec.name = /^(.*)-[^-]+$/.match(dir_name)[1]
|
|
files = ['lib']
|
|
# set files to everything in lib/
|
|
files += Dir[File.join(d, 'lib', '*')].map { |v| v.gsub(/^#{d}\//, '') }
|
|
files += Dir[File.join(d, 'ext', '*')].map { |v| v.gsub(/^#{d}\//, '') } if ext_path
|
|
spec.files = files
|
|
else
|
|
$stderr.puts("config.gem: Unpacked gem #{dir_name} in vendor/gems not in a versioned directory."+
|
|
" Giving up.") unless @@silence_spec_warnings
|
|
next
|
|
end
|
|
end
|
|
spec.loaded_from = File.join(d, '.specification')
|
|
# finally, swap out full_gem_path
|
|
# it would be better to use a Gem::Specification subclass, but the YAML loads an explicit class
|
|
class << spec
|
|
def full_gem_path
|
|
path = File.join installation_path, full_name
|
|
return path if File.directory? path
|
|
File.join installation_path, original_name
|
|
end
|
|
end
|
|
vendor_gems[File.basename(d)] = spec
|
|
end
|
|
@vendor_source_index = Gem::SourceIndex.new(vendor_gems)
|
|
end
|
|
|
|
def version_for_dir(d)
|
|
matches = /-([^-]+)$/.match(d)
|
|
Gem::Version.new(matches[1]) if matches
|
|
end
|
|
|
|
def load_specification(gem_dir)
|
|
spec_file = File.join(gem_dir, '.specification')
|
|
YAML.load_file(spec_file) if File.exist?(spec_file)
|
|
end
|
|
|
|
def find_name(*args)
|
|
@installed_source_index.find_name(*args) + @vendor_source_index.find_name(*args)
|
|
end
|
|
|
|
def search(*args)
|
|
# look for vendor gems, and then installed gems - later elements take priority
|
|
@installed_source_index.search(*args) + @vendor_source_index.search(*args)
|
|
end
|
|
|
|
def each(&block)
|
|
@vendor_source_index.each(&block)
|
|
@installed_source_index.each(&block)
|
|
end
|
|
|
|
def add_spec(spec)
|
|
@vendor_source_index.add_spec spec
|
|
end
|
|
|
|
def remove_spec(spec)
|
|
@vendor_source_index.remove_spec spec
|
|
end
|
|
|
|
def size
|
|
@vendor_source_index.size + @installed_source_index.size
|
|
end
|
|
|
|
end
|
|
end
|