mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
9926b16d35
* lib/rubygems/commands/setup_command.rb (class Gem): * lib/rubygems/commands/setup_command.rb (TEXT): * lib/rubygems/spec_fetcher.rb (class Gem): * lib/rubygems/test_utilities.rb (class Gem): * lib/rubygems.rb (module Gem): * test/rubygems/test_gem_commands_setup_command.rb (class TestGemCommandsSetupCommand): * test/rubygems/test_gem_spec_fetcher.rb (Upgraded http): git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@39622 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
267 lines
6 KiB
Ruby
267 lines
6 KiB
Ruby
require 'rubygems/remote_fetcher'
|
|
require 'rubygems/user_interaction'
|
|
require 'rubygems/errors'
|
|
require 'rubygems/text'
|
|
require 'rubygems/name_tuple'
|
|
|
|
##
|
|
# SpecFetcher handles metadata updates from remote gem repositories.
|
|
|
|
class Gem::SpecFetcher
|
|
|
|
include Gem::UserInteraction
|
|
include Gem::Text
|
|
|
|
##
|
|
# Cache of latest specs
|
|
|
|
attr_reader :latest_specs # :nodoc:
|
|
|
|
##
|
|
# Cache of all released specs
|
|
|
|
attr_reader :specs # :nodoc:
|
|
|
|
##
|
|
# Cache of prerelease specs
|
|
|
|
attr_reader :prerelease_specs # :nodoc:
|
|
|
|
@fetcher = nil
|
|
|
|
def self.fetcher
|
|
@fetcher ||= new
|
|
end
|
|
|
|
def self.fetcher=(fetcher) # :nodoc:
|
|
@fetcher = fetcher
|
|
end
|
|
|
|
def initialize
|
|
@dir = File.join Gem.user_home, '.gem', 'specs'
|
|
@update_cache = File.stat(Gem.user_home).uid == Process.uid
|
|
|
|
@specs = {}
|
|
@latest_specs = {}
|
|
@prerelease_specs = {}
|
|
|
|
@caches = {
|
|
:latest => @latest_specs,
|
|
:prerelease => @prerelease_specs,
|
|
:released => @specs,
|
|
}
|
|
|
|
@fetcher = Gem::RemoteFetcher.fetcher
|
|
end
|
|
|
|
##
|
|
#
|
|
# Find and fetch gem name tuples that match +dependency+.
|
|
#
|
|
# If +matching_platform+ is false, gems for all platforms are returned.
|
|
|
|
def search_for_dependency(dependency, matching_platform=true)
|
|
found = {}
|
|
|
|
rejected_specs = {}
|
|
|
|
if dependency.prerelease?
|
|
type = :complete
|
|
elsif dependency.latest_version?
|
|
type = :latest
|
|
else
|
|
type = :released
|
|
end
|
|
|
|
list, errors = available_specs(type)
|
|
list.each do |source, specs|
|
|
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
|
|
|
|
|
|
##
|
|
# Return all gem name tuples who's names match +obj+
|
|
|
|
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
|
|
|
|
|
|
##
|
|
# Find and fetch specs that match +dependency+.
|
|
#
|
|
# If +matching_platform+ is false, gems for all platforms are returned.
|
|
|
|
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
|
|
|
|
##
|
|
# Suggests gems based on the supplied +gem_name+. Returns an array of
|
|
# alternative gem names.
|
|
|
|
def suggest_gems_from_name gem_name
|
|
gem_name = gem_name.downcase.tr('_-', '')
|
|
max = gem_name.size / 2
|
|
names = available_specs(:complete).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 = matches.uniq.sort_by { |name, dist| dist }
|
|
|
|
matches.first(5).map { |name, dist| name }
|
|
end
|
|
|
|
##
|
|
# 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
|
|
#
|
|
|
|
def available_specs(type)
|
|
errors = []
|
|
list = {}
|
|
|
|
Gem.sources.each_source do |source|
|
|
source = upgrade_http_source source
|
|
|
|
begin
|
|
names = case type
|
|
when :latest
|
|
tuples_for source, :latest
|
|
when :released
|
|
tuples_for source, :released
|
|
when :complete
|
|
tuples_for(source, :prerelease, true) +
|
|
tuples_for(source, :released)
|
|
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
|
|
|
|
def tuples_for(source, type, gracefully_ignore=false)
|
|
cache = @caches[type]
|
|
|
|
if gracefully_ignore
|
|
begin
|
|
cache[source.uri] ||= source.load_specs(type)
|
|
rescue Gem::RemoteFetcher::FetchError
|
|
[]
|
|
end
|
|
else
|
|
cache[source.uri] ||= source.load_specs(type)
|
|
end
|
|
end
|
|
|
|
##
|
|
# Attempts to upgrade +source+ to HTTPS if it is for http://rubygems.org
|
|
|
|
def upgrade_http_source source
|
|
uri = source.uri
|
|
|
|
return source unless uri.scheme.downcase == 'http' &&
|
|
uri.host.downcase == 'rubygems.org'
|
|
|
|
https_uri = uri.dup
|
|
https_uri.scheme = 'https'
|
|
https_uri += '/'
|
|
|
|
https_uri = URI https_uri.to_s # cast to URI::HTTPS
|
|
|
|
begin
|
|
Gem::RemoteFetcher.fetcher.fetch_path https_uri, nil, true
|
|
rescue Gem::RemoteFetcher::FetchError => e
|
|
raise unless e.message =~ / Not Allowed 405 /
|
|
end
|
|
|
|
say "Upgraded #{uri} to HTTPS"
|
|
|
|
https_uri += uri.request_uri
|
|
|
|
source.uri = URI https_uri.to_s # cast to URI::HTTPS
|
|
|
|
source
|
|
rescue Gem::RemoteFetcher::FetchError
|
|
say "Upgrading #{uri} to HTTPS failed, continuing" if
|
|
Gem.configuration.really_verbose
|
|
|
|
source
|
|
end
|
|
|
|
end
|
|
|