2018-11-02 19:07:56 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require "rubygems/user_interaction"
|
|
|
|
|
|
|
|
module Bundler
|
|
|
|
class Source
|
|
|
|
class Rubygems < Source
|
2019-06-01 05:49:40 -04:00
|
|
|
autoload :Remote, File.expand_path("rubygems/remote", __dir__)
|
2018-11-02 19:07:56 -04:00
|
|
|
|
|
|
|
# Use the API when installing less than X gems
|
|
|
|
API_REQUEST_LIMIT = 500
|
|
|
|
# Ask for X gems per API request
|
|
|
|
API_REQUEST_SIZE = 50
|
|
|
|
|
|
|
|
attr_reader :remotes, :caches
|
|
|
|
|
|
|
|
def initialize(options = {})
|
|
|
|
@options = options
|
|
|
|
@remotes = []
|
|
|
|
@dependency_names = []
|
|
|
|
@allow_remote = false
|
|
|
|
@allow_cached = false
|
2021-04-14 23:47:04 -04:00
|
|
|
@allow_local = options["allow_local"] || false
|
2018-11-02 19:07:56 -04:00
|
|
|
@caches = [cache_path, *Bundler.rubygems.gem_cache]
|
|
|
|
|
2021-04-14 23:47:04 -04:00
|
|
|
Array(options["remotes"]).reverse_each {|r| add_remote(r) }
|
|
|
|
end
|
|
|
|
|
2021-08-18 03:58:51 -04:00
|
|
|
def local_only!
|
|
|
|
@specs = nil
|
|
|
|
@allow_local = true
|
2021-08-18 04:28:30 -04:00
|
|
|
@allow_cached = false
|
2021-08-18 03:58:51 -04:00
|
|
|
@allow_remote = false
|
|
|
|
end
|
|
|
|
|
2021-04-14 23:47:04 -04:00
|
|
|
def local!
|
|
|
|
return if @allow_local
|
|
|
|
|
|
|
|
@specs = nil
|
|
|
|
@allow_local = true
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def remote!
|
2021-04-14 23:47:04 -04:00
|
|
|
return if @allow_remote
|
|
|
|
|
2018-11-02 19:07:56 -04:00
|
|
|
@specs = nil
|
|
|
|
@allow_remote = true
|
|
|
|
end
|
|
|
|
|
|
|
|
def cached!
|
2021-04-14 23:47:04 -04:00
|
|
|
return if @allow_cached
|
|
|
|
|
2018-11-02 19:07:56 -04:00
|
|
|
@specs = nil
|
2021-07-31 09:39:45 -04:00
|
|
|
@allow_local = true
|
2018-11-02 19:07:56 -04:00
|
|
|
@allow_cached = true
|
|
|
|
end
|
|
|
|
|
|
|
|
def hash
|
|
|
|
@remotes.hash
|
|
|
|
end
|
|
|
|
|
|
|
|
def eql?(other)
|
|
|
|
other.is_a?(Rubygems) && other.credless_remotes == credless_remotes
|
|
|
|
end
|
|
|
|
|
|
|
|
alias_method :==, :eql?
|
|
|
|
|
|
|
|
def include?(o)
|
|
|
|
o.is_a?(Rubygems) && (o.credless_remotes - credless_remotes).empty?
|
|
|
|
end
|
|
|
|
|
2021-05-28 06:47:49 -04:00
|
|
|
def multiple_remotes?
|
|
|
|
@remotes.size > 1
|
2021-04-14 23:47:04 -04:00
|
|
|
end
|
|
|
|
|
2021-07-24 03:21:42 -04:00
|
|
|
def no_remotes?
|
|
|
|
@remotes.size == 0
|
|
|
|
end
|
|
|
|
|
2018-11-02 19:07:56 -04:00
|
|
|
def can_lock?(spec)
|
2021-05-28 06:47:49 -04:00
|
|
|
return super unless multiple_remotes?
|
|
|
|
include?(spec.source)
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def options
|
|
|
|
{ "remotes" => @remotes.map(&:to_s) }
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.from_lock(options)
|
|
|
|
new(options)
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_lock
|
|
|
|
out = String.new("GEM\n")
|
|
|
|
remotes.reverse_each do |remote|
|
|
|
|
out << " remote: #{suppress_configured_credentials remote}\n"
|
|
|
|
end
|
|
|
|
out << " specs:\n"
|
|
|
|
end
|
|
|
|
|
2021-07-31 08:10:59 -04:00
|
|
|
def to_err
|
|
|
|
if remotes.empty?
|
|
|
|
"locally installed gems"
|
|
|
|
elsif @allow_remote
|
|
|
|
"rubygems repository #{remote_names} or installed locally"
|
|
|
|
elsif @allow_cached
|
|
|
|
"cached gems from rubygems repository #{remote_names} or installed locally"
|
|
|
|
else
|
|
|
|
"locally installed gems"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-02 19:07:56 -04:00
|
|
|
def to_s
|
|
|
|
if remotes.empty?
|
|
|
|
"locally installed gems"
|
|
|
|
else
|
|
|
|
"rubygems repository #{remote_names} or installed locally"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
alias_method :name, :to_s
|
|
|
|
|
|
|
|
def specs
|
|
|
|
@specs ||= begin
|
|
|
|
# remote_specs usually generates a way larger Index than the other
|
|
|
|
# sources, and large_idx.use small_idx is way faster than
|
|
|
|
# small_idx.use large_idx.
|
|
|
|
idx = @allow_remote ? remote_specs.dup : Index.new
|
|
|
|
idx.use(cached_specs, :override_dupes) if @allow_cached || @allow_remote
|
2021-04-14 23:47:04 -04:00
|
|
|
idx.use(installed_specs, :override_dupes) if @allow_local
|
2018-11-02 19:07:56 -04:00
|
|
|
idx
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def install(spec, opts = {})
|
|
|
|
force = opts[:force]
|
|
|
|
ensure_builtin_gems_cached = opts[:ensure_builtin_gems_cached]
|
|
|
|
|
2021-10-11 13:59:46 -04:00
|
|
|
if ensure_builtin_gems_cached && spec.default_gem?
|
2018-11-02 19:07:56 -04:00
|
|
|
if !cached_path(spec)
|
|
|
|
cached_built_in_gem(spec) unless spec.remote
|
|
|
|
force = true
|
|
|
|
else
|
|
|
|
spec.loaded_from = loaded_from(spec)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-08-21 14:13:03 -04:00
|
|
|
if installed?(spec) && !force
|
2018-11-02 19:07:56 -04:00
|
|
|
print_using_message "Using #{version_message(spec)}"
|
|
|
|
return nil # no post-install message
|
|
|
|
end
|
|
|
|
|
|
|
|
# Download the gem to get the spec, because some specs that are returned
|
|
|
|
# by rubygems.org are broken and wrong.
|
|
|
|
if spec.remote
|
|
|
|
# Check for this spec from other sources
|
|
|
|
uris = [spec.remote.anonymized_uri]
|
|
|
|
uris += remotes_for_spec(spec).map(&:anonymized_uri)
|
|
|
|
uris.uniq!
|
|
|
|
Installer.ambiguous_gems << [spec.name, *uris] if uris.length > 1
|
|
|
|
|
2019-04-14 02:01:35 -04:00
|
|
|
path = fetch_gem(spec)
|
|
|
|
begin
|
|
|
|
s = Bundler.rubygems.spec_from_gem(path, Bundler.settings["trust-policy"])
|
|
|
|
spec.__swap__(s)
|
2021-10-27 16:15:31 -04:00
|
|
|
rescue Gem::Package::FormatError
|
2019-04-14 02:01:35 -04:00
|
|
|
Bundler.rm_rf(path)
|
|
|
|
raise
|
|
|
|
end
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
unless Bundler.settings[:no_install]
|
|
|
|
message = "Installing #{version_message(spec)}"
|
|
|
|
message += " with native extensions" if spec.extensions.any?
|
|
|
|
Bundler.ui.confirm message
|
|
|
|
|
|
|
|
path = cached_gem(spec)
|
2021-10-21 06:38:53 -04:00
|
|
|
raise GemNotFound, "Could not find #{spec.file_name} for installation" unless path
|
2018-11-02 19:07:56 -04:00
|
|
|
if requires_sudo?
|
|
|
|
install_path = Bundler.tmp(spec.full_name)
|
|
|
|
bin_path = install_path.join("bin")
|
|
|
|
else
|
|
|
|
install_path = rubygems_dir
|
|
|
|
bin_path = Bundler.system_bindir
|
|
|
|
end
|
|
|
|
|
|
|
|
Bundler.mkdir_p bin_path, :no_sudo => true unless spec.executables.empty? || Bundler.rubygems.provides?(">= 2.7.5")
|
|
|
|
|
2021-02-01 10:17:16 -05:00
|
|
|
require_relative "../rubygems_gem_installer"
|
|
|
|
|
2020-05-21 08:05:07 -04:00
|
|
|
installed_spec = Bundler::RubyGemsGemInstaller.at(
|
|
|
|
path,
|
|
|
|
:install_dir => install_path.to_s,
|
|
|
|
:bin_dir => bin_path.to_s,
|
|
|
|
:ignore_dependencies => true,
|
|
|
|
:wrappers => true,
|
|
|
|
:env_shebang => true,
|
|
|
|
:build_args => opts[:build_args],
|
|
|
|
:bundler_expected_checksum => spec.respond_to?(:checksum) && spec.checksum,
|
|
|
|
:bundler_extension_cache_path => extension_cache_path(spec)
|
|
|
|
).install
|
2018-11-02 19:07:56 -04:00
|
|
|
spec.full_gem_path = installed_spec.full_gem_path
|
|
|
|
|
|
|
|
# SUDO HAX
|
|
|
|
if requires_sudo?
|
|
|
|
Bundler.rubygems.repository_subdirectories.each do |name|
|
|
|
|
src = File.join(install_path, name, "*")
|
|
|
|
dst = File.join(rubygems_dir, name)
|
|
|
|
if name == "extensions" && Dir.glob(src).any?
|
|
|
|
src = File.join(src, "*/*")
|
|
|
|
ext_src = Dir.glob(src).first
|
|
|
|
ext_src.gsub!(src[0..-6], "")
|
|
|
|
dst = File.dirname(File.join(dst, ext_src))
|
|
|
|
end
|
|
|
|
SharedHelpers.filesystem_access(dst) do |p|
|
|
|
|
Bundler.mkdir_p(p)
|
|
|
|
end
|
|
|
|
Bundler.sudo "cp -R #{src} #{dst}" if Dir[src].any?
|
|
|
|
end
|
|
|
|
|
|
|
|
spec.executables.each do |exe|
|
|
|
|
SharedHelpers.filesystem_access(Bundler.system_bindir) do |p|
|
|
|
|
Bundler.mkdir_p(p)
|
|
|
|
end
|
|
|
|
Bundler.sudo "cp -R #{install_path}/bin/#{exe} #{Bundler.system_bindir}/"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
installed_spec.loaded_from = loaded_from(spec)
|
|
|
|
end
|
|
|
|
spec.loaded_from = loaded_from(spec)
|
|
|
|
|
|
|
|
spec.post_install_message
|
|
|
|
ensure
|
|
|
|
Bundler.rm_rf(install_path) if requires_sudo?
|
|
|
|
end
|
|
|
|
|
|
|
|
def cache(spec, custom_path = nil)
|
2021-10-21 06:38:53 -04:00
|
|
|
cached_path = cached_gem(spec)
|
2021-10-20 13:27:06 -04:00
|
|
|
raise GemNotFound, "Missing gem file '#{spec.file_name}'." unless cached_path
|
2018-11-02 19:07:56 -04:00
|
|
|
return if File.dirname(cached_path) == Bundler.app_cache.to_s
|
|
|
|
Bundler.ui.info " * #{File.basename(cached_path)}"
|
|
|
|
FileUtils.cp(cached_path, Bundler.app_cache(custom_path))
|
|
|
|
rescue Errno::EACCES => e
|
|
|
|
Bundler.ui.debug(e)
|
|
|
|
raise InstallError, e.message
|
|
|
|
end
|
|
|
|
|
|
|
|
def cached_built_in_gem(spec)
|
|
|
|
cached_path = cached_path(spec)
|
|
|
|
if cached_path.nil?
|
|
|
|
remote_spec = remote_specs.search(spec).first
|
|
|
|
if remote_spec
|
|
|
|
cached_path = fetch_gem(remote_spec)
|
|
|
|
else
|
|
|
|
Bundler.ui.warn "#{spec.full_name} is built in to Ruby, and can't be cached because your Gemfile doesn't have any sources that contain it."
|
|
|
|
end
|
|
|
|
end
|
|
|
|
cached_path
|
|
|
|
end
|
|
|
|
|
|
|
|
def add_remote(source)
|
|
|
|
uri = normalize_uri(source)
|
|
|
|
@remotes.unshift(uri) unless @remotes.include?(uri)
|
|
|
|
end
|
|
|
|
|
|
|
|
def equivalent_remotes?(other_remotes)
|
|
|
|
other_remotes.map(&method(:remove_auth)) == @remotes.map(&method(:remove_auth))
|
|
|
|
end
|
|
|
|
|
2021-05-28 06:47:49 -04:00
|
|
|
def spec_names
|
|
|
|
if @allow_remote && dependency_api_available?
|
|
|
|
remote_specs.spec_names
|
|
|
|
else
|
|
|
|
[]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-02 19:07:56 -04:00
|
|
|
def unmet_deps
|
2021-05-28 06:47:49 -04:00
|
|
|
if @allow_remote && dependency_api_available?
|
2018-11-02 19:07:56 -04:00
|
|
|
remote_specs.unmet_dependency_names
|
|
|
|
else
|
|
|
|
[]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def fetchers
|
|
|
|
@fetchers ||= remotes.map do |uri|
|
|
|
|
remote = Source::Rubygems::Remote.new(uri)
|
|
|
|
Bundler::Fetcher.new(remote)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def double_check_for(unmet_dependency_names)
|
|
|
|
return unless @allow_remote
|
2021-05-28 06:47:49 -04:00
|
|
|
return unless dependency_api_available?
|
2018-11-02 19:07:56 -04:00
|
|
|
|
|
|
|
unmet_dependency_names = unmet_dependency_names.call
|
|
|
|
unless unmet_dependency_names.nil?
|
|
|
|
if api_fetchers.size <= 1
|
|
|
|
# can't do this when there are multiple fetchers because then we might not fetch from _all_
|
|
|
|
# of them
|
|
|
|
unmet_dependency_names -= remote_specs.spec_names # avoid re-fetching things we've already gotten
|
|
|
|
end
|
|
|
|
return if unmet_dependency_names.empty?
|
|
|
|
end
|
|
|
|
|
|
|
|
Bundler.ui.debug "Double checking for #{unmet_dependency_names || "all specs (due to the size of the request)"} in #{self}"
|
|
|
|
|
|
|
|
fetch_names(api_fetchers, unmet_dependency_names, specs, false)
|
|
|
|
end
|
|
|
|
|
|
|
|
def dependency_names_to_double_check
|
|
|
|
names = []
|
|
|
|
remote_specs.each do |spec|
|
|
|
|
case spec
|
|
|
|
when EndpointSpecification, Gem::Specification, StubSpecification, LazySpecification
|
2021-05-28 06:47:49 -04:00
|
|
|
names.concat(spec.runtime_dependencies.map(&:name))
|
2018-11-02 19:07:56 -04:00
|
|
|
when RemoteSpecification # from the full index
|
|
|
|
return nil
|
|
|
|
else
|
|
|
|
raise "unhandled spec type (#{spec.inspect})"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
names
|
|
|
|
end
|
|
|
|
|
2021-05-28 06:47:49 -04:00
|
|
|
def dependency_api_available?
|
|
|
|
api_fetchers.any?
|
|
|
|
end
|
|
|
|
|
2020-10-15 00:20:25 -04:00
|
|
|
protected
|
2018-11-02 19:07:56 -04:00
|
|
|
|
2021-07-31 08:10:59 -04:00
|
|
|
def remote_names
|
|
|
|
remotes.map(&:to_s).join(", ")
|
|
|
|
end
|
|
|
|
|
2018-11-02 19:07:56 -04:00
|
|
|
def credless_remotes
|
|
|
|
remotes.map(&method(:suppress_configured_credentials))
|
|
|
|
end
|
|
|
|
|
|
|
|
def remotes_for_spec(spec)
|
|
|
|
specs.search_all(spec.name).inject([]) do |uris, s|
|
|
|
|
uris << s.remote if s.remote
|
|
|
|
uris
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def loaded_from(spec)
|
|
|
|
"#{rubygems_dir}/specifications/#{spec.full_name}.gemspec"
|
|
|
|
end
|
|
|
|
|
|
|
|
def cached_gem(spec)
|
2021-10-21 06:38:53 -04:00
|
|
|
if spec.default_gem?
|
|
|
|
cached_built_in_gem(spec)
|
|
|
|
else
|
|
|
|
cached_path(spec)
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def cached_path(spec)
|
2021-10-21 14:28:02 -04:00
|
|
|
global_cache_path = download_cache_path(spec)
|
|
|
|
@caches << global_cache_path if global_cache_path
|
|
|
|
|
|
|
|
possibilities = @caches.map {|p| "#{p}/#{spec.file_name}" }
|
2018-11-02 19:07:56 -04:00
|
|
|
possibilities.find {|p| File.exist?(p) }
|
|
|
|
end
|
|
|
|
|
|
|
|
def normalize_uri(uri)
|
|
|
|
uri = uri.to_s
|
|
|
|
uri = "#{uri}/" unless uri =~ %r{/$}
|
2019-12-14 05:49:16 -05:00
|
|
|
require_relative "../vendored_uri"
|
|
|
|
uri = Bundler::URI(uri)
|
2018-11-02 19:07:56 -04:00
|
|
|
raise ArgumentError, "The source must be an absolute URI. For example:\n" \
|
2019-12-14 05:49:16 -05:00
|
|
|
"source 'https://rubygems.org'" if !uri.absolute? || (uri.is_a?(Bundler::URI::HTTP) && uri.host.nil?)
|
2018-11-02 19:07:56 -04:00
|
|
|
uri
|
|
|
|
end
|
|
|
|
|
|
|
|
def suppress_configured_credentials(remote)
|
|
|
|
remote_nouser = remove_auth(remote)
|
|
|
|
if remote.userinfo && remote.userinfo == Bundler.settings[remote_nouser]
|
|
|
|
remote_nouser
|
|
|
|
else
|
|
|
|
remote
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def remove_auth(remote)
|
|
|
|
if remote.user || remote.password
|
|
|
|
remote.dup.tap {|uri| uri.user = uri.password = nil }.to_s
|
|
|
|
else
|
|
|
|
remote.to_s
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def installed_specs
|
|
|
|
@installed_specs ||= Index.build do |idx|
|
|
|
|
Bundler.rubygems.all_specs.reverse_each do |spec|
|
|
|
|
spec.source = self
|
|
|
|
if Bundler.rubygems.spec_missing_extensions?(spec, false)
|
|
|
|
Bundler.ui.debug "Source #{self} is ignoring #{spec} because it is missing extensions"
|
|
|
|
next
|
|
|
|
end
|
|
|
|
idx << spec
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def cached_specs
|
|
|
|
@cached_specs ||= begin
|
2021-04-14 23:47:04 -04:00
|
|
|
idx = @allow_local ? installed_specs.dup : Index.new
|
2018-11-02 19:07:56 -04:00
|
|
|
|
|
|
|
Dir["#{cache_path}/*.gem"].each do |gemfile|
|
|
|
|
next if gemfile =~ /^bundler\-[\d\.]+?\.gem/
|
|
|
|
s ||= Bundler.rubygems.spec_from_gem(gemfile)
|
|
|
|
s.source = self
|
|
|
|
idx << s
|
|
|
|
end
|
|
|
|
|
|
|
|
idx
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def api_fetchers
|
|
|
|
fetchers.select {|f| f.use_api && f.fetchers.first.api_fetcher? }
|
|
|
|
end
|
|
|
|
|
|
|
|
def remote_specs
|
|
|
|
@remote_specs ||= Index.build do |idx|
|
|
|
|
index_fetchers = fetchers - api_fetchers
|
|
|
|
|
|
|
|
# gather lists from non-api sites
|
|
|
|
fetch_names(index_fetchers, nil, idx, false)
|
|
|
|
|
|
|
|
# because ensuring we have all the gems we need involves downloading
|
|
|
|
# the gemspecs of those gems, if the non-api sites contain more than
|
|
|
|
# about 500 gems, we treat all sites as non-api for speed.
|
|
|
|
allow_api = idx.size < API_REQUEST_LIMIT && dependency_names.size < API_REQUEST_LIMIT
|
|
|
|
Bundler.ui.debug "Need to query more than #{API_REQUEST_LIMIT} gems." \
|
|
|
|
" Downloading full index instead..." unless allow_api
|
|
|
|
|
|
|
|
fetch_names(api_fetchers, allow_api && dependency_names, idx, false)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def fetch_names(fetchers, dependency_names, index, override_dupes)
|
|
|
|
fetchers.each do |f|
|
|
|
|
if dependency_names
|
2021-04-21 07:54:29 -04:00
|
|
|
Bundler.ui.info "Fetching gem metadata from #{URICredentialsFilter.credential_filtered_uri(f.uri)}", Bundler.ui.debug?
|
2018-11-02 19:07:56 -04:00
|
|
|
index.use f.specs_with_retry(dependency_names, self), override_dupes
|
|
|
|
Bundler.ui.info "" unless Bundler.ui.debug? # new line now that the dots are over
|
|
|
|
else
|
2021-04-21 07:54:29 -04:00
|
|
|
Bundler.ui.info "Fetching source index from #{URICredentialsFilter.credential_filtered_uri(f.uri)}"
|
2018-11-02 19:07:56 -04:00
|
|
|
index.use f.specs_with_retry(nil, self), override_dupes
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def fetch_gem(spec)
|
|
|
|
return false unless spec.remote
|
|
|
|
|
|
|
|
spec.fetch_platform
|
|
|
|
|
2021-10-20 18:04:09 -04:00
|
|
|
cache_path = download_cache_path(spec) || default_cache_path_for(rubygems_dir)
|
|
|
|
gem_path = "#{cache_path}/#{spec.file_name}"
|
|
|
|
|
|
|
|
if requires_sudo?
|
|
|
|
download_path = Bundler.tmp(spec.full_name)
|
|
|
|
download_cache_path = default_cache_path_for(download_path)
|
|
|
|
else
|
|
|
|
download_cache_path = cache_path
|
|
|
|
end
|
2018-11-02 19:07:56 -04:00
|
|
|
|
2021-10-20 12:49:34 -04:00
|
|
|
SharedHelpers.filesystem_access(download_cache_path) do |p|
|
2018-11-02 19:07:56 -04:00
|
|
|
FileUtils.mkdir_p(p)
|
|
|
|
end
|
2021-10-20 12:49:34 -04:00
|
|
|
download_gem(spec, download_cache_path)
|
2018-11-02 19:07:56 -04:00
|
|
|
|
|
|
|
if requires_sudo?
|
2021-10-20 18:04:09 -04:00
|
|
|
SharedHelpers.filesystem_access(cache_path) do |p|
|
2018-11-02 19:07:56 -04:00
|
|
|
Bundler.mkdir_p(p)
|
|
|
|
end
|
2021-10-20 13:27:06 -04:00
|
|
|
Bundler.sudo "mv #{download_cache_path}/#{spec.file_name} #{gem_path}"
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
gem_path
|
|
|
|
ensure
|
|
|
|
Bundler.rm_rf(download_path) if requires_sudo?
|
|
|
|
end
|
|
|
|
|
|
|
|
def installed?(spec)
|
2021-10-09 07:42:05 -04:00
|
|
|
installed_specs[spec].any? && !spec.deleted_gem?
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def requires_sudo?
|
|
|
|
Bundler.requires_sudo?
|
|
|
|
end
|
|
|
|
|
|
|
|
def rubygems_dir
|
|
|
|
Bundler.rubygems.gem_dir
|
|
|
|
end
|
|
|
|
|
2021-10-20 18:04:09 -04:00
|
|
|
def default_cache_path_for(dir)
|
|
|
|
"#{dir}/cache"
|
2021-10-20 13:02:21 -04:00
|
|
|
end
|
|
|
|
|
2018-11-02 19:07:56 -04:00
|
|
|
def cache_path
|
|
|
|
Bundler.app_cache
|
|
|
|
end
|
|
|
|
|
2020-10-15 00:20:25 -04:00
|
|
|
private
|
2018-11-02 19:07:56 -04:00
|
|
|
|
|
|
|
# Checks if the requested spec exists in the global cache. If it does,
|
|
|
|
# we copy it to the download path, and if it does not, we download it.
|
|
|
|
#
|
|
|
|
# @param [Specification] spec
|
|
|
|
# the spec we want to download or retrieve from the cache.
|
|
|
|
#
|
2021-10-20 12:49:34 -04:00
|
|
|
# @param [String] download_cache_path
|
2018-11-02 19:07:56 -04:00
|
|
|
# the local directory the .gem will end up in.
|
|
|
|
#
|
2021-10-20 12:49:34 -04:00
|
|
|
def download_gem(spec, download_cache_path)
|
2021-10-20 18:04:09 -04:00
|
|
|
uri = spec.remote.uri
|
|
|
|
Bundler.ui.confirm("Fetching #{version_message(spec)}")
|
|
|
|
Bundler.rubygems.download_gem(spec, uri, download_cache_path)
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# Returns the global cache path of the calling Rubygems::Source object.
|
|
|
|
#
|
|
|
|
# Note that the Source determines the path's subdirectory. We use this
|
|
|
|
# subdirectory in the global cache path so that gems with the same name
|
|
|
|
# -- and possibly different versions -- from different sources are saved
|
|
|
|
# to their respective subdirectories and do not override one another.
|
|
|
|
#
|
|
|
|
# @param [Gem::Specification] specification
|
|
|
|
#
|
|
|
|
# @return [Pathname] The global cache path.
|
|
|
|
#
|
|
|
|
def download_cache_path(spec)
|
|
|
|
return unless Bundler.feature_flag.global_gem_cache?
|
|
|
|
return unless remote = spec.remote
|
|
|
|
return unless cache_slug = remote.cache_slug
|
|
|
|
|
2021-10-20 18:04:09 -04:00
|
|
|
Bundler.user_cache.join("gems", cache_slug)
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def extension_cache_slug(spec)
|
|
|
|
return unless remote = spec.remote
|
|
|
|
remote.cache_slug
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|