Class: Gem::SpecFetcher
| Relationships & Source Files | |
| Super Chains via Extension / Inclusion / Inheritance | |
| 
         Instance Chain: 
        
       | 
    |
| Inherits: | Object | 
| Defined in: | lib/rubygems/spec_fetcher.rb | 
Overview
SpecFetcher handles metadata updates from remote gem repositories.
Class Attribute Summary
- 
    
      .fetcher  
    
    rw
    
Default fetcher instance.
 - .fetcher=(fetcher) rw Internal use only
 
Class Method Summary
- 
    
      .new(sources = nil)  ⇒ SpecFetcher 
    
    constructor
    
Creates a new
SpecFetcher. 
Instance Attribute Summary
- 
    
      #latest_specs  
    
    readonly
    Internal use only
    
Cache of latest specs.
 - 
    
      #prerelease_specs  
    
    readonly
    Internal use only
    
Cache of prerelease specs.
 - 
    
      #sources  
    
    readonly
    Internal use only
    
Sources for this
SpecFetcher. - 
    
      #specs  
    
    readonly
    Internal use only
    
Cache of all released specs.
 
DefaultUserInteraction - Included
Instance Method Summary
- 
    
      #available_specs(type)  
    
    
Returns a list of gems available for each source in sources.
 - 
    
      #detect(type = :complete)  
    
    
Return all gem name tuples who's names match
obj - 
    
      #search_for_dependency(dependency, matching_platform = true)  
    
    
Find and fetch gem name tuples that match
dependency. - 
    
      #spec_for_dependency(dependency, matching_platform = true)  
    
    
Find and fetch specs that match
dependency. - 
    
      #suggest_gems_from_name(gem_name, type = :latest)  
    
    
Suggests gems based on the supplied
gem_name. - 
    
      #tuples_for(source, type, gracefully_ignore = false)  
    
    Internal use only
    
Retrieves NameTuples from
sourceof the giventype(:prerelease, etc.). 
UserInteraction - Included
| #alert | Displays an alert   | 
    
| #alert_error | Displays an error   | 
    
| #alert_warning | Displays a warning   | 
    
| #ask | Asks a   | 
    
| #ask_for_password | Asks for a password with a   | 
    
| #ask_yes_no | Asks a yes or no   | 
    
| #choose_from_list | Asks the user to answer   | 
    
| #say | Displays the given   | 
    
| #terminate_interaction | Terminates the RubyGems process with the given   | 
    
| #verbose | Calls   | 
    
DefaultUserInteraction - Included
Text - Included
| #clean_text | Remove any non-printable characters and make the text suitable for printing.  | 
    
| #format_text | Wraps   | 
    
| #levenshtein_distance | This code is based directly on the Text gem implementation Returns a value representing the “cost” of transforming str1 into str2.  | 
    
| #truncate_text, #min3 | |
Constructor Details
    .new(sources = nil)  ⇒ SpecFetcher 
  
Creates a new SpecFetcher.  Ordinarily you want to use the default fetcher from .fetcher which uses the Gem.sources.
If you need to retrieve specifications from a different source, you can send it as an argument.
# File 'lib/rubygems/spec_fetcher.rb', line 57
def initialize sources = nil @sources = sources || Gem.sources @update_cache = begin File.stat(Gem.user_home).uid == Process.uid rescue Errno::EACCES, Errno::ENOENT false end @specs = {} @latest_specs = {} @prerelease_specs = {} @caches = { :latest => @latest_specs, :prerelease => @prerelease_specs, :released => @specs, } @fetcher = Gem::RemoteFetcher.fetcher end
Class Attribute Details
.fetcher (rw)
Default fetcher instance. Use this instead of .new to reduce object allocation.
# File 'lib/rubygems/spec_fetcher.rb', line 42
def self.fetcher @fetcher ||= new end
.fetcher=(fetcher) (rw)
Instance Attribute Details
#latest_specs (readonly)
Cache of latest specs
# File 'lib/rubygems/spec_fetcher.rb', line 19
attr_reader :latest_specs # :nodoc:
#prerelease_specs (readonly)
Cache of prerelease specs
# File 'lib/rubygems/spec_fetcher.rb', line 34
attr_reader :prerelease_specs # :nodoc:
#sources (readonly)
Sources for this SpecFetcher
# File 'lib/rubygems/spec_fetcher.rb', line 24
attr_reader :sources # :nodoc:
#specs (readonly)
Cache of all released specs
# File 'lib/rubygems/spec_fetcher.rb', line 29
attr_reader :specs # :nodoc:
Instance Method Details
#available_specs(type)
Returns a list of gems available for each source in Gem.sources.
type can be one of 3 values: :released   => Return the list of all released specs :complete   => Return the list of all specs :latest     => Return the list of only the highest version of each gem :prerelease => Return the list of all prerelease only specs
# File 'lib/rubygems/spec_fetcher.rb', line 223
def available_specs(type) errors = [] list = {} @sources.each_source do |source| begin names = case type when :latest tuples_for source, :latest when :released tuples_for source, :released when :complete names = tuples_for(source, :prerelease, true) + tuples_for(source, :released) names.sort when :abs_latest names = tuples_for(source, :prerelease, true) + tuples_for(source, :latest) names.sort when :prerelease tuples_for(source, :prerelease) else raise Gem::Exception, "Unknown type - :#{type}" end rescue Gem::RemoteFetcher::FetchError => e errors << Gem::SourceFetchProblem.new(source, e) else list[source] = names end end [list, errors] end
#detect(type = :complete)
Return all gem name tuples who's names match obj
# File 'lib/rubygems/spec_fetcher.rb', line 145
def detect(type=:complete) tuples = [] list, _ = available_specs(type) list.each do |source, specs| specs.each do |tup| if yield(tup) tuples << [tup, source] end end end tuples end
#search_for_dependency(dependency, matching_platform = true)
Find and fetch gem name tuples that match dependency.
If matching_platform is false, gems for all platforms are returned.
# File 'lib/rubygems/spec_fetcher.rb', line 86
def search_for_dependency(dependency, matching_platform=true) found = {} rejected_specs = {} if dependency.prerelease? if dependency.specific? type = :complete else type = :abs_latest end elsif dependency.latest_version? type = :latest else type = :released end list, errors = available_specs(type) list.each do |source, specs| if dependency.name.is_a?(String) && specs.respond_to?(:bsearch) start_index = (0 ... specs.length).bsearch{ |i| specs[i].name >= dependency.name } end_index = (0 ... specs.length).bsearch{ |i| specs[i].name > dependency.name } specs = specs[start_index ... end_index] if start_index && end_index end found[source] = specs.select do |tup| if dependency.match?(tup) if matching_platform and !Gem::Platform.match(tup.platform) pm = ( rejected_specs[dependency] ||= \ Gem::PlatformMismatch.new(tup.name, tup.version)) pm.add_platform tup.platform false else true end end end end errors += rejected_specs.values tuples = [] found.each do |source, specs| specs.each do |s| tuples << [s, source] end end tuples = tuples.sort_by { |x| x[0] } return [tuples, errors] end
#spec_for_dependency(dependency, matching_platform = true)
Find and fetch specs that match dependency.
If matching_platform is false, gems for all platforms are returned.
# File 'lib/rubygems/spec_fetcher.rb', line 166
def spec_for_dependency(dependency, matching_platform=true) tuples, errors = search_for_dependency(dependency, matching_platform) specs = [] tuples.each do |tup, source| begin spec = source.fetch_spec(tup) rescue Gem::RemoteFetcher::FetchError => e errors << Gem::SourceFetchProblem.new(source, e) else specs << [spec, source] end end return [specs, errors] end
#suggest_gems_from_name(gem_name, type = :latest)
Suggests gems based on the supplied gem_name. Returns an array of alternative gem names.
# File 'lib/rubygems/spec_fetcher.rb', line 187
def suggest_gems_from_name(gem_name, type = :latest) gem_name = gem_name.downcase.tr('_-', '') max = gem_name.size / 2 names = available_specs(type).first.values.flatten(1) matches = names.map { |n| next unless n.match_platform? distance = levenshtein_distance gem_name, n.name.downcase.tr('_-', '') next if distance >= max return [n.name] if distance == 0 [n.name, distance] }.compact matches = if matches.empty? && type != :prerelease suggest_gems_from_name gem_name, :prerelease else matches.uniq.sort_by { |name, dist| dist } end matches.first(5).map { |name, dist| name } end
#tuples_for(source, type, gracefully_ignore = false)
Retrieves NameTuples from source of the given type (:prerelease, etc.).  If gracefully_ignore is true, errors are ignored.
# File 'lib/rubygems/spec_fetcher.rb', line 265
def tuples_for(source, type, gracefully_ignore=false) # :nodoc: @caches[type][source.uri] ||= source.load_specs(type).sort_by { |tup| tup.name } rescue Gem::RemoteFetcher::FetchError raise unless gracefully_ignore [] end