brew/Library/Homebrew/cmd/uninstall.rb

218 lines
5.8 KiB
Ruby
Raw Normal View History

2020-10-10 14:16:11 +02:00
# typed: false
# frozen_string_literal: true
require "keg"
require "formula"
require "diagnostic"
require "migrator"
2019-04-17 18:25:08 +09:00
require "cli/parser"
require "cask/cmd"
require "cask/cask_loader"
module Homebrew
2016-09-26 01:44:51 +02:00
module_function
def uninstall_args
Homebrew::CLI::Parser.new do
usage_banner <<~EOS
`uninstall`, `rm`, `remove` [<options>] <formula>
Uninstall <formula>.
EOS
switch "-f", "--force",
2019-08-06 14:22:24 -04:00
description: "Delete all installed versions of <formula>."
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
min_named :formula
end
end
def uninstall
args = uninstall_args.parse
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|
rack = Formulary.to_rack(name)
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
end
2020-07-02 15:28:41 -04:00
end
else
all_kegs, casks = args.named.to_kegs_to_casks
2020-06-30 12:21:21 -04:00
kegs_by_rack = all_kegs.group_by(&:rack)
end
handle_unsatisfied_dependents(kegs_by_rack,
ignore_dependencies: args.ignore_dependencies?,
named_args: args.named)
return if Homebrew.failed?
kegs_by_rack.each do |rack, kegs|
if args.force?
name = rack.basename
if rack.directory?
puts "Uninstalling #{name}... (#{rack.abv})"
kegs.each do |keg|
2014-06-23 22:00:33 -05:00
keg.unlink
keg.uninstall
end
end
rm_pin rack
else
kegs.each do |keg|
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
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)
puts "#{keg.name} #{versions.to_sentence} #{"is".pluralize(versions.count)} still installed."
puts "Run `brew uninstall --force #{keg.name}` to remove all versions."
end
next unless f
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?
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
end
end
end
end
2020-07-03 10:33:38 -04:00
return if casks.blank?
2020-06-30 13:43:45 -04:00
Cask::Cmd::Uninstall.uninstall_casks(
*casks,
2020-08-01 02:30:46 +02:00
binaries: EnvConfig.cask_opts_binaries?,
verbose: args.verbose?,
force: args.force?,
)
rescue MultipleVersionsInstalledError => e
ofail e
ensure
# If we delete Cellar/newname, then Cellar/oldname symlink
# can become broken and we have to remove it.
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
end
end
def handle_unsatisfied_dependents(kegs_by_rack, ignore_dependencies: false, named_args: [])
return if ignore_dependencies
all_kegs = kegs_by_rack.values.flatten(1)
check_for_dependents(all_kegs, named_args: named_args)
rescue MethodDeprecatedError
# Silently ignore deprecations when uninstalling.
nil
end
def check_for_dependents(kegs, named_args: [])
return false unless result = Keg.find_some_installed_dependents(kegs)
2020-04-05 15:44:50 +01:00
if Homebrew::EnvConfig.developer?
DeveloperDependentsMessage.new(*result, named_args: named_args).output
else
NondeveloperDependentsMessage.new(*result, named_args: named_args).output
end
true
end
2016-11-14 13:09:40 +00:00
class DependentsMessage
attr_reader :reqs, :deps, :named_args
2016-11-14 13:09:40 +00:00
def initialize(requireds, dependents, named_args: [])
@reqs = requireds
@deps = dependents
@named_args = named_args
2016-11-14 13:09:40 +00:00
end
protected
def sample_command
"brew uninstall --ignore-dependencies #{named_args.join(" ")}"
2016-11-14 13:09:40 +00:00
end
def are_required_by_deps
"#{"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
end
2016-11-14 13:09:40 +00:00
class DeveloperDependentsMessage < DependentsMessage
def output
2017-10-15 02:28:32 +02:00
opoo <<~EOS
#{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
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
end
def rm_pin(rack)
2016-09-10 10:24:57 +01:00
Formulary.from_rack(rack).unpin
rescue
nil
end
end