In Files
- rubygems/commands/unpack_command.rb
 
Parent
Included Modules
Class/Module Index
          ![show/hide quicksearch [+]](../../images/find.png)
        
        - Gem::AvailableSet
 - Gem::BasicSpecification
 - Gem::Command
 - Gem::CommandLineError
 - Gem::CommandManager
 - Gem::Commands
 - Gem::Commands::BuildCommand
 - Gem::Commands::CertCommand
 - Gem::Commands::CheckCommand
 - Gem::Commands::CleanupCommand
 - Gem::Commands::ContentsCommand
 - Gem::Commands::DependencyCommand
 - Gem::Commands::EnvironmentCommand
 - Gem::Commands::FetchCommand
 - Gem::Commands::GenerateIndexCommand
 - Gem::Commands::HelpCommand
 - Gem::Commands::InstallCommand
 - Gem::Commands::ListCommand
 - Gem::Commands::LockCommand
 - Gem::Commands::MirrorCommand
 - Gem::Commands::OpenCommand
 - Gem::Commands::OutdatedCommand
 - Gem::Commands::OwnerCommand
 - Gem::Commands::PristineCommand
 - Gem::Commands::PushCommand
 - Gem::Commands::QueryCommand
 - Gem::Commands::RdocCommand
 - Gem::Commands::SearchCommand
 - Gem::Commands::ServerCommand
 - Gem::Commands::SetupCommand
 - Gem::Commands::SourcesCommand
 - Gem::Commands::SpecificationCommand
 - Gem::Commands::StaleCommand
 - Gem::Commands::UninstallCommand
 - Gem::Commands::UnpackCommand
 - Gem::Commands::UpdateCommand
 - Gem::Commands::WhichCommand
 - Gem::Commands::YankCommand
 - Gem::ConfigFile
 - Gem::ConflictError
 - Gem::ConsoleUI
 - Gem::DefaultUserInteraction
 - Gem::Dependency
 - Gem::DependencyError
 - Gem::DependencyInstaller
 - Gem::DependencyList
 - Gem::DependencyRemovalException
 - Gem::DependencyResolutionError
 - Gem::Deprecate
 - Gem::Doctor
 - Gem::DocumentError
 - Gem::EndOfYAMLException
 - Gem::ErrorReason
 - Gem::Exception
 - Gem::Ext
 - Gem::Ext::BuildError
 - Gem::Ext::Builder
 - Gem::Ext::CmakeBuilder
 - Gem::Ext::ConfigureBuilder
 - Gem::Ext::ExtConfBuilder
 - Gem::Ext::RakeBuilder
 - Gem::FakeFetcher
 - Gem::FilePermissionError
 - Gem::FormatException
 - Gem::GemNotFoundException
 - Gem::GemNotInHomeException
 - Gem::GemRunner
 - Gem::GemcutterUtilities
 - Gem::ImpossibleDependenciesError
 - Gem::Indexer
 - Gem::InstallError
 - Gem::InstallUpdateOptions
 - Gem::Installer
 - Gem::Installer::FakePackage
 - Gem::InstallerTestCase
 - Gem::InvalidSpecificationException
 - Gem::Licenses
 - Gem::List
 - Gem::LoadError
 - Gem::LocalRemoteOptions
 - Gem::MockGemUi
 - Gem::MockGemUi::InputEOFError
 - Gem::MockGemUi::SystemExitException
 - Gem::MockGemUi::TTY
 - Gem::MockGemUi::TermError
 - Gem::NameTuple
 - Gem::NoAliasYAMLTree
 - Gem::OperationNotSupportedError
 - Gem::Package::DigestIO
 - Gem::Package::Error
 - Gem::Package::FormatError
 - Gem::Package::NonSeekableIO
 - Gem::Package::Old
 - Gem::Package::PathError
 - Gem::Package::TarHeader
 - Gem::Package::TarInvalidError
 - Gem::Package::TarReader
 - Gem::Package::TarReader::Entry
 - Gem::Package::TarReader::UnexpectedEOF
 - Gem::Package::TarTestCase
 - Gem::Package::TarWriter
 - Gem::Package::TarWriter::BoundedStream
 - Gem::Package::TarWriter::FileOverflow
 - Gem::Package::TarWriter::RestrictedStream
 - Gem::Package::TooLongFileName
 - Gem::PackageTask
 - Gem::PathSupport
 - Gem::Platform
 - Gem::PlatformMismatch
 - Gem::RemoteError
 - Gem::RemoteFetcher
 - Gem::RemoteFetcher::FetchError
 - Gem::RemoteFetcher::UnknownHostError
 - Gem::RemoteInstallationCancelled
 - Gem::RemoteInstallationSkipped
 - Gem::RemoteSourceException
 - Gem::Request::ConnectionPools
 - Gem::Request::ConnectionPools::Net
 - Gem::RequestSet
 - Gem::RequestSet::GemDependencyAPI
 - Gem::RequestSet::Lockfile
 - Gem::RequestSet::Lockfile::ParseError
 - Gem::RequestSet::Lockfile::Parser
 - Gem::RequestSet::Lockfile::Tokenizer
 - Gem::Requirement
 - Gem::Requirement::BadRequirementError
 - Gem::Resolver
 - Gem::Resolver::APISet
 - Gem::Resolver::APISpecification
 - Gem::Resolver::ActivationRequest
 - Gem::Resolver::BestSet
 - Gem::Resolver::ComposedSet
 - Gem::Resolver::Conflict
 - Gem::Resolver::CurrentSet
 - Gem::Resolver::DependencyRequest
 - Gem::Resolver::GitSet
 - Gem::Resolver::GitSpecification
 - Gem::Resolver::IndexSet
 - Gem::Resolver::IndexSpecification
 - Gem::Resolver::InstalledSpecification
 - Gem::Resolver::InstallerSet
 - Gem::Resolver::LocalSpecification
 - Gem::Resolver::LockSet
 - Gem::Resolver::LockSpecification
 - Gem::Resolver::Molinillo
 - Gem::Resolver::Molinillo::CircularDependencyError
 - Gem::Resolver::Molinillo::DependencyGraph
 - Gem::Resolver::Molinillo::DependencyGraph::Vertex
 - Gem::Resolver::Molinillo::DependencyState
 - Gem::Resolver::Molinillo::NoSuchDependencyError
 - Gem::Resolver::Molinillo::PossibilityState
 - Gem::Resolver::Molinillo::ResolutionState
 - Gem::Resolver::Molinillo::Resolver
 - Gem::Resolver::Molinillo::Resolver::Resolution
 - Gem::Resolver::Molinillo::ResolverError
 - Gem::Resolver::Molinillo::SpecificationProvider
 - Gem::Resolver::Molinillo::UI
 - Gem::Resolver::Molinillo::VersionConflict
 - Gem::Resolver::RequirementList
 - Gem::Resolver::Set
 - Gem::Resolver::SourceSet
 - Gem::Resolver::SpecSpecification
 - Gem::Resolver::Specification
 - Gem::Resolver::Stats
 - Gem::Resolver::VendorSet
 - Gem::Resolver::VendorSpecification
 - Gem::RubyVersionMismatch
 - Gem::Security
 - Gem::Security::Exception
 - Gem::Security::Policy
 - Gem::Security::Signer
 - Gem::Security::TrustDir
 - Gem::Server
 - Gem::SilentUI
 - Gem::Source
 - Gem::Source::Git
 - Gem::Source::Installed
 - Gem::Source::Local
 - Gem::Source::Lock
 - Gem::Source::SpecificFile
 - Gem::Source::Vendor
 - Gem::SourceFetchProblem
 - Gem::SourceList
 - Gem::SpecFetcher
 - Gem::SpecificGemNotFoundException
 - Gem::Specification
 - Gem::StreamUI
 - Gem::StreamUI::SilentDownloadReporter
 - Gem::StreamUI::SilentProgressReporter
 - Gem::StreamUI::SimpleProgressReporter
 - Gem::StreamUI::VerboseDownloadReporter
 - Gem::StreamUI::VerboseProgressReporter
 - Gem::StubSpecification
 - Gem::SystemExitException
 - Gem::TestCase
 - Gem::TestCase::SpecFetcherSetup
 - Gem::TestCase::StaticSet
 - Gem::Text
 - Gem::Uninstaller
 - Gem::UnsatisfiableDependencyError
 - Gem::UriFormatter
 - Gem::UserInteraction
 - Gem::Util
 - Gem::Validator
 - Gem::VerificationError
 - Gem::Version
 - Gem::VersionOption
 - Kernel
 - Object
 - OpenSSL
 - TempIO
 - YAML::Syck
 - YAML::Syck
 
Gem::Commands::UnpackCommand
Public Class Methods
            new()
            
          
          
           
               # File rubygems/commands/unpack_command.rb, line 11
def initialize
  require 'fileutils'
  super 'unpack', 'Unpack an installed gem to the current directory',
        :version => Gem::Requirement.default,
        :target  => Dir.pwd
  add_option('--target=DIR',
             'target directory for unpacking') do |value, options|
    options[:target] = value
  end
  add_option('--spec', 'unpack the gem specification') do |value, options|
    options[:spec] = true
  end
  add_version_option
end
             
            Public Instance Methods
            description()
            
          
          
           
               # File rubygems/commands/unpack_command.rb, line 38
  def description
    <<-EOF
The unpack command allows you to examine the contents of a gem or modify
them to help diagnose a bug.
You can add the contents of the unpacked gem to the load path using the
RUBYLIB environment variable or -I:
  $ gem unpack my_gem
  Unpacked gem: '.../my_gem-1.0'
  [edit my_gem-1.0/lib/my_gem.rb]
  $ ruby -Imy_gem-1.0/lib -S other_program
You can repackage an unpacked gem using the build command.  See the build
command help for an example.
    EOF
  end
             
            
            execute()
            
          
          
           
               # File rubygems/commands/unpack_command.rb, line 65
def execute
  get_all_gem_names.each do |name|
    dependency = Gem::Dependency.new name, options[:version]
    path = get_path dependency
    unless path then
      alert_error "Gem '#{name}' not installed nor fetchable."
      next
    end
    if @options[:spec] then
      spec, metadata = get_metadata path
      if metadata.nil? then
        alert_error "--spec is unsupported on '#{name}' (old format gem)"
        next
      end
      spec_file = File.basename spec.spec_file
      open spec_file, 'w' do |io|
        io.write metadata
      end
    else
      basename = File.basename path, '.gem'
      target_dir = File.expand_path basename, options[:target]
      package = Gem::Package.new path
      package.extract_files target_dir
      say "Unpacked gem: '#{target_dir}'"
    end
  end
end
             
            
            find_in_cache(filename)
            
          
          
          Find cached filename in Gem.path. Returns nil if the file cannot be found.
 
               # File rubygems/commands/unpack_command.rb, line 107
def find_in_cache(filename)
  Gem.path.each do |path|
    this_path = File.join(path, "cache", filename)
    return this_path if File.exist? this_path
  end
  return nil
end
             
            
            get_metadata(path)
            
          
          
          Extracts the Gem::Specification and raw
metadata from the .gem file at path.
 
               # File rubygems/commands/unpack_command.rb, line 161
def get_metadata path
  format = Gem::Package.new path
  spec = format.spec
  metadata = nil
  open path, Gem.binary_mode do |io|
    tar = Gem::Package::TarReader.new io
    tar.each_entry do |entry|
      case entry.full_name
      when 'metadata' then
        metadata = entry.read
      when 'metadata.gz' then
        metadata = Gem.gunzip entry.read
      end
    end
  end
  return spec, metadata
end
             
            
            get_path(dependency)
            
          
          
          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)
 
               # File rubygems/commands/unpack_command.rb, line 133
def get_path dependency
  return dependency.name if dependency.name =~ /\.gem$/i
  specs = dependency.matching_specs
  selected = specs.max_by { |s| s.version }
  return Gem::RemoteFetcher.fetcher.download_to_cache(dependency) unless
    selected
  return unless dependency.name =~ /^#{selected.name}$/i
  # We expect to find (basename).gem in the 'cache' directory.  Furthermore,
  # the name match must be exact (ignoring case).
  path = find_in_cache File.basename selected.cache_file
  return Gem::RemoteFetcher.fetcher.download_to_cache(dependency) unless path
  path
end