mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00

Fixes #2355 Create unremovable_kegs instance var Check cellar cleanup failure after full cleanup completes Use module_function in Homebrew::Cleanup as we never instantiate the class
170 lines
4.3 KiB
Ruby
170 lines
4.3 KiB
Ruby
require "utils/bottles"
|
|
require "formula"
|
|
require "thread"
|
|
|
|
module Homebrew
|
|
module Cleanup
|
|
@disk_cleanup_size = 0
|
|
|
|
module_function
|
|
|
|
def cleanup
|
|
cleanup_cellar
|
|
cleanup_cache
|
|
cleanup_logs
|
|
return if ARGV.dry_run?
|
|
cleanup_lockfiles
|
|
rm_ds_store
|
|
end
|
|
|
|
def update_disk_cleanup_size(path_size)
|
|
@disk_cleanup_size += path_size
|
|
end
|
|
|
|
def disk_cleanup_size
|
|
@disk_cleanup_size
|
|
end
|
|
|
|
def unremovable_kegs
|
|
@unremovable_kegs ||= []
|
|
end
|
|
|
|
def cleanup_cellar(formulae = Formula.installed)
|
|
formulae.each(&method(:cleanup_formula))
|
|
end
|
|
|
|
def cleanup_formula(formula)
|
|
formula.eligible_kegs_for_cleanup.each(&method(:cleanup_keg))
|
|
end
|
|
|
|
def cleanup_keg(keg)
|
|
cleanup_path(keg) { keg.uninstall }
|
|
rescue Errno::EACCES => e
|
|
opoo e.message
|
|
unremovable_kegs << keg
|
|
end
|
|
|
|
def cleanup_logs
|
|
return unless HOMEBREW_LOGS.directory?
|
|
HOMEBREW_LOGS.subdirs.each do |dir|
|
|
cleanup_path(dir) { dir.rmtree } if prune?(dir, days_default: 14)
|
|
end
|
|
end
|
|
|
|
def cleanup_cache(cache = HOMEBREW_CACHE)
|
|
return unless cache.directory?
|
|
cache.children.each do |path|
|
|
if path.to_s.end_with? ".incomplete"
|
|
cleanup_path(path) { path.unlink }
|
|
next
|
|
end
|
|
if %w[glide_home java_cache npm_cache].include?(path.basename.to_s) && path.directory?
|
|
cleanup_path(path) { FileUtils.rm_rf path }
|
|
next
|
|
end
|
|
if prune?(path)
|
|
if path.file?
|
|
cleanup_path(path) { path.unlink }
|
|
elsif path.directory? && path.to_s.include?("--")
|
|
cleanup_path(path) { FileUtils.rm_rf path }
|
|
end
|
|
next
|
|
end
|
|
|
|
next unless path.file?
|
|
file = path
|
|
|
|
if file.to_s =~ Pathname::BOTTLE_EXTNAME_RX
|
|
version = begin
|
|
Utils::Bottles.resolve_version(file)
|
|
rescue
|
|
file.version
|
|
end
|
|
else
|
|
version = file.version
|
|
end
|
|
next unless version
|
|
next unless (name = file.basename.to_s[/(.*)-(?:#{Regexp.escape(version)})/, 1])
|
|
|
|
next unless HOMEBREW_CELLAR.directory?
|
|
|
|
begin
|
|
f = Formulary.from_rack(HOMEBREW_CELLAR/name)
|
|
rescue FormulaUnavailableError, TapFormulaAmbiguityError, TapFormulaWithOldnameAmbiguityError
|
|
next
|
|
end
|
|
|
|
file_is_stale = if version.is_a?(PkgVersion)
|
|
f.pkg_version > version
|
|
else
|
|
f.version > version
|
|
end
|
|
|
|
if file_is_stale || ARGV.switch?("s") && !f.installed? || Utils::Bottles.file_outdated?(f, file)
|
|
cleanup_path(file) { file.unlink }
|
|
end
|
|
end
|
|
end
|
|
|
|
def cleanup_path(path)
|
|
if ARGV.dry_run?
|
|
puts "Would remove: #{path} (#{path.abv})"
|
|
else
|
|
puts "Removing: #{path}... (#{path.abv})"
|
|
yield
|
|
end
|
|
|
|
update_disk_cleanup_size(path.disk_usage)
|
|
end
|
|
|
|
def cleanup_lockfiles
|
|
return unless HOMEBREW_LOCK_DIR.directory?
|
|
candidates = HOMEBREW_LOCK_DIR.children
|
|
lockfiles = candidates.select(&:file?)
|
|
lockfiles.each do |file|
|
|
next unless file.readable?
|
|
file.open.flock(File::LOCK_EX | File::LOCK_NB) && file.unlink
|
|
end
|
|
end
|
|
|
|
def rm_ds_store
|
|
paths = Queue.new
|
|
%w[Cellar Frameworks Library bin etc include lib opt sbin share var]
|
|
.map { |p| HOMEBREW_PREFIX/p }.each { |p| paths << p if p.exist? }
|
|
workers = (0...Hardware::CPU.cores).map do
|
|
Thread.new do
|
|
Kernel.loop do
|
|
begin
|
|
quiet_system "find", paths.deq(true), "-name", ".DS_Store", "-delete"
|
|
rescue ThreadError
|
|
break # if queue is empty
|
|
end
|
|
end
|
|
end
|
|
end
|
|
workers.map(&:join)
|
|
end
|
|
|
|
def prune?(path, options = {})
|
|
@time ||= Time.now
|
|
|
|
path_modified_time = path.mtime
|
|
days_default = options[:days_default]
|
|
|
|
prune = ARGV.value "prune"
|
|
|
|
return true if prune == "all"
|
|
|
|
prune_time = if prune
|
|
@time - 60 * 60 * 24 * prune.to_i
|
|
elsif days_default
|
|
@time - 60 * 60 * 24 * days_default.to_i
|
|
end
|
|
|
|
return false unless prune_time
|
|
|
|
path_modified_time < prune_time
|
|
end
|
|
end
|
|
end
|