 48fdd163bc
			
		
	
	
		48fdd163bc
		
	
	
	
	
		
			
			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
 |