brew/Library/Homebrew/cmd/bottle.rb

229 lines
7.1 KiB
Ruby
Raw Normal View History

require 'formula'
require 'bottles'
require 'tab'
require 'keg'
require 'cmd/versions'
require 'utils/inreplace'
2013-09-21 21:21:42 +01:00
require 'erb'
require 'open3'
require 'extend/pathname'
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'
def self.reset_bottle; @bottle = Bottle.new; end
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
module Homebrew extend self
class << self
include Utils::Inreplace
end
def keg_contains string, keg
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
keg_ref_files = `/usr/bin/fgrep --files-with-matches --recursive "#{string}" "#{keg}" 2>/dev/null`
keg_ref_files = (keg_ref_files.map{ |file| Pathname.new(file.strip) }).reject(&:symlink?)
# 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}"
# If we can't use otool on this file, just skip to the next file
next if not file.mach_o_executable? and not file.mach_o_bundle? and not file.dylib? and not file.extname == '.a'
# Get all libraries this file links to, then display only links to libraries that contain string in the path
linked_libraries = `otool -L "#{file}"`.split("\n").drop(1)
linked_libraries.map!{ |lib| lib.strip.split()[0] }
linked_libraries = linked_libraries.select{ |lib| lib.include? string }
linked_libraries.each do |lib|
puts " #{Tty.gray}-->#{Tty.reset} links to #{lib}"
end
# Use strings to search through the file for each string
strings = `strings -t x - "#{file}"`.select{ |str| str.include? string }.map{ |s| s.strip }
# Don't bother reporting a string if it was found by otool
strings.reject!{ |str| linked_libraries.include? str.split[1] }
strings.each do |str|
offset, match = str.split
puts " #{Tty.gray}-->#{Tty.reset} match '#{match}' at offset #{Tty.em}0x#{offset}#{Tty.reset}"
end
end
puts
true
end
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/, '')
end
def bottle_formula f
unless f.installed?
return ofail "Formula not installed: #{f.name}"
end
unless built_as_bottle? f
return ofail "Formula not installed with '--build-bottle': #{f.name}"
end
master_bottle_filenames = f.bottle_filenames 'origin/master'
bottle_revision = -1
begin
bottle_revision += 1
filename = bottle_filename(f, bottle_revision)
end while not ARGV.include? '--no-revision' \
and master_bottle_filenames.include? filename
if bottle_filename_formula_name(filename).empty?
return ofail "Add a new regex to bottle_version.rb to parse the bottle filename."
end
bottle_path = Pathname.pwd/filename
sha1 = nil
prefix = HOMEBREW_PREFIX.to_s
tmp_prefix = '/tmp'
cellar = HOMEBREW_CELLAR.to_s
tmp_cellar = '/tmp/Cellar'
2013-09-21 21:21:42 +01:00
output = nil
HOMEBREW_CELLAR.cd do
2013-09-21 15:15:50 +01:00
ohai "Bottling #{filename}..."
# 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}"
sha1 = bottle_path.sha1
relocatable = false
if File.size?(bottle_path) > 1*1024*1024
ohai "Detecting if #{filename} is relocatable..."
end
keg = Keg.new f.prefix
keg.lock do
# Relocate bottle library references before testing for built-in
# references to the Cellar e.g. Qt's QMake annoyingly does this.
keg.relocate_install_names prefix, tmp_prefix, cellar, tmp_cellar, :keg_only => f.keg_only?
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) if relocatable
# And do the same thing in reverse to change the library references
# back to how they were.
keg.relocate_install_names tmp_prefix, prefix, tmp_cellar, cellar
end
bottle = Bottle.new
bottle.prefix HOMEBREW_PREFIX
bottle.cellar relocatable ? :any : HOMEBREW_CELLAR
bottle.revision bottle_revision
bottle.sha1 sha1 => bottle_tag
puts "./#{filename}"
2013-09-21 21:21:42 +01:00
output = bottle_output bottle
puts output
end
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
end
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|
BottleMerger.class_eval bottle_block
end
bottle = BottleMerger.new.bottle
next unless bottle
output = bottle_output bottle
puts output
if ARGV.include? '--write'
f = Formula.factory formula_name
formula_relative_path = "Library/Formula/#{f.name}.rb"
if File.exists? formula_relative_path
formula_path = Pathname.new(formula_relative_path)
else
formula_path = HOMEBREW_REPOSITORY+formula_relative_path
end
2013-10-31 19:03:23 -07:00
has_bottle_block = f.class.send(:bottle).checksums.any?
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)
else
2013-10-31 19:03:23 -07:00
s.sub!(/( (url|sha1|head|version) '\S*'\n+)+/m, '\0' + output + "\n")
end
end
2013-10-31 19:03:23 -07:00
update_or_add = has_bottle_block ? 'update' : 'add'
safe_system 'git', 'commit', formula_path, '-m',
"#{f.name}: #{update_or_add} bottle."
end
end
exit 0
end
def bottle
merge if ARGV.include? '--merge'
ARGV.formulae.each do |f|
bottle_formula f
end
end
end