2012-03-07 17:29:05 -05:00
|
|
|
require 'formula'
|
2012-03-07 21:30:03 -05:00
|
|
|
require 'bottles'
|
2012-03-07 17:29:05 -05:00
|
|
|
require 'tab'
|
2013-03-11 18:56:26 +00:00
|
|
|
require 'keg'
|
2013-09-21 15:16:45 +01:00
|
|
|
require 'cmd/versions'
|
2013-09-21 21:30:57 +01:00
|
|
|
require 'utils/inreplace'
|
2013-09-21 21:21:42 +01:00
|
|
|
require 'erb'
|
2013-10-31 01:20:28 -07:00
|
|
|
require 'extend/pathname'
|
2012-03-07 17:29:05 -05:00
|
|
|
|
2013-06-08 16:48:43 +01:00
|
|
|
class BottleMerger < Formula
|
|
|
|
# This provides a URL and Version which are the only needed properties of
|
|
|
|
# a Formula. This object is used to access the Formula bottle DSL to merge
|
|
|
|
# multiple outputs of `brew bottle`.
|
|
|
|
url '1'
|
2013-09-21 15:16:16 +01:00
|
|
|
def self.reset_bottle; @bottle = Bottle.new; end
|
2013-06-08 16:48:43 +01:00
|
|
|
end
|
|
|
|
|
2013-09-21 21:21:42 +01:00
|
|
|
BOTTLE_ERB = <<-EOS
|
|
|
|
bottle do
|
|
|
|
<% if prefix.to_s != '/usr/local' %>
|
|
|
|
prefix '<%= prefix %>'
|
|
|
|
<% end %>
|
|
|
|
<% if cellar.is_a? Symbol %>
|
|
|
|
cellar :<%= cellar %>
|
2013-09-23 17:30:33 +01:00
|
|
|
<% elsif cellar.to_s != '/usr/local/Cellar' %>
|
2013-09-21 21:21:42 +01:00
|
|
|
cellar '<%= cellar %>'
|
|
|
|
<% end %>
|
|
|
|
<% if revision > 0 %>
|
|
|
|
revision <%= revision %>
|
|
|
|
<% end %>
|
2013-09-23 17:30:47 +01:00
|
|
|
<% checksums.each do |checksum_type, checksum_values| %>
|
|
|
|
<% checksum_values.each do |checksum_value| %>
|
|
|
|
<% checksum, osx = checksum_value.shift %>
|
2013-09-21 21:21:42 +01:00
|
|
|
<%= checksum_type %> '<%= checksum %>' => :<%= osx %>
|
|
|
|
<% end %>
|
2013-09-23 17:30:47 +01:00
|
|
|
<% end %>
|
2013-09-21 21:21:42 +01:00
|
|
|
end
|
|
|
|
EOS
|
|
|
|
|
2012-03-07 17:29:05 -05:00
|
|
|
module Homebrew extend self
|
2013-09-21 21:30:57 +01:00
|
|
|
class << self
|
|
|
|
include Utils::Inreplace
|
|
|
|
end
|
|
|
|
|
2013-12-08 16:17:11 -06:00
|
|
|
def uniq_by_ino(list)
|
|
|
|
h = {}
|
|
|
|
list.each do |e|
|
|
|
|
ino = e.stat.ino
|
|
|
|
h[ino] = e unless h.key? ino
|
|
|
|
end
|
|
|
|
h.values
|
|
|
|
end
|
|
|
|
|
2013-03-11 18:56:26 +00:00
|
|
|
def keg_contains string, keg
|
2013-10-31 01:20:28 -07:00
|
|
|
if not ARGV.homebrew_developer?
|
|
|
|
return quiet_system 'fgrep', '--recursive', '--quiet', '--max-count=1', string, keg
|
|
|
|
end
|
|
|
|
|
|
|
|
# Find all files that still reference the keg via a string search
|
2013-12-05 16:39:38 -06:00
|
|
|
keg_ref_files = `/usr/bin/fgrep --files-with-matches --recursive "#{string}" "#{keg}" 2>/dev/null`.split("\n")
|
2013-12-05 15:08:37 -06:00
|
|
|
keg_ref_files.map! { |file| Pathname.new(file) }.reject!(&:symlink?)
|
2013-10-31 01:20:28 -07:00
|
|
|
|
2013-12-08 16:17:11 -06:00
|
|
|
# If files are hardlinked, only check one of them
|
|
|
|
keg_ref_files = uniq_by_ino(keg_ref_files)
|
|
|
|
|
2013-10-31 01:20:28 -07:00
|
|
|
# If there are no files with that string found, return immediately
|
|
|
|
return false if keg_ref_files.empty?
|
|
|
|
|
|
|
|
# Start printing out each file and any extra information we can find
|
|
|
|
opoo "String '#{string}' still exists in these files:"
|
|
|
|
keg_ref_files.each do |file|
|
|
|
|
puts "#{Tty.red}#{file}#{Tty.reset}"
|
|
|
|
|
2013-12-05 16:39:39 -06:00
|
|
|
# Check dynamic library linkage. Importantly, do not run otool on static
|
|
|
|
# libraries, which will falsely report "linkage" to themselves.
|
|
|
|
if file.mach_o_executable? or file.dylib? or file.mach_o_bundle?
|
2013-12-14 09:35:58 -06:00
|
|
|
linked_libraries = file.dynamically_linked_libraries
|
2013-12-05 16:39:39 -06:00
|
|
|
linked_libraries = linked_libraries.select { |lib| lib.include? string }
|
2013-12-14 09:35:58 -06:00
|
|
|
else
|
|
|
|
linked_libraries = []
|
2013-12-05 16:39:39 -06:00
|
|
|
end
|
2013-10-31 01:20:28 -07:00
|
|
|
|
|
|
|
linked_libraries.each do |lib|
|
|
|
|
puts " #{Tty.gray}-->#{Tty.reset} links to #{lib}"
|
|
|
|
end
|
|
|
|
|
|
|
|
# Use strings to search through the file for each string
|
2013-12-14 15:43:15 -06:00
|
|
|
IO.popen("strings -t x - '#{file}'") do |io|
|
|
|
|
until io.eof?
|
|
|
|
str = io.readline.chomp
|
2013-10-31 01:20:28 -07:00
|
|
|
|
2013-12-14 15:43:15 -06:00
|
|
|
next unless str.include? string
|
|
|
|
|
|
|
|
offset, match = str.split(" ", 2)
|
|
|
|
|
|
|
|
next if linked_libraries.include? match # Don't bother reporting a string if it was found by otool
|
|
|
|
puts " #{Tty.gray}-->#{Tty.reset} match '#{match}' at offset #{Tty.em}0x#{offset}#{Tty.reset}"
|
|
|
|
end
|
2013-10-31 01:20:28 -07:00
|
|
|
end
|
|
|
|
end
|
|
|
|
puts
|
|
|
|
true
|
2013-03-11 18:56:26 +00:00
|
|
|
end
|
|
|
|
|
2013-06-04 20:39:09 +01:00
|
|
|
def bottle_output bottle
|
2013-09-21 21:21:42 +01:00
|
|
|
erb = ERB.new BOTTLE_ERB
|
|
|
|
erb.result(bottle.instance_eval { binding }).gsub(/^\s*$\n/, '')
|
2013-06-04 20:39:09 +01:00
|
|
|
end
|
|
|
|
|
2012-03-07 17:29:05 -05:00
|
|
|
def bottle_formula f
|
2012-03-15 10:57:34 +13:00
|
|
|
unless f.installed?
|
2012-04-30 14:08:59 +10:00
|
|
|
return ofail "Formula not installed: #{f.name}"
|
2012-03-15 10:57:34 +13:00
|
|
|
end
|
|
|
|
|
2012-08-25 11:31:57 -07:00
|
|
|
unless built_as_bottle? f
|
2012-04-30 14:08:59 +10:00
|
|
|
return ofail "Formula not installed with '--build-bottle': #{f.name}"
|
2012-03-15 10:57:34 +13:00
|
|
|
end
|
2012-03-07 17:29:05 -05:00
|
|
|
|
2013-12-10 15:16:22 -06:00
|
|
|
if ARGV.include? '--no-revision'
|
|
|
|
bottle_revision = 0
|
|
|
|
else
|
|
|
|
max = f.bottle_version_map('origin/master')[f.version].max
|
|
|
|
bottle_revision = max ? max + 1 : 0
|
|
|
|
end
|
|
|
|
|
|
|
|
filename = bottle_filename(f, :tag => bottle_tag, :revision => bottle_revision)
|
2013-07-04 11:23:09 +01:00
|
|
|
|
|
|
|
if bottle_filename_formula_name(filename).empty?
|
2013-08-04 08:25:51 -07:00
|
|
|
return ofail "Add a new regex to bottle_version.rb to parse the bottle filename."
|
2013-07-04 11:23:09 +01:00
|
|
|
end
|
|
|
|
|
2013-02-09 19:06:54 -08:00
|
|
|
bottle_path = Pathname.pwd/filename
|
2012-03-07 17:29:05 -05:00
|
|
|
|
2013-03-11 18:56:26 +00:00
|
|
|
prefix = HOMEBREW_PREFIX.to_s
|
|
|
|
cellar = HOMEBREW_CELLAR.to_s
|
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
ohai "Bottling #{filename}..."
|
2013-09-21 21:21:42 +01:00
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
keg = Keg.new(f.prefix)
|
|
|
|
relocatable = false
|
2013-12-04 22:37:57 -06:00
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
keg.lock do
|
|
|
|
begin
|
|
|
|
keg.relocate_install_names prefix, Keg::PREFIX_PLACEHOLDER,
|
|
|
|
cellar, Keg::CELLAR_PLACEHOLDER, :keg_only => f.keg_only?
|
2013-12-04 22:37:57 -06:00
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
HOMEBREW_CELLAR.cd do
|
2013-12-04 22:37:57 -06:00
|
|
|
# Use gzip, faster to compress than bzip2, faster to uncompress than bzip2
|
|
|
|
# or an uncompressed tarball (and more bandwidth friendly).
|
|
|
|
safe_system 'tar', 'czf', bottle_path, "#{f.name}/#{f.version}"
|
2013-12-12 19:46:37 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
if File.size?(bottle_path) > 1*1024*1024
|
|
|
|
ohai "Detecting if #{filename} is relocatable..."
|
|
|
|
end
|
2013-12-04 22:37:57 -06:00
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
if prefix == '/usr/local'
|
|
|
|
prefix_check = HOMEBREW_PREFIX/'opt'
|
|
|
|
else
|
|
|
|
prefix_check = HOMEBREW_PREFIX
|
|
|
|
end
|
|
|
|
|
|
|
|
relocatable = !keg_contains(prefix_check, keg)
|
|
|
|
relocatable = !keg_contains(HOMEBREW_CELLAR, keg) && relocatable
|
|
|
|
rescue Interrupt
|
|
|
|
ignore_interrupts { bottle_path.unlink if bottle_path.exist? }
|
|
|
|
raise
|
|
|
|
ensure
|
|
|
|
ignore_interrupts do
|
|
|
|
keg.relocate_install_names Keg::PREFIX_PLACEHOLDER, prefix,
|
|
|
|
Keg::CELLAR_PLACEHOLDER, cellar, :keg_only => f.keg_only?
|
2013-12-04 22:37:57 -06:00
|
|
|
end
|
|
|
|
end
|
2013-12-12 19:46:37 -06:00
|
|
|
end
|
2013-12-04 22:37:57 -06:00
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
bottle = Bottle.new
|
|
|
|
bottle.prefix HOMEBREW_PREFIX
|
|
|
|
bottle.cellar relocatable ? :any : HOMEBREW_CELLAR
|
|
|
|
bottle.revision bottle_revision
|
|
|
|
bottle.sha1 bottle_path.sha1 => bottle_tag
|
2013-03-11 18:56:26 +00:00
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
output = bottle_output bottle
|
2013-03-11 18:56:26 +00:00
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
puts "./#{filename}"
|
|
|
|
puts output
|
2013-09-21 21:24:50 +01:00
|
|
|
|
|
|
|
if ARGV.include? '--rb'
|
|
|
|
bottle_base = filename.gsub(bottle_suffix(bottle_revision), '')
|
|
|
|
File.open "#{bottle_base}.bottle.rb", 'w' do |file|
|
|
|
|
file.write output
|
|
|
|
end
|
|
|
|
end
|
2012-03-07 17:29:05 -05:00
|
|
|
end
|
|
|
|
|
2013-09-21 15:16:16 +01:00
|
|
|
def merge
|
|
|
|
merge_hash = {}
|
|
|
|
ARGV.named.each do |argument|
|
|
|
|
formula_name = bottle_filename_formula_name argument
|
|
|
|
merge_hash[formula_name] ||= []
|
|
|
|
bottle_block = IO.read argument
|
|
|
|
merge_hash[formula_name] << bottle_block
|
|
|
|
end
|
|
|
|
merge_hash.keys.each do |formula_name|
|
|
|
|
BottleMerger.reset_bottle
|
|
|
|
ohai formula_name
|
|
|
|
bottle_blocks = merge_hash[formula_name]
|
|
|
|
bottle_blocks.each do |bottle_block|
|
2013-06-08 16:48:43 +01:00
|
|
|
BottleMerger.class_eval bottle_block
|
|
|
|
end
|
|
|
|
bottle = BottleMerger.new.bottle
|
2013-09-21 21:30:57 +01:00
|
|
|
next unless bottle
|
|
|
|
output = bottle_output bottle
|
|
|
|
puts output
|
|
|
|
|
|
|
|
if ARGV.include? '--write'
|
|
|
|
f = Formula.factory formula_name
|
2013-10-31 18:21:03 -07:00
|
|
|
formula_relative_path = "Library/Formula/#{f.name}.rb"
|
2013-11-12 11:29:50 +00:00
|
|
|
formula_path = HOMEBREW_REPOSITORY+formula_relative_path
|
2013-10-31 19:03:23 -07:00
|
|
|
has_bottle_block = f.class.send(:bottle).checksums.any?
|
2013-09-21 21:30:57 +01:00
|
|
|
inreplace formula_path do |s|
|
2013-10-31 19:03:23 -07:00
|
|
|
if has_bottle_block
|
|
|
|
s.sub!(/ bottle do.+?end\n/m, output)
|
2013-09-21 21:30:57 +01:00
|
|
|
else
|
2013-10-31 19:03:23 -07:00
|
|
|
s.sub!(/( (url|sha1|head|version) '\S*'\n+)+/m, '\0' + output + "\n")
|
2013-09-21 21:30:57 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-10-31 19:03:23 -07:00
|
|
|
update_or_add = has_bottle_block ? 'update' : 'add'
|
2013-09-21 21:30:57 +01:00
|
|
|
|
2013-12-01 10:25:07 -08:00
|
|
|
safe_system 'git', 'commit', '--no-edit', '--verbose',
|
|
|
|
"--message=#{f.name}: #{update_or_add} #{f.version} bottle.",
|
|
|
|
'--', formula_path
|
2013-09-21 21:30:57 +01:00
|
|
|
end
|
2013-06-08 16:48:43 +01:00
|
|
|
end
|
2013-09-21 15:16:16 +01:00
|
|
|
exit 0
|
|
|
|
end
|
|
|
|
|
|
|
|
def bottle
|
|
|
|
merge if ARGV.include? '--merge'
|
2013-06-08 16:48:43 +01:00
|
|
|
|
|
|
|
ARGV.formulae.each do |f|
|
2013-06-23 13:02:02 -07:00
|
|
|
bottle_formula f
|
2012-03-07 17:29:05 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|