require 'rubygems'
require 'socket'
require 'fileutils'
module Gem
class DependencyError < Gem::Exception; end
class RemoteSourceException < Gem::Exception; end
class GemNotFoundException < Gem::Exception; end
class RemoteInstallationCancelled < Gem::Exception; end
# RemoteSourceFetcher handles the details of fetching gems and gem
# information from a remote source.
class RemoteSourceFetcher
include UserInteraction
# Initialize a remote fetcher using the source URI (and possible
# proxy information).
def initialize(source_uri, proxy)
@uri = source_uri
@http_proxy = proxy
if @http_proxy == true
@http_proxy = ENV['http_proxy'] || ENV['HTTP_PROXY']
end
end
# The uncompressed +size+ of the source's directory (e.g. source
# info).
def size
@size ||= get_size("/yaml")
end
# Fetch the data from the source at the given path.
def fetch_path(path="")
read_data(@uri + path)
end
# Get the source index from the gem source. The source index is a
# directory of the gems available on the source, formatted as a
# Gem::Cache object. The cache object allows easy searching for
# gems by name and version requirement.
#
# Notice that the gem specs in the cache are adequate for searches
# and queries, but may have some information elided (hence
# "abbreviated").
def source_index
say "Updating Gem source index for: #{@uri}"
begin
require 'zlib'
yaml_spec = fetch_path("/yaml.Z")
yaml_spec = Zlib::Inflate.inflate(yaml_spec)
rescue
yaml_spec = nil
end
begin
yaml_spec = fetch_path("/yaml") unless yaml_spec
r = convert_spec(yaml_spec)
rescue SocketError => e
raise RemoteSourceException.new("Error fetching remote gem cache: #{e.to_s}")
end
end
private
# Connect to the source host/port, using a proxy if needed.
def connect_to(host, port)
if @http_proxy
proxy_uri = URI.parse(@http_proxy)
Net::HTTP::Proxy(proxy_uri.host, proxy_uri.port).new(host, port)
else
Net::HTTP.new(host, port)
end
end
# Get the size of the (non-compressed) data from the source at the
# given path.
def get_size(path)
read_size(@uri + path)
end
# Read the size of the (source based) URI using an HTTP HEAD
# command.
def read_size(uri)
require 'net/http'
require 'uri'
u = URI.parse(uri)
http = connect_to(u.host, u.port)
path = (u.path == "") ? "/" : u.path
resp = http.head(path)
fail RemoteSourceException, "HTTP Response #{resp.code}" if resp.code !~ /^2/
resp['content-length'].to_i
end
# Read the data from the (source based) URI.
def read_data(uri)
require 'rubygems/open-uri'
begin
open(uri,
"User-Agent" => "RubyGems/#{Gem::RubyGemsVersion}",
:proxy => @http_proxy
) do |input|
input.read
end
rescue
old_uri = uri
uri = uri.downcase
retry if old_uri != uri
raise
end
end
# Convert the yamlized string spec into a real spec (actually,
# these are hashes of specs.).
def convert_spec(yaml_spec)
YAML.load(reduce_spec(yaml_spec)) or
raise "Didn't get a valid YAML document"
end
# This reduces the source spec in size so that YAML bugs with
# large data sets will be dodged. Obviously this is a workaround,
# but it allows Gems to continue to work until the YAML bug is
# fixed.
def reduce_spec(yaml_spec)
result = ""
state = :copy
yaml_spec.each do |line|
if state == :copy && line =~ /^\s+files:\s*$/
state = :skip
result << line.sub(/$/, " []")
elsif state == :skip
if line !~ /^\s+-/
state = :copy
end
end
result << line if state == :copy
end
result
end
class << self
# Sent by the client when it is done with all the sources,
# allowing any cleanup activity to take place.
def finish
# Nothing to do
end
end
end
# LocalSourceInfoCache implements the cache management policy on
# where the source info is stored on local file system. There are
# two possible cache locations: (1) the system wide cache, and (2)
# the user specific cache.
#
# * The system cache is prefered if it is writable (or can be
# created).
# * The user cache is used if the system cache is not writable (or
# can not be created).
#
# Once a cache is selected, it will be used for all operations. It
# will not switch between cache files dynamically.
#
class LocalSourceInfoCache
# The most recent cache data.
def cache_data
@dirty = false
@cache_data ||= read_cache
end
# Write data to the proper cache.
def write_cache
data = cache_data
open(writable_file, "w") do |f|
f.puts Marshal.dump(data)
end
end
# The name of the system cache file.
def system_cache_file
@sysetm_cache ||= File.join(Gem.dir, "source_cache")
end
# The name of the user cache file.
def user_cache_file
@user_cache ||=
ENV['GEMCACHE'] || File.join(Gem.user_home, ".gem/source_cache")
end
# Mark the cache as updated (i.e. dirty).
def update
@dirty = true
end
# Write the cache to a local file (if it is dirty).
def flush
write_cache if @dirty
@dirty = false
end
private
# Find a writable cache file.
def writable_file
@cache_file
end
# Read the most current cache data.
def read_cache
@cache_file = select_cache_file
begin
open(@cache_file) { |f| load_local_cache(f) } || {}
rescue StandardError => ex
{}
end
end
def load_local_cache(f)
Marshal.load(f)
rescue StandardError => ex
{}
end
# Select a writable cache file
def select_cache_file
try_file(system_cache_file) or
try_file(user_cache_file) or
fail "Unable to locate a writable cache file."
end
# Determine if +fn+ is a candidate for a cache file. Return fn if
# it is. Return nil if it is not.
def try_file(fn)
return fn if File.writable?(fn)
return nil if File.exist?(fn)
dir = File.dirname(fn)
if ! File.exist? dir
begin
FileUtils.mkdir_p(dir)
rescue RuntimeError
return nil
end
end
if File.writable?(dir)
FileUtils.touch fn
return fn
end
nil
end
end
# CachedFetcher is a decorator that adds local file caching to
# RemoteSourceFetcher objects.
class CachedFetcher
# Create a cached fetcher (based on a RemoteSourceFetcher) for the
# source at +source_uri+ (through the proxy +proxy+).
def initialize(source_uri, proxy)
@source_uri = source_uri
@fetcher = RemoteSourceFetcher.new(source_uri, proxy)
end
# The uncompressed +size+ of the source's directory (e.g. source
# info).
def size
@fetcher.size
end
# Fetch the data from the source at the given path.
def fetch_path(path="")
@fetcher.fetch_path(path)
end
# Get the source index from the gem source. The source index is a
# directory of the gems available on the source, formatted as a
# Gem::Cache object. The cache object allows easy searching for
# gems by name and version requirement.
#
# Notice that the gem specs in the cache are adequate for searches
# and queries, but may have some information elided (hence
# "abbreviated").
def source_index
cache = manager.cache_data[@source_uri]
if cache && cache['size'] == @fetcher.size
cache['cache']
else
result = @fetcher.source_index
manager.cache_data[@source_uri] = {
'size' => @fetcher.size,
'cache' => result,
}
manager.update
result
end
end
# Flush the cache to a local file, if needed.
def flush
manager.flush
end
private
# The cache manager for this cached source.
def manager
self.class.manager
end
# The cache is shared between all caching fetchers, so the cache
# is put in the class object.
class << self
# The Cache manager for all instances of this class.
def manager
@manager ||= LocalSourceInfoCache.new
end
# Sent by the client when it is done with all the sources,
# allowing any cleanup activity to take place.
def finish
manager.flush
end
end
end
class RemoteInstaller
include UserInteraction
# http_proxy::
# * [String]: explicit specification of proxy; overrides any
# environment variable setting
# * nil: respect environment variables
# * :no_proxy: ignore environment variables and _don't_
# use a proxy
#
def initialize(options={})
# Ensure http_proxy env vars are used if no proxy explicitly supplied.
@options = options
@http_proxy =
case @options[:http_proxy]
when :no_proxy
false
when nil
true
else
@options[:http_proxy].to_str
end
@fetcher_class = CachedFetcher
end
# This method will install package_name onto the local system.
#
# gem_name::
# [String] Name of the Gem to install
#
# version_requirement::
# [default = "> 0.0.0"] Gem version requirement to install
#
# Returns::
# an array of Gem::Specification objects, one for each gem installed.
#
def install(gem_name,
version_requirement = "> 0.0.0",
force=false,
install_dir=Gem.dir,
install_stub=true)
unless version_requirement.respond_to?(:satisfied_by?)
version_requirement = Version::Requirement.new(version_requirement)
end
installed_gems = []
caches = source_index_hash
spec, source = find_gem_to_install(gem_name, version_requirement, caches)
dependencies = find_dependencies_not_installed(spec.dependencies)
installed_gems << install_dependencies(dependencies, force, install_dir)
cache_dir = File.join(install_dir, "cache")
destination_file = File.join(cache_dir, spec.full_name + ".gem")
download_gem(destination_file, source, spec)
installer = new_installer(destination_file)
installed_gems.unshift installer.install(force, install_dir, install_stub)
installed_gems.flatten
end
# Search Gem repository for a gem by specifying all or part of
# the Gem's name
def search(pattern_to_match)
results = []
caches = source_index_hash
caches.each do |cache|
results << cache[1].search(pattern_to_match)
end
results
end
# Return a list of the sources that we can download gems from
def sources
unless @sources
require_gem("sources")
@sources = Gem.sources
end
@sources
end
# Return a hash mapping the available source names to the source
# index of that source.
def source_index_hash
result = {}
sources.each do |source|
result[source] = fetch_source(source)
end
@fetcher_class.finish
result
end
# Return the source info for the given source. The
def fetch_source(source)
rsf = @fetcher_class.new(source, @http_proxy)
rsf.source_index
end
# Find a gem to be installed by interacting with the user.
def find_gem_to_install(gem_name, version_requirement, caches)
max_version = Version.new("0.0.0")
specs_n_sources = []
caches.each do |source, cache|
cache.each do |name, spec|
if (/^#{gem_name}-/i === name &&
version_requirement.satisfied_by?(spec.version))
specs_n_sources << [spec, source]
end
end
end
if specs_n_sources.size == 0
raise GemNotFoundException.new(
"Could not find #{gem_name} (#{version_requirement}) in the repository")
end
specs_n_sources = specs_n_sources.sort_by { |x| x[0].version }.reverse
if specs_n_sources.reject { |item|
item[0].platform.nil? || item[0].platform==Platform::RUBY
}.size == 0
# only non-binary gems...return latest
return specs_n_sources.first
end
list = specs_n_sources.collect {|item|
"#{item[0].name} #{item[0].version} (#{item[0].platform.to_s})"
}
list << "Cancel installation"
string, index = choose_from_list(
"Select which gem to install for your platform (#{RUBY_PLATFORM})",
list)
raise RemoteInstallationCancelled.new("Installation of #{gem_name} cancelled.") if
index == (list.size - 1)
specs_n_sources[index]
end
def find_dependencies_not_installed(dependencies)
to_install = []
dependencies.each do |dependency|
begin
require_gem(dependency.name, *dependency.requirement_list)
rescue LoadError => e
to_install.push dependency
end
end
to_install
end
# Install all the given dependencies. Returns an array of
# Gem::Specification objects, one for each dependency installed.
#
# TODO: For now, we recursively install, but this is not the right
# way to do things (e.g. if a package fails to download, we
# shouldn't install anything).
def install_dependencies(dependencies, force, install_dir)
installed_gems = []
dependencies.each do |dependency|
if ask_yes_no("Install required dependency #{dependency.name}?", true)
remote_installer = RemoteInstaller.new(@options)
installed_gems << remote_installer.install(
dependency.name,
dependency.version_requirements,
force,
install_dir)
else
raise DependencyError.new("Required dependency #{dependency.name} not installed")
end
end
installed_gems
end
def download_gem(destination_file, source, spec)
rsf = @fetcher_class.new(source, @http_proxy)
path = "/gems/#{spec.full_name}.gem"
response = rsf.fetch_path(path)
write_gem_to_file(response, destination_file)
end
def write_gem_to_file(body, destination_file)
FileUtils.mkdir_p(File.dirname(destination_file)) unless File.exist?(destination_file)
File.open(destination_file, 'wb') do |out|
out.write(body)
end
end
def new_installer(gem)
return Installer.new(gem)
end
end
end