2011-01-28 18:46:47 -05:00
|
|
|
require 'rubygems/installer_test_case'
|
2008-03-31 18:40:06 -04:00
|
|
|
require 'rubygems/uninstaller'
|
|
|
|
|
2011-01-28 18:46:47 -05:00
|
|
|
class TestGemUninstaller < Gem::InstallerTestCase
|
2008-03-31 18:40:06 -04:00
|
|
|
|
|
|
|
def setup
|
|
|
|
super
|
2013-09-14 04:59:02 -04:00
|
|
|
common_installer_setup
|
2008-03-31 18:40:06 -04:00
|
|
|
|
2008-07-01 08:33:11 -04:00
|
|
|
build_rake_in do
|
|
|
|
use_ui ui do
|
|
|
|
@installer.install
|
2012-11-29 01:52:18 -05:00
|
|
|
@spec = @installer.spec
|
2011-05-31 23:45:05 -04:00
|
|
|
|
2012-11-29 01:52:18 -05:00
|
|
|
@user_installer.install
|
|
|
|
@user_spec = @user_installer.spec
|
2008-07-01 08:33:11 -04:00
|
|
|
end
|
2008-03-31 18:40:06 -04:00
|
|
|
end
|
2012-11-29 01:52:18 -05:00
|
|
|
|
|
|
|
Gem::Specification.reset
|
2008-03-31 18:40:06 -04:00
|
|
|
end
|
|
|
|
|
2008-06-17 18:04:18 -04:00
|
|
|
def test_initialize_expand_path
|
|
|
|
uninstaller = Gem::Uninstaller.new nil, :install_dir => '/foo//bar'
|
|
|
|
|
|
|
|
assert_match %r|/foo/bar$|, uninstaller.instance_variable_get(:@gem_home)
|
|
|
|
end
|
|
|
|
|
2012-12-08 01:01:49 -05:00
|
|
|
def test_ask_if_ok
|
2013-11-11 19:16:41 -05:00
|
|
|
c = util_spec 'c'
|
2012-12-08 01:01:49 -05:00
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new nil
|
|
|
|
|
|
|
|
ok = :junk
|
|
|
|
|
|
|
|
ui = Gem::MockGemUi.new "\n"
|
|
|
|
|
|
|
|
use_ui ui do
|
|
|
|
ok = uninstaller.ask_if_ok c
|
|
|
|
end
|
|
|
|
|
|
|
|
refute ok
|
|
|
|
end
|
|
|
|
|
2011-05-31 23:45:05 -04:00
|
|
|
def test_remove_all
|
|
|
|
uninstaller = Gem::Uninstaller.new nil
|
|
|
|
|
|
|
|
ui = Gem::MockGemUi.new "y\n"
|
|
|
|
|
|
|
|
use_ui ui do
|
|
|
|
uninstaller.remove_all [@spec]
|
|
|
|
end
|
|
|
|
|
|
|
|
refute_path_exists @spec.gem_dir
|
|
|
|
end
|
|
|
|
|
2008-03-31 18:40:06 -04:00
|
|
|
def test_remove_executables_force_keep
|
|
|
|
uninstaller = Gem::Uninstaller.new nil, :executables => false
|
|
|
|
|
2012-11-29 01:52:18 -05:00
|
|
|
executable = File.join Gem.bindir(@user_spec.base_dir), 'executable'
|
2011-03-07 03:44:45 -05:00
|
|
|
assert File.exist?(executable), 'executable not written'
|
2011-01-28 18:46:47 -05:00
|
|
|
|
2008-03-31 18:40:06 -04:00
|
|
|
use_ui @ui do
|
2011-01-28 18:46:47 -05:00
|
|
|
uninstaller.remove_executables @user_spec
|
2008-03-31 18:40:06 -04:00
|
|
|
end
|
|
|
|
|
2011-01-28 18:46:47 -05:00
|
|
|
assert File.exist? executable
|
2008-03-31 18:40:06 -04:00
|
|
|
|
|
|
|
assert_equal "Executables and scripts will remain installed.\n", @ui.output
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_remove_executables_force_remove
|
|
|
|
uninstaller = Gem::Uninstaller.new nil, :executables => true
|
|
|
|
|
2012-11-29 01:52:18 -05:00
|
|
|
executable = File.join Gem.bindir(@user_spec.base_dir), 'executable'
|
2011-03-07 03:44:45 -05:00
|
|
|
assert File.exist?(executable), 'executable not written'
|
2011-01-28 18:46:47 -05:00
|
|
|
|
2008-03-31 18:40:06 -04:00
|
|
|
use_ui @ui do
|
2011-01-28 18:46:47 -05:00
|
|
|
uninstaller.remove_executables @user_spec
|
2008-03-31 18:40:06 -04:00
|
|
|
end
|
|
|
|
|
2011-03-07 03:44:45 -05:00
|
|
|
assert_equal "Removing executable\n", @ui.output
|
2008-03-31 18:40:06 -04:00
|
|
|
|
2011-01-28 18:46:47 -05:00
|
|
|
refute File.exist? executable
|
2008-03-31 18:40:06 -04:00
|
|
|
end
|
|
|
|
|
2009-06-09 17:38:59 -04:00
|
|
|
def test_remove_executables_user
|
|
|
|
uninstaller = Gem::Uninstaller.new nil, :executables => true
|
|
|
|
|
|
|
|
use_ui @ui do
|
|
|
|
uninstaller.remove_executables @user_spec
|
|
|
|
end
|
|
|
|
|
2011-03-07 03:44:45 -05:00
|
|
|
exec_path = File.join Gem.user_dir, 'bin', 'executable'
|
2012-11-29 01:52:18 -05:00
|
|
|
refute File.exist?(exec_path), 'exec still exists in user bin dir'
|
2009-06-09 17:38:59 -04:00
|
|
|
|
2011-03-07 03:44:45 -05:00
|
|
|
assert_equal "Removing executable\n", @ui.output
|
2009-06-09 17:38:59 -04:00
|
|
|
end
|
|
|
|
|
2011-03-07 03:44:45 -05:00
|
|
|
def test_remove_executables_user_format
|
|
|
|
Gem::Installer.exec_format = 'foo-%s-bar'
|
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new nil, :executables => true, :format_executable => true
|
|
|
|
|
|
|
|
use_ui @ui do
|
|
|
|
uninstaller.remove_executables @user_spec
|
|
|
|
end
|
|
|
|
|
|
|
|
exec_path = File.join Gem.user_dir, 'bin', 'foo-executable-bar'
|
|
|
|
assert_equal false, File.exist?(exec_path), 'removed exec from bin dir'
|
|
|
|
|
2011-07-26 23:33:45 -04:00
|
|
|
assert_equal "Removing foo-executable-bar\n", @ui.output
|
2011-03-07 03:44:45 -05:00
|
|
|
ensure
|
|
|
|
Gem::Installer.exec_format = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_remove_executables_user_format_disabled
|
|
|
|
Gem::Installer.exec_format = 'foo-%s-bar'
|
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new nil, :executables => true
|
|
|
|
|
|
|
|
use_ui @ui do
|
|
|
|
uninstaller.remove_executables @user_spec
|
|
|
|
end
|
|
|
|
|
|
|
|
exec_path = File.join Gem.user_dir, 'bin', 'executable'
|
2011-05-31 23:45:05 -04:00
|
|
|
refute File.exist?(exec_path), 'removed exec from bin dir'
|
2011-03-07 03:44:45 -05:00
|
|
|
|
|
|
|
assert_equal "Removing executable\n", @ui.output
|
|
|
|
ensure
|
|
|
|
Gem::Installer.exec_format = nil
|
|
|
|
end
|
|
|
|
|
2014-01-06 20:19:28 -05:00
|
|
|
def test_remove_not_in_home
|
|
|
|
uninstaller = Gem::Uninstaller.new nil, :install_dir => "#{@gemhome}2"
|
|
|
|
|
|
|
|
e = assert_raises Gem::GemNotInHomeException do
|
|
|
|
use_ui ui do
|
|
|
|
uninstaller.remove @spec
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
expected =
|
|
|
|
"Gem '#{@spec.full_name}' is not installed in directory #{@gemhome}2"
|
|
|
|
|
|
|
|
assert_equal expected, e.message
|
|
|
|
|
|
|
|
assert_path_exists @spec.gem_dir
|
|
|
|
end
|
|
|
|
|
2008-06-17 18:04:18 -04:00
|
|
|
def test_path_ok_eh
|
|
|
|
uninstaller = Gem::Uninstaller.new nil
|
|
|
|
|
2009-06-09 17:38:59 -04:00
|
|
|
assert_equal true, uninstaller.path_ok?(@gemhome, @spec)
|
2008-06-17 18:04:18 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_path_ok_eh_legacy
|
|
|
|
uninstaller = Gem::Uninstaller.new nil
|
|
|
|
|
2011-05-31 23:45:05 -04:00
|
|
|
@spec.loaded_from = @spec.loaded_from.gsub @spec.full_name, '\&-legacy'
|
2008-06-17 18:04:18 -04:00
|
|
|
@spec.platform = 'legacy'
|
|
|
|
|
2009-06-09 17:38:59 -04:00
|
|
|
assert_equal true, uninstaller.path_ok?(@gemhome, @spec)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_path_ok_eh_user
|
|
|
|
uninstaller = Gem::Uninstaller.new nil
|
|
|
|
|
|
|
|
assert_equal true, uninstaller.path_ok?(Gem.user_dir, @user_spec)
|
2008-06-17 18:04:18 -04:00
|
|
|
end
|
|
|
|
|
2008-09-25 06:13:50 -04:00
|
|
|
def test_uninstall
|
|
|
|
uninstaller = Gem::Uninstaller.new @spec.name, :executables => true
|
|
|
|
|
|
|
|
gem_dir = File.join @gemhome, 'gems', @spec.full_name
|
|
|
|
|
|
|
|
Gem.pre_uninstall do
|
|
|
|
assert File.exist?(gem_dir), 'gem_dir should exist'
|
|
|
|
end
|
|
|
|
|
|
|
|
Gem.post_uninstall do
|
2009-06-09 17:38:59 -04:00
|
|
|
refute File.exist?(gem_dir), 'gem_dir should not exist'
|
|
|
|
end
|
|
|
|
|
|
|
|
uninstaller.uninstall
|
|
|
|
|
|
|
|
refute File.exist?(gem_dir)
|
|
|
|
|
|
|
|
assert_same uninstaller, @pre_uninstall_hook_arg
|
|
|
|
assert_same uninstaller, @post_uninstall_hook_arg
|
|
|
|
end
|
|
|
|
|
2012-12-19 02:19:10 -05:00
|
|
|
def test_uninstall_default_gem
|
|
|
|
spec = new_default_spec 'default', '2'
|
|
|
|
|
|
|
|
install_default_gems spec
|
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new spec.name, :executables => true
|
|
|
|
|
|
|
|
e = assert_raises Gem::InstallError do
|
|
|
|
uninstaller.uninstall
|
|
|
|
end
|
|
|
|
|
|
|
|
assert_equal 'gem "default" cannot be uninstalled ' +
|
|
|
|
'because it is a default gem',
|
|
|
|
e.message
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_uninstall_default_gem_with_same_version
|
|
|
|
default_spec = new_default_spec 'default', '2'
|
|
|
|
install_default_gems default_spec
|
|
|
|
|
|
|
|
spec = new_spec 'default', '2'
|
|
|
|
install_gem spec
|
|
|
|
|
|
|
|
Gem::Specification.reset
|
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new spec.name, :executables => true
|
|
|
|
|
|
|
|
uninstaller.uninstall
|
|
|
|
|
|
|
|
refute_path_exists spec.gem_dir
|
|
|
|
end
|
|
|
|
|
2013-10-15 20:14:16 -04:00
|
|
|
def test_uninstall_extension
|
|
|
|
@spec.extensions << 'extconf.rb'
|
|
|
|
write_file File.join(@tempdir, 'extconf.rb') do |io|
|
|
|
|
io.write <<-RUBY
|
|
|
|
require 'mkmf'
|
|
|
|
create_makefile '#{@spec.name}'
|
|
|
|
RUBY
|
|
|
|
end
|
|
|
|
|
|
|
|
@spec.files += %w[extconf.rb]
|
|
|
|
|
|
|
|
use_ui @ui do
|
|
|
|
path = Gem::Package.build @spec
|
|
|
|
|
|
|
|
installer = Gem::Installer.new path
|
|
|
|
installer.install
|
|
|
|
end
|
|
|
|
|
2013-12-10 14:54:19 -05:00
|
|
|
assert_path_exists @spec.extension_dir, 'sanity check'
|
2013-10-15 20:14:16 -04:00
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new @spec.name, :executables => true
|
|
|
|
uninstaller.uninstall
|
|
|
|
|
2013-12-10 14:54:19 -05:00
|
|
|
refute_path_exists @spec.extension_dir
|
2013-10-15 20:14:16 -04:00
|
|
|
end
|
|
|
|
|
2011-07-26 23:33:45 -04:00
|
|
|
def test_uninstall_nonexistent
|
|
|
|
uninstaller = Gem::Uninstaller.new 'bogus', :executables => true
|
|
|
|
|
|
|
|
e = assert_raises Gem::InstallError do
|
|
|
|
uninstaller.uninstall
|
|
|
|
end
|
|
|
|
|
|
|
|
assert_equal 'gem "bogus" is not installed', e.message
|
|
|
|
end
|
|
|
|
|
2011-03-07 03:44:45 -05:00
|
|
|
def test_uninstall_not_ok
|
|
|
|
quick_gem 'z' do |s|
|
|
|
|
s.add_runtime_dependency @spec.name
|
|
|
|
end
|
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new @spec.name
|
|
|
|
|
|
|
|
gem_dir = File.join @gemhome, 'gems', @spec.full_name
|
|
|
|
executable = File.join @gemhome, 'bin', 'executable'
|
|
|
|
|
|
|
|
assert File.exist?(gem_dir), 'gem_dir must exist'
|
|
|
|
assert File.exist?(executable), 'executable must exist'
|
|
|
|
|
|
|
|
ui = Gem::MockGemUi.new "n\n"
|
|
|
|
|
|
|
|
assert_raises Gem::DependencyRemovalException do
|
|
|
|
use_ui ui do
|
|
|
|
uninstaller.uninstall
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
assert File.exist?(gem_dir), 'gem_dir must still exist'
|
|
|
|
assert File.exist?(executable), 'executable must still exist'
|
|
|
|
end
|
|
|
|
|
2013-09-18 17:29:41 -04:00
|
|
|
def test_uninstall_user_install
|
2011-05-31 23:45:05 -04:00
|
|
|
@user_spec = Gem::Specification.find_by_name 'b'
|
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new(@user_spec.name,
|
|
|
|
:executables => true,
|
|
|
|
:user_install => true)
|
2009-06-09 17:38:59 -04:00
|
|
|
|
2012-11-29 01:52:18 -05:00
|
|
|
gem_dir = File.join @user_spec.gem_dir
|
2009-06-09 17:38:59 -04:00
|
|
|
|
|
|
|
Gem.pre_uninstall do
|
2011-05-31 23:45:05 -04:00
|
|
|
assert_path_exists gem_dir
|
2009-06-09 17:38:59 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
Gem.post_uninstall do
|
2011-05-31 23:45:05 -04:00
|
|
|
refute_path_exists gem_dir
|
2008-09-25 06:13:50 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
uninstaller.uninstall
|
|
|
|
|
2011-05-31 23:45:05 -04:00
|
|
|
refute_path_exists gem_dir
|
2008-09-25 06:13:50 -04:00
|
|
|
|
|
|
|
assert_same uninstaller, @pre_uninstall_hook_arg
|
|
|
|
assert_same uninstaller, @post_uninstall_hook_arg
|
|
|
|
end
|
|
|
|
|
2012-11-29 01:52:18 -05:00
|
|
|
def test_uninstall_wrong_repo
|
|
|
|
Gem.use_paths "#{@gemhome}2", [@gemhome]
|
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new @spec.name, :executables => true
|
|
|
|
|
|
|
|
e = assert_raises Gem::InstallError do
|
|
|
|
uninstaller.uninstall
|
|
|
|
end
|
|
|
|
|
|
|
|
expected = <<-MESSAGE.strip
|
|
|
|
#{@spec.name} is not installed in GEM_HOME, try:
|
|
|
|
\tgem uninstall -i #{@gemhome} a
|
|
|
|
MESSAGE
|
|
|
|
|
|
|
|
assert_equal expected, e.message
|
|
|
|
end
|
|
|
|
|
2011-05-31 23:45:05 -04:00
|
|
|
def test_uninstall_selection_greater_than_one
|
|
|
|
util_make_gems
|
2012-08-21 00:50:18 -04:00
|
|
|
|
2011-05-31 23:45:05 -04:00
|
|
|
list = Gem::Specification.find_all_by_name('a')
|
|
|
|
|
|
|
|
uninstaller = Gem::Uninstaller.new('a')
|
|
|
|
|
2012-11-29 01:52:18 -05:00
|
|
|
use_ui Gem::MockGemUi.new("2\ny\n") do
|
2011-05-31 23:45:05 -04:00
|
|
|
uninstaller.uninstall
|
|
|
|
end
|
2008-03-31 18:40:06 -04:00
|
|
|
|
2011-05-31 23:45:05 -04:00
|
|
|
updated_list = Gem::Specification.find_all_by_name('a')
|
|
|
|
assert_equal list.length - 1, updated_list.length
|
|
|
|
end
|
2012-11-29 01:52:18 -05:00
|
|
|
|
|
|
|
def test_uninstall_prompts_about_broken_deps
|
2012-12-19 02:19:10 -05:00
|
|
|
quick_gem 'r', '1' do |s| s.add_dependency 'q', '= 1' end
|
|
|
|
quick_gem 'q', '1'
|
2012-11-29 01:52:18 -05:00
|
|
|
|
|
|
|
un = Gem::Uninstaller.new('q')
|
|
|
|
ui = Gem::MockGemUi.new("y\n")
|
|
|
|
|
|
|
|
use_ui ui do
|
|
|
|
un.uninstall
|
|
|
|
end
|
|
|
|
|
|
|
|
lines = ui.output.split("\n")
|
|
|
|
lines.shift
|
|
|
|
|
|
|
|
assert_match %r!You have requested to uninstall the gem:!, lines.shift
|
|
|
|
lines.shift
|
|
|
|
lines.shift
|
|
|
|
|
|
|
|
assert_match %r!r-1 depends on q \(= 1\)!, lines.shift
|
|
|
|
assert_match %r!Successfully uninstalled q-1!, lines.last
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_uninstall_only_lists_unsatified_deps
|
2012-12-19 02:19:10 -05:00
|
|
|
quick_gem 'r', '1' do |s| s.add_dependency 'q', '~> 1.0' end
|
|
|
|
quick_gem 'x', '1' do |s| s.add_dependency 'q', '= 1.0' end
|
|
|
|
quick_gem 'q', '1.0'
|
|
|
|
quick_gem 'q', '1.1'
|
2012-11-29 01:52:18 -05:00
|
|
|
|
|
|
|
un = Gem::Uninstaller.new('q', :version => "1.0")
|
|
|
|
ui = Gem::MockGemUi.new("y\n")
|
|
|
|
|
|
|
|
use_ui ui do
|
|
|
|
un.uninstall
|
|
|
|
end
|
|
|
|
|
|
|
|
lines = ui.output.split("\n")
|
|
|
|
lines.shift
|
|
|
|
|
|
|
|
assert_match %r!You have requested to uninstall the gem:!, lines.shift
|
|
|
|
lines.shift
|
|
|
|
lines.shift
|
|
|
|
|
|
|
|
assert_match %r!x-1 depends on q \(= 1.0\)!, lines.shift
|
|
|
|
assert_match %r!Successfully uninstalled q-1.0!, lines.last
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_uninstall_doesnt_prompt_when_other_gem_satifies_requirement
|
2012-12-19 02:19:10 -05:00
|
|
|
quick_gem 'r', '1' do |s| s.add_dependency 'q', '~> 1.0' end
|
|
|
|
quick_gem 'q', '1.0'
|
|
|
|
quick_gem 'q', '1.1'
|
2012-11-29 01:52:18 -05:00
|
|
|
|
|
|
|
un = Gem::Uninstaller.new('q', :version => "1.0")
|
|
|
|
ui = Gem::MockGemUi.new("y\n")
|
|
|
|
|
|
|
|
use_ui ui do
|
|
|
|
un.uninstall
|
|
|
|
end
|
|
|
|
|
|
|
|
lines = ui.output.split("\n")
|
|
|
|
|
|
|
|
assert_equal "Successfully uninstalled q-1.0", lines.shift
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_uninstall_doesnt_prompt_when_removing_a_dev_dep
|
2012-12-19 02:19:10 -05:00
|
|
|
quick_gem 'r', '1' do |s| s.add_development_dependency 'q', '= 1.0' end
|
|
|
|
quick_gem 'q', '1.0'
|
2012-11-29 01:52:18 -05:00
|
|
|
|
|
|
|
un = Gem::Uninstaller.new('q', :version => "1.0")
|
|
|
|
ui = Gem::MockGemUi.new("y\n")
|
|
|
|
|
|
|
|
use_ui ui do
|
|
|
|
un.uninstall
|
|
|
|
end
|
|
|
|
|
|
|
|
lines = ui.output.split("\n")
|
|
|
|
|
|
|
|
assert_equal "Successfully uninstalled q-1.0", lines.shift
|
|
|
|
end
|
|
|
|
|
2013-09-14 04:59:02 -04:00
|
|
|
def test_uninstall_doesnt_prompt_and_raises_when_abort_on_dependent_set
|
|
|
|
quick_gem 'r', '1' do |s| s.add_dependency 'q', '= 1' end
|
|
|
|
quick_gem 'q', '1'
|
|
|
|
|
|
|
|
un = Gem::Uninstaller.new('q', :abort_on_dependent => true)
|
|
|
|
ui = Gem::MockGemUi.new("y\n")
|
|
|
|
|
|
|
|
assert_raises Gem::DependencyRemovalException do
|
|
|
|
use_ui ui do
|
|
|
|
un.uninstall
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2012-11-29 01:52:18 -05:00
|
|
|
|
|
|
|
def test_uninstall_prompt_includes_dep_type
|
2012-12-19 02:19:10 -05:00
|
|
|
quick_gem 'r', '1' do |s|
|
2012-11-29 01:52:18 -05:00
|
|
|
s.add_development_dependency 'q', '= 1'
|
|
|
|
end
|
|
|
|
|
2012-12-19 02:19:10 -05:00
|
|
|
quick_gem 'q', '1'
|
2012-11-29 01:52:18 -05:00
|
|
|
|
|
|
|
un = Gem::Uninstaller.new('q', :check_dev => true)
|
|
|
|
ui = Gem::MockGemUi.new("y\n")
|
|
|
|
|
|
|
|
use_ui ui do
|
|
|
|
un.uninstall
|
|
|
|
end
|
|
|
|
|
|
|
|
lines = ui.output.split("\n")
|
|
|
|
lines.shift
|
|
|
|
|
|
|
|
assert_match %r!You have requested to uninstall the gem:!, lines.shift
|
|
|
|
lines.shift
|
|
|
|
lines.shift
|
|
|
|
|
|
|
|
assert_match %r!r-1 depends on q \(= 1, development\)!, lines.shift
|
|
|
|
assert_match %r!Successfully uninstalled q-1!, lines.last
|
|
|
|
end
|
2011-05-31 23:45:05 -04:00
|
|
|
end
|