brew/Library/Homebrew/linkage_checker.rb

188 lines
5.6 KiB
Ruby
Raw Normal View History

2016-07-07 20:41:14 +08:00
require "set"
require "keg"
require "formula"
class LinkageChecker
attr_reader :keg, :formula
2016-07-07 20:41:14 +08:00
attr_reader :brewed_dylibs, :system_dylibs, :broken_dylibs, :variable_dylibs
attr_reader :undeclared_deps, :unnecessary_deps, :reverse_links
2016-07-07 20:41:14 +08:00
def initialize(keg, formula = nil)
2016-07-07 20:41:14 +08:00
@keg = keg
@formula = formula || resolve_formula(keg)
2016-07-07 20:41:14 +08:00
@brewed_dylibs = Hash.new { |h, k| h[k] = Set.new }
@system_dylibs = Set.new
@broken_dylibs = Hash.new { |h, k| h[k] = Set.new }
2016-07-07 20:41:14 +08:00
@variable_dylibs = Set.new
@indirect_deps = []
2016-07-07 20:41:14 +08:00
@undeclared_deps = []
@reverse_links = Hash.new { |h, k| h[k] = Set.new }
@unnecessary_deps = []
2016-07-07 20:41:14 +08:00
check_dylibs
end
def dylib_to_dep(dylib)
2018-03-20 12:30:14 -05:00
dylib =~ %r{#{Regexp.escape(HOMEBREW_PREFIX)}/(opt|Cellar)/([\w+-.@]+)/}
Regexp.last_match(2)
end
2016-07-07 20:41:14 +08:00
def check_dylibs
checked_dylibs = Set.new
2016-07-07 20:41:14 +08:00
@keg.find do |file|
next if file.symlink? || file.directory?
2017-12-01 16:43:00 -08:00
next unless file.dylib? || file.binary_executable? || file.mach_o_bundle?
# weakly loaded dylibs may not actually exist on disk, so skip them
# when checking for broken linkage
file.dynamically_linked_libraries(except: :LC_LOAD_WEAK_DYLIB).each do |dylib|
2016-07-07 20:41:14 +08:00
@reverse_links[dylib] << file
next if checked_dylibs.include? dylib
2016-07-07 20:41:14 +08:00
if dylib.start_with? "@"
@variable_dylibs << dylib
else
begin
owner = Keg.for Pathname.new(dylib)
rescue NotAKegError
@system_dylibs << dylib
rescue Errno::ENOENT
next if harmless_broken_link?(dylib)
@broken_dylibs[dylib_to_dep(dylib)] << dylib
2016-07-07 20:41:14 +08:00
else
tap = Tab.for_keg(owner).tap
f = if tap.nil? || tap.core_tap?
owner.name
else
"#{tap}/#{owner.name}"
end
@brewed_dylibs[f] << dylib
2016-07-07 20:41:14 +08:00
end
end
checked_dylibs << dylib
2016-07-07 20:41:14 +08:00
end
end
@indirect_deps, @undeclared_deps, @unnecessary_deps = check_undeclared_deps if formula
end
def check_undeclared_deps
filter_out = proc do |dep|
next true if dep.build?
next false unless dep.optional? || dep.recommended?
formula.build.without?(dep)
end
declared_deps = formula.deps.reject { |dep| filter_out.call(dep) }.map(&:name)
recursive_deps = keg.to_formula.runtime_dependencies.map { |dep| dep.to_formula.name }
declared_dep_names = declared_deps.map { |dep| dep.split("/").last }
indirect_deps = []
undeclared_deps = []
@brewed_dylibs.each_key do |full_name|
name = full_name.split("/").last
next if name == formula.name
if recursive_deps.include?(name)
indirect_deps << full_name unless declared_dep_names.include?(name)
else
undeclared_deps << full_name
end
end
sort_by_formula_full_name!(indirect_deps)
sort_by_formula_full_name!(undeclared_deps)
unnecessary_deps = declared_dep_names.reject do |full_name|
name = full_name.split("/").last
next true if Formula[name].bin.directory?
@brewed_dylibs.keys.map { |x| x.split("/").last }.include?(name)
end
2018-03-20 10:06:52 -05:00
unnecessary_deps -= @broken_dylibs.map { |_, v| v.map { |d| dylib_to_dep(d) } }.flatten
[indirect_deps, undeclared_deps, unnecessary_deps]
end
def sort_by_formula_full_name!(arr)
arr.sort! do |a, b|
if a.include?("/") && !b.include?("/")
1
elsif !a.include?("/") && b.include?("/")
-1
else
a <=> b
2016-07-07 20:41:14 +08:00
end
end
2016-07-07 20:41:14 +08:00
end
def display_normal_output
display_items "System libraries", @system_dylibs
display_items "Homebrew libraries", @brewed_dylibs
display_items "Indirect dependencies with linkage", @indirect_deps
2016-07-07 20:41:14 +08:00
display_items "Variable-referenced libraries", @variable_dylibs
display_items "Missing libraries", @broken_dylibs
display_items "Undeclared dependencies with linkage", @undeclared_deps
display_items "Dependencies with no linkage", @unnecessary_deps
2016-07-07 20:41:14 +08:00
end
def display_reverse_output
return if @reverse_links.empty?
sorted = @reverse_links.sort
sorted.each do |dylib, files|
puts dylib
files.each do |f|
unprefixed = f.to_s.strip_prefix "#{@keg}/"
puts " #{unprefixed}"
end
puts unless dylib == sorted.last[0]
end
end
def display_test_output
display_items "Missing libraries", @broken_dylibs
display_items "Dependencies with no linkage", @unnecessary_deps
2016-07-07 20:41:14 +08:00
puts "No broken dylib links" if @broken_dylibs.empty?
end
def broken_dylibs?
!@broken_dylibs.empty?
end
def undeclared_deps?
!@undeclared_deps.empty?
end
def unnecessary_deps?
!@unnecessary_deps.empty?
end
2016-07-07 20:41:14 +08:00
private
# Whether or not dylib is a harmless broken link, meaning that it's
# okay to skip (and not report) as broken.
def harmless_broken_link?(dylib)
# libgcc_s_* is referenced by programs that use the Java Service Wrapper,
# and is harmless on x86(_64) machines
[
"/usr/lib/libgcc_s_ppc64.1.dylib",
"/opt/local/lib/libgcc/libgcc_s.1.dylib",
].include?(dylib)
end
2016-07-07 20:41:14 +08:00
# Display a list of things.
# Things may either be an array, or a hash of (label -> array)
def display_items(label, things)
return if things.empty?
puts "#{label}:"
if things.is_a? Hash
2018-03-20 12:30:14 -05:00
things.sort_by { |k, _| k.to_s }.each do |list_label, list|
2016-07-07 20:41:14 +08:00
list.sort.each do |item|
puts " #{item} (#{list_label})"
end
end
else
things.sort.each do |item|
puts " #{item}"
end
end
end
def resolve_formula(keg)
2016-07-15 17:45:21 +08:00
Formulary.from_keg(keg)
rescue FormulaUnavailableError
opoo "Formula unavailable: #{keg.name}"
end
2016-07-07 20:41:14 +08:00
end