2019-04-19 15:38:03 +09:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2015-08-03 13:09:07 +01:00
|
|
|
require "keg"
|
|
|
|
require "formula"
|
2016-09-30 19:34:14 +01:00
|
|
|
require "diagnostic"
|
2015-08-09 14:57:15 +03:00
|
|
|
require "migrator"
|
2019-04-17 18:25:08 +09:00
|
|
|
require "cli/parser"
|
2020-06-24 11:59:55 -04:00
|
|
|
require "cask/cmd"
|
|
|
|
require "cask/cask_loader"
|
2010-09-11 20:22:54 +01:00
|
|
|
|
2014-06-18 22:41:47 -05:00
|
|
|
module Homebrew
|
2016-09-26 01:44:51 +02:00
|
|
|
module_function
|
|
|
|
|
2018-11-11 17:43:31 +05:30
|
|
|
def uninstall_args
|
|
|
|
Homebrew::CLI::Parser.new do
|
|
|
|
usage_banner <<~EOS
|
|
|
|
`uninstall`, `rm`, `remove` [<options>] <formula>
|
|
|
|
|
|
|
|
Uninstall <formula>.
|
|
|
|
EOS
|
2020-07-27 03:59:52 +02:00
|
|
|
switch "-f", "--force",
|
2019-08-06 14:22:24 -04:00
|
|
|
description: "Delete all installed versions of <formula>."
|
2018-11-11 17:43:31 +05:30
|
|
|
switch "--ignore-dependencies",
|
2019-04-30 08:44:35 +01:00
|
|
|
description: "Don't fail uninstall, even if <formula> is a dependency of any installed "\
|
|
|
|
"formulae."
|
2020-07-30 18:40:10 +02:00
|
|
|
|
2020-03-04 17:28:15 +00:00
|
|
|
min_named :formula
|
2018-11-11 17:43:31 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-09-11 20:22:54 +01:00
|
|
|
def uninstall
|
2020-07-26 09:45:26 +02:00
|
|
|
args = uninstall_args.parse
|
2018-11-11 17:43:31 +05:30
|
|
|
|
2020-06-24 11:59:55 -04:00
|
|
|
if args.force?
|
2020-06-30 12:08:03 -04:00
|
|
|
casks = []
|
2020-07-02 15:29:58 -04:00
|
|
|
kegs_by_rack = {}
|
2020-07-02 15:28:41 -04:00
|
|
|
|
|
|
|
args.named.each do |name|
|
2016-09-28 19:14:33 +01:00
|
|
|
rack = Formulary.to_rack(name)
|
2020-06-24 11:59:55 -04:00
|
|
|
|
2020-07-02 15:28:41 -04:00
|
|
|
if rack.directory?
|
|
|
|
kegs_by_rack[rack] = rack.subdirs.map { |d| Keg.new(d) }
|
|
|
|
else
|
2020-06-30 12:08:03 -04:00
|
|
|
begin
|
|
|
|
casks << Cask::CaskLoader.load(name)
|
|
|
|
rescue Cask::CaskUnavailableError
|
|
|
|
# Since the uninstall was forced, ignore any unavailable casks
|
|
|
|
end
|
2020-06-24 11:59:55 -04:00
|
|
|
end
|
2020-07-02 15:28:41 -04:00
|
|
|
end
|
2010-09-11 20:22:54 +01:00
|
|
|
else
|
2020-06-30 12:21:21 -04:00
|
|
|
all_kegs, casks = args.kegs_casks
|
|
|
|
kegs_by_rack = all_kegs.group_by(&:rack)
|
2016-09-28 19:14:33 +01:00
|
|
|
end
|
|
|
|
|
2020-07-26 09:45:26 +02:00
|
|
|
handle_unsatisfied_dependents(kegs_by_rack,
|
|
|
|
ignore_dependencies: args.ignore_dependencies?,
|
|
|
|
named_args: args.named)
|
2016-11-11 20:08:26 +00:00
|
|
|
return if Homebrew.failed?
|
2016-09-28 19:21:47 +01:00
|
|
|
|
2016-09-28 19:14:33 +01:00
|
|
|
kegs_by_rack.each do |rack, kegs|
|
2018-11-11 17:43:31 +05:30
|
|
|
if args.force?
|
2015-08-09 14:57:15 +03:00
|
|
|
name = rack.basename
|
|
|
|
|
2012-03-06 13:43:41 +00:00
|
|
|
if rack.directory?
|
2015-04-09 15:18:25 +08:00
|
|
|
puts "Uninstalling #{name}... (#{rack.abv})"
|
2016-09-28 19:14:33 +01:00
|
|
|
kegs.each do |keg|
|
2014-06-23 22:00:33 -05:00
|
|
|
keg.unlink
|
|
|
|
keg.uninstall
|
|
|
|
end
|
2010-09-11 20:22:54 +01:00
|
|
|
end
|
2012-08-10 16:33:22 -04:00
|
|
|
|
2015-05-17 21:22:29 +08:00
|
|
|
rm_pin rack
|
2016-09-28 19:14:33 +01:00
|
|
|
else
|
|
|
|
kegs.each do |keg|
|
2018-02-05 09:31:17 +00:00
|
|
|
begin
|
|
|
|
f = Formulary.from_rack(rack)
|
|
|
|
if f.pinned?
|
|
|
|
onoe "#{f.full_name} is pinned. You must unpin it to uninstall."
|
|
|
|
next
|
|
|
|
end
|
|
|
|
rescue
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
|
2016-09-28 19:14:33 +01:00
|
|
|
keg.lock do
|
|
|
|
puts "Uninstalling #{keg}... (#{keg.abv})"
|
|
|
|
keg.unlink
|
|
|
|
keg.uninstall
|
|
|
|
rack = keg.rack
|
|
|
|
rm_pin rack
|
|
|
|
|
|
|
|
if rack.directory?
|
|
|
|
versions = rack.subdirs.map(&:basename)
|
2018-09-17 20:11:11 +02:00
|
|
|
puts "#{keg.name} #{versions.to_sentence} #{"is".pluralize(versions.count)} still installed."
|
2019-12-13 15:39:55 -05:00
|
|
|
puts "Run `brew uninstall --force #{keg.name}` to remove all versions."
|
2016-09-28 19:14:33 +01:00
|
|
|
end
|
2020-05-08 10:37:51 +01:00
|
|
|
|
2020-05-13 17:35:59 -07:00
|
|
|
next unless f
|
|
|
|
|
2020-05-08 10:37:51 +01:00
|
|
|
paths = f.pkgetc.find.map(&:to_s) if f.pkgetc.exist?
|
|
|
|
if paths.present?
|
|
|
|
puts
|
|
|
|
opoo <<~EOS
|
|
|
|
The following #{f.name} configuration files have not been removed!
|
|
|
|
If desired, remove them manually with `rm -rf`:
|
|
|
|
#{paths.sort.uniq.join("\n ")}
|
|
|
|
EOS
|
|
|
|
end
|
|
|
|
|
|
|
|
unversioned_name = f.name.gsub(/@.+$/, "")
|
2020-05-10 19:22:54 +01:00
|
|
|
maybe_paths = Dir.glob("#{f.etc}/*#{unversioned_name}*")
|
|
|
|
maybe_paths -= paths if paths.present?
|
2020-05-08 10:37:51 +01:00
|
|
|
if maybe_paths.present?
|
|
|
|
puts
|
|
|
|
opoo <<~EOS
|
|
|
|
The following may be #{f.name} configuration files and have not been removed!
|
|
|
|
If desired, remove them manually with `rm -rf`:
|
|
|
|
#{maybe_paths.sort.uniq.join("\n ")}
|
|
|
|
EOS
|
|
|
|
end
|
2016-09-28 19:14:33 +01:00
|
|
|
end
|
|
|
|
end
|
2010-09-11 20:22:54 +01:00
|
|
|
end
|
|
|
|
end
|
2020-06-24 11:59:55 -04:00
|
|
|
|
2020-07-03 10:33:38 -04:00
|
|
|
return if casks.blank?
|
2020-06-30 13:43:45 -04:00
|
|
|
|
2020-08-05 10:28:58 -04:00
|
|
|
Cask::Cmd::Uninstall.uninstall_casks(
|
|
|
|
*casks,
|
2020-08-01 02:30:46 +02:00
|
|
|
binaries: EnvConfig.cask_opts_binaries?,
|
2020-08-05 10:28:58 -04:00
|
|
|
verbose: args.verbose?,
|
|
|
|
force: args.force?,
|
|
|
|
)
|
2010-09-11 20:22:54 +01:00
|
|
|
rescue MultipleVersionsInstalledError => e
|
2012-04-30 14:08:59 +10:00
|
|
|
ofail e
|
2019-12-13 15:39:55 -05:00
|
|
|
puts "Run `brew uninstall --force #{e.name}` to remove all versions."
|
2015-08-17 22:56:00 +08:00
|
|
|
ensure
|
|
|
|
# If we delete Cellar/newname, then Cellar/oldname symlink
|
|
|
|
# can become broken and we have to remove it.
|
2015-10-17 03:57:40 +08:00
|
|
|
if HOMEBREW_CELLAR.directory?
|
2015-10-17 03:59:28 +08:00
|
|
|
HOMEBREW_CELLAR.children.each do |rack|
|
|
|
|
rack.unlink if rack.symlink? && !rack.resolved_path_exists?
|
|
|
|
end
|
2015-10-17 03:57:40 +08:00
|
|
|
end
|
2010-09-11 20:22:54 +01:00
|
|
|
end
|
2012-08-10 16:33:22 -04:00
|
|
|
|
2020-07-26 09:45:26 +02:00
|
|
|
def handle_unsatisfied_dependents(kegs_by_rack, ignore_dependencies: false, named_args: [])
|
|
|
|
return if ignore_dependencies
|
2016-11-11 20:08:26 +00:00
|
|
|
|
|
|
|
all_kegs = kegs_by_rack.values.flatten(1)
|
2020-07-26 09:45:26 +02:00
|
|
|
check_for_dependents(all_kegs, named_args: named_args)
|
2016-11-18 09:34:58 +00:00
|
|
|
rescue MethodDeprecatedError
|
|
|
|
# Silently ignore deprecations when uninstalling.
|
|
|
|
nil
|
2016-10-25 23:53:10 +01:00
|
|
|
end
|
|
|
|
|
2020-07-26 09:45:26 +02:00
|
|
|
def check_for_dependents(kegs, named_args: [])
|
2016-10-05 22:22:32 +01:00
|
|
|
return false unless result = Keg.find_some_installed_dependents(kegs)
|
2016-09-30 19:34:14 +01:00
|
|
|
|
2020-04-05 15:44:50 +01:00
|
|
|
if Homebrew::EnvConfig.developer?
|
2020-07-26 09:45:26 +02:00
|
|
|
DeveloperDependentsMessage.new(*result, named_args: named_args).output
|
2016-11-11 20:08:26 +00:00
|
|
|
else
|
2020-07-26 09:45:26 +02:00
|
|
|
NondeveloperDependentsMessage.new(*result, named_args: named_args).output
|
2016-11-11 20:08:26 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
true
|
|
|
|
end
|
2016-09-30 19:34:14 +01:00
|
|
|
|
2016-11-14 13:09:40 +00:00
|
|
|
class DependentsMessage
|
2020-07-26 09:45:26 +02:00
|
|
|
attr_reader :reqs, :deps, :named_args
|
2016-11-14 13:09:40 +00:00
|
|
|
|
2020-07-26 09:45:26 +02:00
|
|
|
def initialize(requireds, dependents, named_args: [])
|
2016-11-15 21:56:42 +00:00
|
|
|
@reqs = requireds
|
|
|
|
@deps = dependents
|
2020-07-26 09:45:26 +02:00
|
|
|
@named_args = named_args
|
2016-11-14 13:09:40 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
def sample_command
|
2020-07-26 09:45:26 +02:00
|
|
|
"brew uninstall --ignore-dependencies #{named_args.join(" ")}"
|
2016-11-14 13:09:40 +00:00
|
|
|
end
|
|
|
|
|
2016-11-14 13:39:17 +00:00
|
|
|
def are_required_by_deps
|
2018-09-17 20:11:11 +02:00
|
|
|
"#{"is".pluralize(reqs.count)} required by #{deps.to_sentence}, " \
|
|
|
|
"which #{"is".pluralize(deps.count)} currently installed"
|
2016-11-14 13:09:40 +00:00
|
|
|
end
|
2016-11-11 20:08:26 +00:00
|
|
|
end
|
|
|
|
|
2016-11-14 13:09:40 +00:00
|
|
|
class DeveloperDependentsMessage < DependentsMessage
|
|
|
|
def output
|
2017-10-15 02:28:32 +02:00
|
|
|
opoo <<~EOS
|
2018-09-17 20:11:11 +02:00
|
|
|
#{reqs.to_sentence} #{are_required_by_deps}.
|
2016-11-14 13:09:40 +00:00
|
|
|
You can silence this warning with:
|
|
|
|
#{sample_command}
|
|
|
|
EOS
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class NondeveloperDependentsMessage < DependentsMessage
|
|
|
|
def output
|
2017-10-15 02:28:32 +02:00
|
|
|
ofail <<~EOS
|
2018-09-17 20:11:11 +02:00
|
|
|
Refusing to uninstall #{reqs.to_sentence}
|
|
|
|
because #{"it".pluralize(reqs.count)} #{are_required_by_deps}.
|
2016-11-14 13:09:40 +00:00
|
|
|
You can override this and force removal with:
|
|
|
|
#{sample_command}
|
|
|
|
EOS
|
|
|
|
end
|
2016-09-30 19:34:14 +01:00
|
|
|
end
|
|
|
|
|
2015-08-03 13:09:07 +01:00
|
|
|
def rm_pin(rack)
|
2016-09-10 10:24:57 +01:00
|
|
|
Formulary.from_rack(rack).unpin
|
|
|
|
rescue
|
|
|
|
nil
|
2013-03-31 21:28:20 -05:00
|
|
|
end
|
2010-09-11 20:22:54 +01:00
|
|
|
end
|