brew/Library/Homebrew/cleaner.rb

207 lines
6.1 KiB
Ruby
Raw Permalink Normal View History

# typed: strict
# frozen_string_literal: true
2012-07-03 07:24:35 -07:00
# Cleans a newly installed keg.
# By default:
#
# * removes `.la` files
# * removes `.tbd` files
# * removes `perllocal.pod` files
# * removes `.packlist` files
2012-07-03 07:24:35 -07:00
# * removes empty directories
# * sets permissions on executables
2014-02-23 21:18:09 -08:00
# * removes unresolved symlinks
2010-08-21 11:06:02 -07:00
class Cleaner
include Context
# Create a cleaner for the given formula.
sig { params(formula: Formula).void }
def initialize(formula)
@formula = formula
end
# Clean the keg of the formula.
sig { void }
def clean
ObserverPathnameExtension.reset_counts!
2014-02-23 17:39:01 -08:00
2024-04-30 11:10:23 +02:00
# 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 @formula.lib/"charset.alias"
2014-02-23 17:39:01 -08:00
[@formula.bin, @formula.sbin, @formula.lib].each { |dir| clean_dir(dir) if dir.exist? }
2010-08-21 11:23:54 -07:00
2024-04-30 11:10:23 +02:00
# Get rid of any info `dir` files, so they don't conflict at the link stage.
2022-04-28 10:19:16 -04:00
#
2024-04-30 11:10:23 +02:00
# The `dir` files come in at least 3 locations:
2022-04-28 10:19:16 -04:00
#
2024-04-30 11:10:23 +02:00
# 1. `info/dir`
# 2. `info/#{name}/dir`
# 3. `info/#{arch}/dir`
2022-04-28 10:19:16 -04:00
#
2024-04-30 11:10:23 +02:00
# Of these 3 only `info/#{name}/dir` is safe to keep since the rest will
2022-04-28 10:19:16 -04:00
# conflict with other formulae because they use a shared location.
#
2024-04-30 11:10:23 +02:00
# See
# [cleaner: recursively delete info `dir`s][1],
# [emacs 28.1 bottle does not contain `dir` file][2] and
# [Keep `info/#{f.name}/dir` files in cleaner][3]
# for more info.
2022-04-28 10:19:16 -04:00
#
# [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
@formula.info.glob("**/dir").each do |info_dir_file|
2022-04-28 00:15:29 -04:00
next unless info_dir_file.file?
next if info_dir_file == @formula.info/@formula.name/"dir"
next if @formula.skip_clean?(info_dir_file)
2022-04-28 00:09:18 -04:00
2022-04-28 00:15:29 -04:00
observe_file_removal info_dir_file
end
rewrite_shebangs
clean_python_metadata
prune
end
private
sig { params(path: Pathname).void }
def observe_file_removal(path)
path.extend(ObserverPathnameExtension).unlink if path.exist?
end
2014-02-23 21:18:09 -08:00
# Removes any empty directories in the formula's prefix subtree
2019-04-08 10:33:49 -07:00
# Keeps any empty directories protected by skip_clean
2014-07-20 09:57:33 -07:00
# Removes any unresolved symlinks
sig { void }
def prune
dirs = []
symlinks = []
@formula.prefix.find do |path|
if path == @formula.libexec || @formula.skip_clean?(path)
Find.prune
elsif path.symlink?
symlinks << path
elsif path.directory?
dirs << path
end
end
2014-02-23 21:18:09 -08:00
# 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
2010-08-21 11:06:02 -07:00
2014-02-23 21:18:09 -08:00
# Remove unresolved symlinks
symlinks.reverse_each do |s|
s.unlink unless s.resolved_path_exists?
end
end
2010-11-09 13:00:33 +00:00
sig { params(path: Pathname).returns(T::Boolean) }
2016-07-04 16:10:24 +01:00
def executable_path?(path)
path.text_executable? || path.executable?
2016-07-04 16:10:24 +01:00
end
2020-11-13 17:21:51 +01:00
# 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 = T.let(Set.new(%w[perllocal.pod .packlist]).freeze, T::Set[String])
2025-02-17 18:34:18 -08:00
private_constant :PERL_BASENAMES
2020-11-13 17:21:51 +01:00
2024-04-30 11:10:23 +02:00
# Clean a top-level (`bin`, `sbin`, `lib`) directory, recursively, by fixing file
2014-07-19 23:27:58 -07:00
# permissions and removing .la files, unless the files (or parent
# directories) are protected by skip_clean.
2014-07-20 09:51:11 -07:00
#
2024-04-30 11:10:23 +02:00
# `bin` and `sbin` should not have any subdirectories; if either do that is
# caught as an audit warning.
2014-07-20 09:51:11 -07:00
#
2024-04-30 11:10:23 +02:00
# `lib` may have a large directory tree (see Erlang for instance) and
# clean_dir applies cleaning rules to the entire tree.
sig { params(directory: Pathname).void }
def clean_dir(directory)
directory.find do |path|
2013-12-22 13:43:51 -06:00
path.extend(ObserverPathnameExtension)
Find.prune if @formula.skip_clean? path
next if path.directory?
2016-09-22 20:12:28 +02:00
if path.extname == ".la" || path.extname == ".tbd" || PERL_BASENAMES.include?(path.basename.to_s)
path.unlink
2020-11-13 10:07:02 -05:00
elsif path.symlink?
# Skip it.
else
2024-04-30 11:10:23 +02:00
# Set permissions for executables and non-executables.
2016-07-04 16:10:24 +01:00
perms = if executable_path?(path)
0555
else
0444
end
if debug?
2014-02-23 16:46:44 -08:00
old_perms = path.stat.mode & 0777
2019-11-29 14:54:41 -05:00
odebug "Fixing #{path} permissions from #{old_perms.to_s(8)} to #{perms.to_s(8)}" if perms != old_perms
2014-02-23 16:46:44 -08:00
end
path.chmod perms
2010-08-21 11:06:02 -07:00
end
end
end
sig { void }
def rewrite_shebangs
2024-07-05 21:16:35 -07:00
require "language/node"
require "language/perl"
require "utils/shebang"
2024-07-05 21:16:35 -07:00
rewrites = [Language::Node::Shebang.method(:detected_node_shebang),
Language::Perl::Shebang.method(:detected_perl_shebang)].filter_map do |detector|
detector.call(@formula)
rescue ShebangDetectionError
nil
end
return if rewrites.empty?
basepath = @formula.prefix.realpath
basepath.find do |path|
Find.prune if @formula.skip_clean? path
next if path.directory? || path.symlink?
2024-07-05 21:16:35 -07:00
rewrites.each { |rw| Utils::Shebang.rewrite_shebang rw, path }
end
end
# Remove non-reproducible pip direct_url.json which records the /tmp build directory.
# Remove RECORD files to prevent changes to the installed Python package.
# Modify INSTALLER to provide information that files are managed by brew.
#
# @see https://packaging.python.org/en/latest/specifications/recording-installed-packages/
sig { void }
def clean_python_metadata
basepath = @formula.prefix.realpath
basepath.find do |path|
Find.prune if @formula.skip_clean?(path)
next if path.directory? || path.symlink?
next if path.parent.extname != ".dist-info"
case path.basename.to_s
when "direct_url.json", "RECORD"
observe_file_removal path
when "INSTALLER"
odebug "Modifying #{path} contents from #{path.read.chomp} to brew"
path.atomic_write("brew\n")
end
end
end
2010-08-21 11:06:02 -07:00
end
2016-07-04 16:10:24 +01:00
require "extend/os/cleaner"