2018-11-02 19:07:56 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
module Bundler
|
|
|
|
# Represents a lazily loaded gem specification, where the full specification
|
|
|
|
# is on the source server in rubygems' "quick" index. The proxy object is to
|
|
|
|
# be seeded with what we're given from the source's abbreviated index - the
|
|
|
|
# full specification will only be fetched when necessary.
|
|
|
|
class RemoteSpecification
|
|
|
|
include MatchPlatform
|
|
|
|
include Comparable
|
|
|
|
|
|
|
|
attr_reader :name, :version, :platform
|
|
|
|
attr_writer :dependencies
|
|
|
|
attr_accessor :source, :remote
|
|
|
|
|
|
|
|
def initialize(name, version, platform, spec_fetcher)
|
|
|
|
@name = name
|
|
|
|
@version = Gem::Version.create version
|
2022-07-26 00:43:48 -04:00
|
|
|
@original_platform = platform || Gem::Platform::RUBY
|
|
|
|
@platform = Gem::Platform.new(platform)
|
2018-11-02 19:07:56 -04:00
|
|
|
@spec_fetcher = spec_fetcher
|
|
|
|
@dependencies = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
# Needed before installs, since the arch matters then and quick
|
|
|
|
# specs don't bother to include the arch in the platform string
|
|
|
|
def fetch_platform
|
|
|
|
@platform = _remote_specification.platform
|
|
|
|
end
|
|
|
|
|
2022-01-17 12:55:10 -05:00
|
|
|
# A fallback is included because the original version of the specification
|
|
|
|
# API didn't include that field, so some marshalled specs in the index have it
|
|
|
|
# set to +nil+.
|
|
|
|
def required_rubygems_version
|
|
|
|
@required_rubygems_version ||= _remote_specification.required_rubygems_version || Gem::Requirement.default
|
|
|
|
end
|
|
|
|
|
2018-11-02 19:07:56 -04:00
|
|
|
def full_name
|
2022-07-26 00:43:48 -04:00
|
|
|
if @original_platform == Gem::Platform::RUBY
|
2018-11-02 19:07:56 -04:00
|
|
|
"#{@name}-#{@version}"
|
|
|
|
else
|
2022-07-26 00:43:48 -04:00
|
|
|
"#{@name}-#{@version}-#{@original_platform}"
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Compare this specification against another object. Using sort_obj
|
|
|
|
# is compatible with Gem::Specification and other Bundler or RubyGems
|
|
|
|
# objects. Otherwise, use the default Object comparison.
|
|
|
|
def <=>(other)
|
|
|
|
if other.respond_to?(:sort_obj)
|
|
|
|
sort_obj <=> other.sort_obj
|
|
|
|
else
|
|
|
|
super
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Because Rubyforge cannot be trusted to provide valid specifications
|
|
|
|
# once the remote gem is downloaded, the backend specification will
|
|
|
|
# be swapped out.
|
|
|
|
def __swap__(spec)
|
2020-05-08 01:19:04 -04:00
|
|
|
raise APIResponseInvalidDependenciesError unless spec.dependencies.all? {|d| d.is_a?(Gem::Dependency) }
|
|
|
|
|
2018-11-02 19:07:56 -04:00
|
|
|
SharedHelpers.ensure_same_dependencies(self, dependencies, spec.dependencies)
|
|
|
|
@_remote_specification = spec
|
|
|
|
end
|
|
|
|
|
|
|
|
# Create a delegate used for sorting. This strategy is copied from
|
|
|
|
# RubyGems 2.23 and ensures that Bundler's specifications can be
|
|
|
|
# compared and sorted with RubyGems' own specifications.
|
|
|
|
#
|
|
|
|
# @see #<=>
|
|
|
|
# @see Gem::Specification#sort_obj
|
|
|
|
#
|
|
|
|
# @return [Array] an object you can use to compare and sort this
|
|
|
|
# specification against other specifications
|
|
|
|
def sort_obj
|
|
|
|
[@name, @version, @platform == Gem::Platform::RUBY ? -1 : 1]
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_s
|
|
|
|
"#<#{self.class} name=#{name} version=#{version} platform=#{platform}>"
|
|
|
|
end
|
|
|
|
|
|
|
|
def dependencies
|
|
|
|
@dependencies ||= begin
|
|
|
|
deps = method_missing(:dependencies)
|
|
|
|
|
|
|
|
# allow us to handle when the specs dependencies are an array of array of string
|
2020-05-08 01:19:04 -04:00
|
|
|
# in order to delay the crash to `#__swap__` where it results in a friendlier error
|
|
|
|
# see https://github.com/rubygems/bundler/issues/5797
|
2018-11-02 19:07:56 -04:00
|
|
|
deps = deps.map {|d| d.is_a?(Gem::Dependency) ? d : Gem::Dependency.new(*d) }
|
|
|
|
|
|
|
|
deps
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def git_version
|
|
|
|
return unless loaded_from && source.is_a?(Bundler::Source::Git)
|
|
|
|
" #{source.revision[0..6]}"
|
|
|
|
end
|
|
|
|
|
2020-10-15 00:20:25 -04:00
|
|
|
private
|
2018-11-02 19:07:56 -04:00
|
|
|
|
|
|
|
def to_ary
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
|
|
|
|
def _remote_specification
|
2022-07-26 00:43:48 -04:00
|
|
|
@_remote_specification ||= @spec_fetcher.fetch_spec([@name, @version, @original_platform])
|
2018-11-02 19:07:56 -04:00
|
|
|
@_remote_specification || raise(GemspecError, "Gemspec data for #{full_name} was" \
|
|
|
|
" missing from the server! Try installing with `--full-index` as a workaround.")
|
|
|
|
end
|
|
|
|
|
|
|
|
def method_missing(method, *args, &blk)
|
|
|
|
_remote_specification.send(method, *args, &blk)
|
|
|
|
end
|
|
|
|
|
|
|
|
def respond_to?(method, include_all = false)
|
|
|
|
super || _remote_specification.respond_to?(method, include_all)
|
|
|
|
end
|
|
|
|
public :respond_to?
|
|
|
|
end
|
|
|
|
end
|