mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
168 lines
4.7 KiB
Ruby
168 lines
4.7 KiB
Ruby
# typed: true
|
|
# frozen_string_literal: true
|
|
|
|
# Cleans a newly installed keg.
|
|
# By default:
|
|
#
|
|
# * removes `.la` files
|
|
# * removes `perllocal.pod` files
|
|
# * removes `.packlist` files
|
|
# * removes empty directories
|
|
# * sets permissions on executables
|
|
# * removes unresolved symlinks
|
|
class Cleaner
|
|
include Context
|
|
|
|
# Create a cleaner for the given formula.
|
|
def initialize(f)
|
|
@f = f
|
|
end
|
|
|
|
# Clean the keg of the formula.
|
|
def clean
|
|
ObserverPathnameExtension.reset_counts!
|
|
|
|
# Many formulae include 'lib/charset.alias', but it is not strictly needed
|
|
# and will conflict if more than one formula provides it
|
|
observe_file_removal @f.lib/"charset.alias"
|
|
|
|
[@f.bin, @f.sbin, @f.lib].each { |d| clean_dir(d) if d.exist? }
|
|
|
|
# Get rid of any info 'dir' files, so they don't conflict at the link stage
|
|
#
|
|
# The 'dir' files come in at least 3 locations:
|
|
#
|
|
# 1. 'info/dir'
|
|
# 2. 'info/#{name}/dir'
|
|
# 3. 'info/#{arch}/dir'
|
|
#
|
|
# Of these 3 only 'info/#{name}/dir' is safe to keep since the rest will
|
|
# conflict with other formulae because they use a shared location.
|
|
#
|
|
# See [cleaner: recursively delete info `dir`s by gromgit · Pull Request
|
|
# #11597][1], [emacs 28.1 bottle does not contain `dir` file · Issue
|
|
# #100190][2], and [Keep `info/#{f.name}/dir` files in cleaner by
|
|
# timvisher][3] for more info.
|
|
#
|
|
# [1]: https://github.com/Homebrew/brew/pull/11597
|
|
# [2]: https://github.com/Homebrew/homebrew-core/issues/100190
|
|
# [3]: https://github.com/Homebrew/brew/pull/13215
|
|
Dir.glob(@f.info/"**/dir").each do |f|
|
|
info_dir_file = Pathname(f)
|
|
next unless info_dir_file.file?
|
|
next if info_dir_file == @f.info/@f.name/"dir"
|
|
next if @f.skip_clean?(info_dir_file)
|
|
|
|
observe_file_removal info_dir_file
|
|
end
|
|
|
|
rewrite_shebangs
|
|
|
|
prune
|
|
end
|
|
|
|
private
|
|
|
|
def observe_file_removal(path)
|
|
path.extend(ObserverPathnameExtension).unlink if path.exist?
|
|
end
|
|
|
|
# Removes any empty directories in the formula's prefix subtree
|
|
# Keeps any empty directories protected by skip_clean
|
|
# Removes any unresolved symlinks
|
|
def prune
|
|
dirs = []
|
|
symlinks = []
|
|
@f.prefix.find do |path|
|
|
if path == @f.libexec || @f.skip_clean?(path)
|
|
Find.prune
|
|
elsif path.symlink?
|
|
symlinks << path
|
|
elsif path.directory?
|
|
dirs << path
|
|
end
|
|
end
|
|
|
|
# Remove directories opposite from traversal, so that a subtree with no
|
|
# actual files gets removed correctly.
|
|
dirs.reverse_each do |d|
|
|
if d.children.empty?
|
|
puts "rmdir: #{d} (empty)" if verbose?
|
|
d.rmdir
|
|
end
|
|
end
|
|
|
|
# Remove unresolved symlinks
|
|
symlinks.reverse_each do |s|
|
|
s.unlink unless s.resolved_path_exists?
|
|
end
|
|
end
|
|
|
|
def executable_path?(path)
|
|
path.text_executable? || path.executable?
|
|
end
|
|
|
|
# Both these files are completely unnecessary to package and cause
|
|
# pointless conflicts with other formulae. They are removed by Debian,
|
|
# Arch & MacPorts amongst other packagers as well. The files are
|
|
# created as part of installing any Perl module.
|
|
PERL_BASENAMES = Set.new(%w[perllocal.pod .packlist]).freeze
|
|
|
|
# Clean a top-level (bin, sbin, lib) directory, recursively, by fixing file
|
|
# permissions and removing .la files, unless the files (or parent
|
|
# directories) are protected by skip_clean.
|
|
#
|
|
# bin and sbin should not have any subdirectories; if either do that is
|
|
# caught as an audit warning
|
|
#
|
|
# lib may have a large directory tree (see Erlang for instance), and
|
|
# clean_dir applies cleaning rules to the entire tree
|
|
def clean_dir(directory)
|
|
directory.find do |path|
|
|
path.extend(ObserverPathnameExtension)
|
|
|
|
Find.prune if @f.skip_clean? path
|
|
|
|
next if path.directory?
|
|
|
|
if path.extname == ".la" || PERL_BASENAMES.include?(path.basename.to_s)
|
|
path.unlink
|
|
elsif path.symlink?
|
|
# Skip it.
|
|
else
|
|
# Set permissions for executables and non-executables
|
|
perms = if executable_path?(path)
|
|
0555
|
|
else
|
|
0444
|
|
end
|
|
if debug?
|
|
old_perms = path.stat.mode & 0777
|
|
odebug "Fixing #{path} permissions from #{old_perms.to_s(8)} to #{perms.to_s(8)}" if perms != old_perms
|
|
end
|
|
path.chmod perms
|
|
end
|
|
end
|
|
end
|
|
|
|
def rewrite_shebangs
|
|
require "language/perl"
|
|
require "utils/shebang"
|
|
|
|
basepath = @f.prefix.realpath
|
|
basepath.find do |path|
|
|
Find.prune if @f.skip_clean? path
|
|
|
|
next if path.directory? || path.symlink?
|
|
|
|
begin
|
|
Utils::Shebang.rewrite_shebang Language::Perl::Shebang.detected_perl_shebang(@f), path
|
|
rescue ShebangDetectionError
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
require "extend/os/cleaner"
|