2016-04-17 21:25:11 -04:00
|
|
|
#
|
|
|
|
# Description: check linkage of installed keg
|
|
|
|
# Usage:
|
|
|
|
# brew linkage <formulae>
|
|
|
|
#
|
|
|
|
# Only works on installed formulae. An error is raised if it is run on uninstalled
|
|
|
|
# formulae.
|
|
|
|
#
|
|
|
|
# Options:
|
|
|
|
# --test - testing version: only display broken libs; exit non-zero if any
|
|
|
|
# breakage was found.
|
|
|
|
|
|
|
|
require "set"
|
|
|
|
require "keg"
|
|
|
|
require "formula"
|
|
|
|
|
|
|
|
module Homebrew
|
|
|
|
def linkage
|
|
|
|
found_broken_dylibs = false
|
|
|
|
ARGV.kegs.each do |keg|
|
|
|
|
ohai "Checking #{keg.name} linkage" if ARGV.kegs.size > 1
|
|
|
|
result = LinkageChecker.new(keg)
|
|
|
|
if ARGV.include?("--test")
|
|
|
|
result.display_test_output
|
|
|
|
else
|
|
|
|
result.display_normal_output
|
|
|
|
end
|
2016-06-19 21:19:20 +02:00
|
|
|
found_broken_dylibs = true unless result.broken_dylibs.empty?
|
2016-04-17 21:25:11 -04:00
|
|
|
end
|
|
|
|
if ARGV.include?("--test") && found_broken_dylibs
|
|
|
|
exit 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class LinkageChecker
|
|
|
|
attr_reader :keg
|
|
|
|
attr_reader :broken_dylibs
|
|
|
|
|
|
|
|
def initialize(keg)
|
|
|
|
@keg = keg
|
|
|
|
@brewed_dylibs = Hash.new { |h, k| h[k] = Set.new }
|
|
|
|
@system_dylibs = Set.new
|
|
|
|
@broken_dylibs = Set.new
|
|
|
|
@variable_dylibs = Set.new
|
|
|
|
check_dylibs
|
|
|
|
end
|
|
|
|
|
|
|
|
def check_dylibs
|
|
|
|
@keg.find do |file|
|
2016-06-19 21:10:05 +02:00
|
|
|
next if file.symlink? || file.directory?
|
2016-04-17 21:25:11 -04:00
|
|
|
next unless file.dylib? || file.mach_o_executable? || file.mach_o_bundle?
|
|
|
|
file.dynamically_linked_libraries.each do |dylib|
|
|
|
|
if dylib.start_with? "@"
|
|
|
|
@variable_dylibs << dylib
|
|
|
|
else
|
|
|
|
begin
|
|
|
|
owner = Keg.for Pathname.new(dylib)
|
|
|
|
rescue NotAKegError
|
|
|
|
@system_dylibs << dylib
|
|
|
|
rescue Errno::ENOENT
|
|
|
|
@broken_dylibs << dylib
|
|
|
|
else
|
|
|
|
@brewed_dylibs[owner.name] << dylib
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
begin
|
2016-04-19 13:55:47 +08:00
|
|
|
f = Formulary.from_rack(keg.rack)
|
2016-04-17 21:25:11 -04:00
|
|
|
@undeclared_deps = @brewed_dylibs.keys - f.deps.map(&:name)
|
|
|
|
@undeclared_deps -= [f.name]
|
|
|
|
rescue FormulaUnavailableError
|
|
|
|
opoo "Formula unavailable: #{keg.name}"
|
|
|
|
@undeclared_deps = []
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def display_normal_output
|
2016-06-19 21:24:16 +02:00
|
|
|
display_items "System libraries", @system_dylibs
|
|
|
|
display_items "Homebrew libraries", @brewed_dylibs
|
|
|
|
display_items "Variable-referenced libraries", @variable_dylibs
|
|
|
|
display_items "Missing libraries", @broken_dylibs
|
|
|
|
display_items "Possible undeclared dependencies", @undeclared_deps
|
2016-04-17 21:25:11 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def display_test_output
|
2016-06-19 21:24:16 +02:00
|
|
|
display_items "Missing libraries", @broken_dylibs
|
|
|
|
puts "No broken dylib links" if @broken_dylibs.empty?
|
2016-04-17 21:25:11 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
# Display a list of things.
|
|
|
|
# Things may either be an array, or a hash of (label -> array)
|
|
|
|
def display_items(label, things)
|
2016-06-19 21:24:16 +02:00
|
|
|
return if things.empty?
|
2016-04-17 21:25:11 -04:00
|
|
|
puts "#{label}:"
|
|
|
|
if things.is_a? Hash
|
2016-04-19 15:07:59 +08:00
|
|
|
things.sort.each do |list_label, list|
|
2016-04-17 21:25:11 -04:00
|
|
|
list.sort.each do |item|
|
2016-04-19 15:07:59 +08:00
|
|
|
puts " #{item} (#{list_label})"
|
2016-04-17 21:25:11 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
else
|
|
|
|
things.sort.each do |item|
|
|
|
|
puts " #{item}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-04-19 13:55:47 +08:00
|
|
|
end
|