123456789_123456789_123456789_123456789_123456789_

Class: Gem::SpecificationRecord

Relationships & Source Files
Super Chains via Extension / Inclusion / Inheritance
Instance Chain:
self, Enumerable
Inherits: Object
Defined in: lib/rubygems/specification_record.rb

Constant Summary

Class Method Summary

Instance Attribute Summary

  • #all rw

    Returns the list of all specifications in the record.

  • #all=(specs) rw

    Sets the specs known by the record to specs.

Instance Method Summary

Constructor Details

.new(dirs) ⇒ SpecificationRecord

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 15

def initialize(dirs)
  @all = nil
  @stubs = nil
  @stubs_by_name = {}
  @spec_with_requirable_file = {}
  @active_stub_with_requirable_file = {}

  @dirs = dirs
end

Class Method Details

.dirs_from(paths)

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 5

def self.dirs_from(paths)
  paths.map do |path|
    File.join(path, "specifications")
  end
end

.from_path(path)

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 11

def self.from_path(path)
  new(dirs_from([path]))
end

Instance Attribute Details

#all (rw)

Returns the list of all specifications in the record

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 32

def all
  @all ||= Gem.loaded_specs.values | stubs.map(&:to_spec)
end

#all=(specs) (rw)

Sets the specs known by the record to specs.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 102

def all=(specs)
  @stubs_by_name = specs.group_by(&:name)
  @all = @stubs = specs
end

Instance Method Details

#add_spec(spec)

Adds spec to the the record, keeping the collection properly sorted.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 79

def add_spec(spec)
  return if all.include? spec

  all << spec
  stubs << spec
  (@stubs_by_name[spec.name] ||= []) << spec

  Gem::Specification._resort!(@stubs_by_name[spec.name])
  Gem::Specification._resort!(stubs)
end

#all_names

Return full names of all specs in the record in sorted order.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 110

def all_names
  all.map(&:full_name)
end

#each

Enumerate every known spec.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 119

def each
  return enum_for(:each) unless block_given?

  all.each do |x|
    yield x
  end
end

#find_active_stub_by_path(path)

Return the best specification in the record that contains the file matching Gem.path, among those already activated.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 174

def find_active_stub_by_path(path)
  stub = @active_stub_with_requirable_file[path] ||= stubs.find do |s|
    s.activated? && s.contains_requirable_file?(path)
  end || NOT_FOUND

  stub.this
end

#find_all_by_name(name, *requirements)

Returns every spec in the record that matches name and optional requirements.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 130

def find_all_by_name(name, *requirements)
  req = Gem::Requirement.create(*requirements)
  env_req = Gem.env_requirement(name)

  matches = stubs_for(name).find_all do |spec|
    req.satisfied_by?(spec.version) && env_req.satisfied_by?(spec.version)
  end.map(&:to_spec)

  if name == "bundler" && !req.specific?
    require_relative "bundler_version_finder"
    Gem::BundlerVersionFinder.prioritize!(matches)
  end

  matches
end

#find_by_path(path)

Return the best specification in the record that contains the file matching Gem.path.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 149

def find_by_path(path)
  path = path.dup.freeze
  spec = @spec_with_requirable_file[path] ||= stubs.find do |s|
    s.contains_requirable_file? path
  end || NOT_FOUND

  spec.to_spec
end

#find_inactive_by_path(path)

Return the best specification in the record that contains the file matching Gem.path amongst the specs that are not activated.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 162

def find_inactive_by_path(path)
  stub = stubs.find do |s|
    next if s.activated?
    s.contains_requirable_file? path
  end
  stub&.to_spec
end

#installed_stubs(pattern) (private)

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 199

def installed_stubs(pattern)
  map_stubs(pattern) do |path, base_dir, gems_dir|
    Gem::StubSpecification.gemspec_stub(path, base_dir, gems_dir)
  end
end

#latest_spec_for(name)

Return the latest installed spec in the record for gem name.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 193

def latest_spec_for(name)
  latest_specs(true).find {|installed_spec| installed_spec.name == name }
end

#latest_specs(prerelease)

Return the latest specs in the record, optionally including prerelease specs if prerelease is true.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 186

def latest_specs(prerelease)
  Gem::Specification._latest_specs stubs, prerelease
end

#map_stubs(pattern) (private)

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 205

def map_stubs(pattern)
  @dirs.flat_map do |dir|
    base_dir = File.dirname dir
    gems_dir = File.join base_dir, "gems"
    Gem::Specification.gemspec_stubs_in(dir, pattern) {|path| yield path, base_dir, gems_dir }
  end
end

#remove_spec(spec)

Removes spec from the record.

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 93

def remove_spec(spec)
  all.delete spec.to_spec
  stubs.delete spec
  (@stubs_by_name[spec.name] || []).delete spec
end

#stubs

Returns a StubSpecification for every specification in the record

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 39

def stubs
  @stubs ||= begin
    pattern = "*.gemspec"
    stubs = stubs_for_pattern(pattern, false)

    @stubs_by_name = stubs.select {|s| Gem::Platform.match_spec? s }.group_by(&:name)
    stubs
  end
end

#stubs_for(name)

Returns a StubSpecification for every specification in the record named name only returns stubs that match Gem.platforms

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 53

def stubs_for(name)
  if @stubs
    @stubs_by_name[name] || []
  else
    @stubs_by_name[name] ||= stubs_for_pattern("#{name}-*.gemspec").select do |s|
      s.name == name
    end
  end
end

#stubs_for_pattern(pattern, match_platform = true)

Finds stub specifications matching a pattern in the record, optionally filtering out specs not matching the current platform

[ GitHub ]

  
# File 'lib/rubygems/specification_record.rb', line 67

def stubs_for_pattern(pattern, match_platform = true)
  installed_stubs = installed_stubs(pattern)
  installed_stubs.select! {|s| Gem::Platform.match_spec? s } if match_platform
  stubs = installed_stubs + Gem::Specification.default_stubs(pattern)
  stubs = stubs.uniq(&:full_name)
  Gem::Specification._resort!(stubs)
  stubs
end