#!/usr/bin/env ruby module Gem class CommandLineError < Gem::Exception; end module CommandAids def get_one_gem_name args = options[:args] if args.nil? or args.empty? fail Gem::CommandLineError, "Please specify a gem name on the command line (e.g. gem build GEMNAME)" end if args.size > 1 fail Gem::CommandLineError, "Too many gem names (#{args.join(', ')}); please specify only one" end args.first end def get_one_optional_argument args = options[:args] || [] args.first end def begins?(long, short) return false if short.nil? long[0, short.length] == short end end module LocalRemoteOptions def add_local_remote_options add_option('-l', '--local', 'Restrict operations to the LOCAL domain (default)') do |value, options| options[:domain] = :local end add_option('-r', '--remote', 'Restrict operations to the REMOTE domain') do |value, options| options[:domain] = :remote end add_option('-b', '--both', 'Allow LOCAL and REMOTE operations') do |value, options| options[:domain] = :both end end def local? options[:domain] == :local || options[:domain] == :both end def remote? options[:domain] == :remote || options[:domain] == :both end end module InstallUpdateOptions def add_install_update_options add_option('-i', '--install-dir DIR', '') do |value, options| options[:install_dir] = value end add_option('-d', '--[no-]rdoc', 'Generate RDoc documentation for the gem on install') do |value, options| options[:generate_rdoc] = value end add_option('-f', '--[no-]force', 'Force gem to install, bypassing dependency checks') do |value, options| options[:force] = value end add_option('-t', '--[no-]test', 'Run unit tests prior to installation') do |value, options| options[:test] = value end add_option('--ignore-dependencies', 'Do not install any required dependent gems') do |value, options| options[:ignore_dependencies] = value end add_option('--include-dependencies', 'Unconditionally install the required dependent gems') do |value, options| options[:include_dependencies] = value end end def install_update_defaults_str '--rdoc --no-force --no-test' end end module VersionOption def add_version_option(taskname) add_option('-v', '--version VERSION', "Specify version of gem to #{taskname}") do |value, options| options[:version] = value end end end #################################################################### class InstallCommand < Command include CommandAids include VersionOption include LocalRemoteOptions include InstallUpdateOptions def initialize super( 'install', 'Install a gem into the local repository', { :domain => :both, :generate_rdoc => true, :force => false, :test => false, :version => "> 0", :install_dir => Gem.dir }) add_version_option('install') add_local_remote_options add_install_update_options end def usage "#{program_name} GEMNAME" end def arguments "GEMNAME name of gem to install" end def defaults_str "--both --version '> 0' --rdoc --no-force --no-test\n" + "--install-dir #{Gem.dir}" end def execute ENV['GEM_PATH'] = options[:install_dir] # TODO: If a dependency isn't met, first check to see if it's in # the install list if(options[:args].empty?) fail Gem::CommandLineError, "Please specify a gem name on the command line (e.g. gem build GEMNAME)" end options[:args].each do |gem_name| if local? begin say "Attempting local installation of '#{gem_name}'" entries = [] if(File.exist?(gem_name) && !File.directory?(gem_name)) entries << gem_name else filepattern = gem_name + "*.gem" entries = Dir[filepattern] end unless entries.size > 0 if options[:domain] == :both say "Local gem file not found: #{filepattern}" else alert_error "Local gem file not found: #{filepattern}" end else result = Gem::Installer.new(entries.last, options).install( options[:force], options[:install_dir]) installed_gems = [result].flatten say "Successfully installed #{installed_gems[0].name}, " + "version #{installed_gems[0].version}" if installed_gems end rescue LocalInstallationError => e say " -> Local installation can't proceed: #{e.message}" rescue Gem::LoadError => e say " -> Local installation can't proceed due to LoadError: #{e.message}" rescue => e # TODO: Fix this handle to allow the error to propagate to # the top level handler. Example the other errors as # well. This implementation here looks suspicious to me -- # JimWeirich (4/Jan/05) alert_error "Error installing gem #{gem_name}[.gem]: #{e.message}" return end end if remote? && installed_gems.nil? say "Attempting remote installation of '#{gem_name}'" installer = Gem::RemoteInstaller.new(options) installed_gems = installer.install( gem_name, options[:version], options[:force], options[:install_dir]) if installed_gems installed_gems.compact! installed_gems.each do |spec| say "Successfully installed #{spec.full_name}" end end end unless installed_gems alert_error "Could not install a local or remote copy of the gem: #{gem_name}" terminate_interaction(1) end if options[:generate_rdoc] installed_gems.each do |gem| Gem::DocManager.new(gem, options[:rdoc_args]).generate_rdoc end # TODO: catch exceptions and inform user that doc generation was not successful. end if options[:test] installed_gems.each do |spec| gem_spec = Gem::SourceIndex.from_installed_gems.search(spec.name, spec.version.version).first result = Gem::Validator.new.unit_test(gem_spec) unless result.passed? unless ask_yes_no("...keep Gem?", true) then Gem::Uninstaller.new(spec.name, spec.version.version).uninstall end end end end end end end #################################################################### class UninstallCommand < Command include VersionOption include CommandAids def initialize super('uninstall', 'Uninstall a gem from the local repository', {:version=>"> 0"}) add_option('-a', '--[no-]all', 'Uninstall all matching versions' ) do |value, options| options[:all] = value end add_option('-i', '--[no-]ignore-dependencies', 'Ignore dependency requirements while uninstalling' ) do |value, options| options[:ignore] = value end add_option('-x', '--[no-]executables', 'Uninstall applicable executables without confirmation' ) do |value, options| options[:executables] = value end add_version_option('uninstall') end def defaults_str "--version '> 0' --no-force" end def usage "#{program_name} GEMNAME" end def arguments "GEMNAME name of gem to uninstall" end def execute gem_name = get_one_gem_name say "Attempting to uninstall gem '#{gem_name}'" Gem::Uninstaller.new(gem_name, options).uninstall end end #################################################################### class DependencyCommand < Command include VersionOption include CommandAids def initialize super('dependency', 'Show the dependencies of an installed gem', {:version=>"> 0"}) add_version_option('uninstall') add_option('-r', '--[no-]reverse-dependencies', 'Include reverse dependencies in the output' ) do |value, options| options[:reverse_dependencies] = value end add_option('-p', '--pipe', "Pipe Format (name --version ver)") do |value, options| options[:pipe_format] = value end end def defaults_str "--version '> 0' --no-reverse" end def usage "#{program_name} GEMNAME" end def arguments "GEMNAME name of gems to show" end def execute specs = {} srcindex = SourceIndex.from_installed_gems options[:args] << '.' if options[:args].empty? options[:args].each do |name| speclist = srcindex.search(name, options[:version]) if speclist.empty? say "No match found for #{name} (#{options[:version]})" else speclist.each do |spec| specs[spec.full_name] = spec end end end reverse = Hash.new { |h, k| h[k] = [] } if options[:reverse_dependencies] specs.values.each do |spec| reverse[spec.full_name] = find_reverse_dependencies(spec, srcindex) end end if options[:pipe_format] specs.values.sort.each do |spec| unless spec.dependencies.empty? spec.dependencies.each do |dep| puts %{#{dep.name} --version '#{dep.version_requirements}'} end end end else response = '' specs.values.sort.each do |spec| response << "Gem #{spec.full_name}\n" unless spec.dependencies.empty? response << " Requires\n" spec.dependencies.each do |dep| response << " #{dep}\n" end end unless reverse[spec.full_name].empty? response << " Used by\n" reverse[spec.full_name].each do |sp, dep| response << " #{sp} (#{dep})\n" end end response << "\n" end say response end end # Retuns list of [specification, dep] that are satisfied by spec. def find_reverse_dependencies(spec, srcindex) result = [] srcindex.each do |name, sp| sp.dependencies.each do |dep| if spec.name == dep.name && dep.version_requirements.satisfied_by?(spec.version) result << [sp.full_name, dep] end end end result end end #################################################################### class CheckCommand < Command include CommandAids def initialize super('check', 'Check installed gems', {:verify => false, :alien => false}) add_option('-v', '--verify FILE', 'Verify gem file against its internal checksum') do |value, options| options[:verify] = value end add_option('-a', '--alien', "Report 'unmanaged' or rogue files in the gem repository") do |value, options| options[:alien] = true end add_option('-t', '--test', "Run unit tests for gem") do |value, options| options[:test] = true end add_option('-V', '--version', "Specify version for which to run unit tests") do |value, options| options[:version] = value end end def execute if options[:test] version = options[:version] || "> 0.0.0" gem_spec = Gem::SourceIndex.from_installed_gems.search(get_one_gem_name, version).first Gem::Validator.new.unit_test(gem_spec) end if options[:alien] say "Performing the 'alien' operation" Gem::Validator.new.alien.each do |key, val| if(val.size > 0) say "#{key} has #{val.size} problems" val.each do |error_entry| say "\t#{error_entry.path}:" say "\t#{error_entry.problem}" say end else say "#{key} is error-free" end say end end if options[:verify] gem_name = options[:verify] unless gem_name alert_error "Must specifiy a .gem file with --verify NAME" return end unless File.exist?(gem_name) alert_error "Unknown file: #{gem_name}." return end say "Verifying gem: '#{gem_name}'" begin Gem::Validator.new.verify_gem_file(gem_name) rescue Exception => e alert_error "#{gem_name} is invalid." end end end end # class #################################################################### class BuildCommand < Command include CommandAids def initialize super('build', 'Build a gem from a gemspec') end def usage "#{program_name} GEMSPEC_FILE" end def arguments "GEMSPEC_FILE name of gemspec file used to build the gem" end def execute gemspec = get_one_gem_name if File.exist?(gemspec) say "Attempting to build gem spec '#{gemspec}'" specs = load_gemspecs(gemspec) specs.each do |spec| Gem::Builder.new(spec).build end return else alert_error "Gemspec file not found: #{gemspec}" end end def load_gemspecs(filename) if yaml?(filename) require 'yaml' result = [] open(filename) do |f| begin while spec = Gem::Specification.from_yaml(f) result << spec end rescue EndOfYAMLException => e # OK end end else result = [Gem::Specification.load(filename)] end result end def yaml?(filename) line = open(filename) { |f| line = f.gets } result = line =~ %r{^--- *!ruby/object:Gem::Specification} result end end #################################################################### class QueryCommand < Command include LocalRemoteOptions def initialize(name='query', summary='Query gem information in local or remote repositories') super(name, summary, {:name=>/.*/, :domain=>:local, :details=>false} ) add_option('-n', '--name-matches REGEXP', 'Name of gem(s) to query on maches the provided REGEXP') do |value, options| options[:name] = /#{value}/i end add_option('-d', '--[no-]details', 'Display detailed information of gem(s)') do |value, options| options[:details] = value end add_local_remote_options end def defaults_str "--local --name-matches '.*' --no-details" end def execute if local? say say "*** LOCAL GEMS ***" output_query_results(Gem::cache.search(options[:name])) end if remote? say say "*** REMOTE GEMS ***" begin output_query_results(Gem::RemoteInstaller.new(options).search(options[:name])) rescue Gem::RemoteSourceException => e alert_error e.to_s end end end private def output_query_results(gemspecs) gem_list_with_version = {} gemspecs.flatten.each do |gemspec| gem_list_with_version[gemspec.name] ||= [] gem_list_with_version[gemspec.name] << gemspec end gem_list_with_version = gem_list_with_version.sort do |first, second| first[0].downcase <=> second[0].downcase end gem_list_with_version.each do |gem_name, list_of_matching| say list_of_matching = list_of_matching.sort_by { |x| x.version }.reverse seen_versions = [] list_of_matching.delete_if do |item| if(seen_versions.member?(item.version)) true else seen_versions << item.version false end end say "#{gem_name} (#{list_of_matching.map{|gem| gem.version.to_s}.join(", ")})" say format_text(list_of_matching[0].summary, 68, 4) end end ## # Used for wrapping and indenting text # def format_text(text, wrap, indent=0) result = [] pattern = Regexp.new("^(.{0,#{wrap}})[ \n]") work = text.dup while work.length > wrap if work =~ pattern result << $1 work.slice!(0, $&.length) else result << work.slice!(0, wrap) end end result << work if work.length.nonzero? result.join("\n").gsub(/^/, " " * indent) end end #################################################################### class ListCommand < QueryCommand include CommandAids def initialize super( 'list', 'Display all gems whose name starts with STRING' ) remove_option('--name-matches') end def defaults_str "--local --no-details" end def usage "#{program_name} [STRING]" end def arguments "STRING start of gem name to look for" end def execute string = get_one_optional_argument || '' options[:name] = /^#{string}/i super end end #################################################################### class SearchCommand < QueryCommand include CommandAids def initialize super( 'search', 'Display all gems whose name contains STRING' ) remove_option('--name-matches') end def defaults_str "--local --no-details" end def usage "#{program_name} [STRING]" end def arguments "STRING fragment of gem name to look for" end def execute string = get_one_optional_argument options[:name] = /#{string}/i super end end #################################################################### class UpdateCommand < Command include InstallUpdateOptions def initialize super( 'update', 'Upgrade currently installed gems in the local repository', { :generate_rdoc => true, :force => false, :test => false, :install_dir => Gem.dir }) add_install_update_options add_option('--system', 'Update the RubyGems system software') do |value, options| options[:system] = value end end def defaults_str "--rdoc --no-force --no-test\n" + "--install-dir #{Gem.dir}" end def arguments "GEMNAME(s) name of gem(s) to update" end def execute if options[:system] say "Upgrading RubyGems..." if ! options[:args].empty? fail "No gem names are allowed with the --system option" end options[:args] = ["rubygems-update"] else say "Upgrading installed gems..." end hig = highest_installed_gems = {} Gem::SourceIndex.from_installed_gems.each do |name, spec| if hig[spec.name].nil? or hig[spec.name].version < spec.version hig[spec.name] = spec end end remote_gemspecs = Gem::RemoteInstaller.new(options).search(//) # For some reason, this is an array of arrays. The actual list # of specifications is the first and only element. If there # were more remote sources, perhaps there would be more. remote_gemspecs = remote_gemspecs.flatten gems_to_update = if(options[:args].empty?) then which_to_update(highest_installed_gems, remote_gemspecs) else options[:args] end options[:domain] = :remote # install from remote source install_command = command_manager['install'] gems_to_update.uniq.sort.each do |name| say "Attempting remote upgrade of #{name}" options[:args] = [name] install_command.merge_options(options) install_command.execute end if gems_to_update.include?("rubygems-update") say "Updating version of RubyGems" do_rubygems_update end if(options[:system]) then say "RubyGems system software updated" else say "Gems: [#{gems_to_update.uniq.sort.collect{|g| g.to_s}.join(', ')}] updated" end end def do_rubygems_update # Need to clear out the argument list because the # update_rubygems script expects to handle command line # argument. ARGV.clear require_gem 'rubygems-update' load 'update_rubygems' end def which_to_update(highest_installed_gems, remote_gemspecs) result = [] highest_installed_gems.each do |l_name, l_spec| highest_remote_gem = remote_gemspecs.select { |spec| spec.name == l_name }. sort_by { |spec| spec.version }. last if highest_remote_gem and l_spec.version < highest_remote_gem.version result << l_name end end result end end #################################################################### class CleanupCommand < Command def initialize super( 'cleanup', 'Cleanup old versions of installed gems in the local repository', { :force => false, :test => false, :install_dir => Gem.dir }) add_option('-d', '--dryrun', "") do |value, options| options[:dryrun] = true end end def defaults_str "--no-dryrun" end def arguments "GEMNAME(s) name of gem(s) to cleanup" end def execute say "Cleaning up installed gems..." srcindex = Gem::SourceIndex.from_installed_gems primary_gems = {} srcindex.each do |name, spec| if primary_gems[spec.name].nil? or primary_gems[spec.name].version < spec.version primary_gems[spec.name] = spec end end gems_to_cleanup = [] if ! options[:args].empty? options[:args].each do |gem_name| specs = Gem.cache.search(/^#{gem_name}$/i) specs.each do |spec| gems_to_cleanup << spec end end else srcindex.each do |name, spec| gems_to_cleanup << spec end end gems_to_cleanup = gems_to_cleanup.select { |spec| primary_gems[spec.name].version != spec.version } uninstall_command = command_manager['uninstall'] deplist = DependencyList.new gems_to_cleanup.uniq.each do |spec| deplist.add(spec) end deplist.dependency_order.each do |spec| if options[:dryrun] say "Dry Run Mode: Would uninstall #{spec.full_name}" else say "Attempting uninstall on #{spec.full_name}" options[:args] = [spec.name] options[:version] = "= #{spec.version}" options[:executables] = true uninstall_command.merge_options(options) begin uninstall_command.execute rescue Gem::DependencyRemovalException => ex say "Unable to uninstall #{spec.full_name} ... continuing with remaining gems" end end end say "Clean Up Complete" end end #################################################################### class RDocCommand < Command include VersionOption include CommandAids def initialize super('rdoc', 'Generates RDoc for pre-installed gems', {:version=>"> 0.0.0"}) add_option('--all', 'Generate RDoc documentation for all installed gems') do |value, options| options[:all] = value end add_version_option('rdoc') end def defaults_str "--version '> 0.0.0'" end def usage "#{program_name} [args]" end def arguments "GEMNAME The gem to generate RDoc for (unless --all)" end def execute if options[:all] Gem::SourceIndex.from_installed_gems.each do |name, spec| say "Doing gem #{spec.name}" Gem::DocManager.new(spec).generate_rdoc end else gem_name = get_one_gem_name specs = Gem::SourceIndex.from_installed_gems.search(gem_name, options[:version]) if specs.empty? #version = options[:version] || "> 0.0.0" fail "Failed to find gem #{gem_name} to generate RDoc for #{options[:version]}" end specs.each do |spec| Gem::DocManager.new(spec).generate_rdoc end end true end end #################################################################### class EnvironmentCommand < Command include CommandAids def initialize super('environment', 'Display RubyGems environmental information') end def usage "#{program_name} [args]" end def arguments args = <<-EOF packageversion display the package version gemdir display the path where gems are installed gempath display path used to search for gems version display the gem format version remotesources display the remote gem servers display everything EOF return args.gsub(/^\s+/, '') end def execute out = '' arg = options[:args][0] if begins?("packageversion", arg) out = Gem::RubyGemsPackageVersion.to_s elsif begins?("version", arg) out = Gem::RubyGemsVersion.to_s elsif begins?("gemdir", arg) out = Gem.dir elsif begins?("gempath", arg) Gem.path.collect { |p| out << "#{p}\n" } elsif begins?("remotesources", arg) Gem::RemoteInstaller.new.sources.collect do |s| out << "#{s}\n" end elsif arg fail Gem::CommandLineError, "Unknown enviroment option [#{arg}]" else out = "Rubygems Environment:\n" out << " - VERSION: #{Gem::RubyGemsVersion} (#{Gem::RubyGemsPackageVersion})\n" out << " - INSTALLATION DIRECTORY: #{Gem.dir}\n" out << " - GEM PATH:\n" Gem.path.collect { |p| out << " - #{p}\n" } out << " - REMOTE SOURCES:\n" Gem::RemoteInstaller.new.sources.collect do |s| out << " - #{s}\n" end end say out true end end #################################################################### class SpecificationCommand < Command include VersionOption include LocalRemoteOptions include CommandAids def initialize super('specification', 'Display gem specification (in yaml)', {:domain=>:local, :version=>"> 0.0.0"}) add_version_option('examine') add_local_remote_options add_option('--all', 'Output specifications for all versions of the gem') do options[:all] = true end end def defaults_str "--local --version '(latest)'" end def usage "#{program_name} GEMFILE" end def arguments "GEMFILE Name of a .gem file to examine" end def execute if local? gem = get_one_gem_name gem_specs = Gem::SourceIndex.from_installed_gems.search(gem, options[:version]) unless gem_specs.empty? require 'yaml' output = lambda { |spec| say spec.to_yaml; say "\n" } if options[:all] gem_specs.each(&output) else spec = gem_specs.sort_by { |spec| spec.version }.last output[spec] end else alert_error "Unknown gem #{gem}" end end if remote? say "(Remote 'info' operation is not yet implemented.)" # NOTE: when we do implement remote info, make sure we don't duplicate huge swabs of # local data. If it's the same, just say it's the same. end end end #################################################################### class UnpackCommand < Command include VersionOption include CommandAids def initialize super( 'unpack', 'Unpack an installed gem to the current directory', { :version => '> 0' } ) add_version_option('unpack') end def defaults_str "--version '> 0'" end def usage "#{program_name} GEMNAME" end def arguments "GEMNAME Name of the gem to unpack" end # TODO: allow, e.g., 'gem unpack rake-0.3.1'. Find a general solution for this, so that it # works for uninstall as well. (And check other commands at the same time.) def execute gemname = get_one_gem_name path = get_path(gemname, options[:version]) if path require 'fileutils' target_dir = File.basename(path).sub(/\.gem$/, '') FileUtils.mkdir_p target_dir Installer.new(path).unpack(target_dir) say "Unpacked gem: '#{target_dir}'" else alert_error "Gem '#{gemname}' not installed." end end # Return the full path to the cached gem file matching the given name and version # requirement. Returns 'nil' if no match. Example: # # get_path('rake', '> 0.4') # -> '/usr/lib/ruby/gems/1.8/cache/rake-0.4.2.gem' # get_path('rake', '< 0.1') # -> nil # get_path('rak') # -> nil (exact name required) # # XXX: This should be refactored so that it's a general service. I don't think any of our # existing classes are the right place though. Just maybe 'Cache'? # # XXX: It just uses Gem.dir for now. What's an easy way to get the list of source directories? # def get_path(gemname, version_req) specs = SourceIndex.from_installed_gems.search(gemname, version_req) selected = specs.sort_by { |s| s.full_name }.last return nil if selected.nil? # We expect to find (basename).gem in the 'cache' directory. Furthermore, the name match # must be exact. if gemname == selected.name filename = selected.full_name + '.gem' return File.join(Gem.dir, 'cache', filename) else return nil end end end #################################################################### class HelpCommand < Command include CommandAids def initialize super('help', "Provide help on the 'gem' command") end def usage "#{program_name} ARGUMENT" end def arguments args = <<-EOF commands List all 'gem' commands examples Show examples of 'gem' usage Show specific help for EOF return args.gsub(/^\s+/, '') end def execute arg = options[:args][0] if begins?("commands", arg) require 'stringio' out = StringIO.new out.puts "\nGEM commands are:\n\n" desc_indent = command_manager.command_names.collect {|n| n.size}.max + 4 format = " %-#{desc_indent}s %s\n" command_manager.command_names.each do |cmd_name| out.printf format, "#{cmd_name}", command_manager[cmd_name].summary end out.puts "\nFor help on a particular command, use 'gem help COMMAND'." out.puts "\nCommands may be abbreviated, so long as they are unambiguous." out.puts "e.g. 'gem i rake' is short for 'gem install rake'." say out.string elsif begins?("options", arg) say Gem::HELP elsif begins?("examples", arg) say Gem::EXAMPLES elsif options[:help] command = command_manager[options[:help]] if command # help with provided command command.invoke("--help") else alert_error "Unknown command #{options[:help]}. Try 'gem help commands'" end elsif arg possibilities = command_manager.find_command_possibilities(arg.downcase) if possibilities.size == 1 command = command_manager[possibilities.first] command.invoke("--help") elsif possibilities.size > 1 alert_warning "Ambiguous command #{arg} (#{possibilities.join(', ')})" else alert_warning "Unknown command #{arg}. Try gem help commands" end else say Gem::HELP end end end end # module ## Documentation Constants module Gem HELP = %{ RubyGems is a sophisticated package manager for Ruby. This is a basic help message containing pointers to more information. Usage: gem -h/--help gem -v/--version gem command [arguments...] [options...] Examples: gem install rake gem list --local gem build package.gemspec gem help install Further help: gem help commands list all 'gem' commands gem help examples show some examples of usage gem help show help on COMMAND (e.g. 'gem help install') Further information: http://rubygems.rubyforge.org }.gsub(/^ /, "") EXAMPLES = %{ Some examples of 'gem' usage. * Install 'rake', either from local directory or remote server: gem install rake * Install 'rake', only from remote server: gem install rake --remote * Install 'rake' from remote server, and run unit tests, and generate RDocs: gem install --remote rake --test --rdoc * Install 'rake', but only version 0.3.1, even if dependencies are not met, and into a specific directory: gem install rake --version 0.3.1 --force --install-dir $HOME/.gems * List local gems whose name begins with 'D': gem list D * List local and remote gems whose name contains 'log': gem search log --both * List only remote gems whose name contains 'log': gem search log --remote * Uninstall 'rake': gem uninstall rake * Create a gem: See http://rubygems.rubyforge.org/wiki/wiki.pl?CreateAGemInTenMinutes * See information about RubyGems: gem environment }.gsub(/^ /, "") end