brew/Library/Homebrew/cmd/bottle.rb

326 lines
9.9 KiB
Ruby
Raw Normal View History

require "formula"
require "bottles"
require "tab"
require "keg"
require "formula_versions"
require "utils/inreplace"
require "erb"
require "extend/pathname"
2013-09-21 21:21:42 +01:00
BOTTLE_ERB = <<-EOS
bottle do
<% if !root_url.start_with?(BottleSpecification::DEFAULT_DOMAIN) %>
root_url "<%= root_url %>"
<% end %>
<% if prefix != BottleSpecification::DEFAULT_PREFIX %>
2014-01-18 21:40:52 +00:00
prefix "<%= prefix %>"
2013-09-21 21:21:42 +01:00
<% end %>
<% if cellar.is_a? Symbol %>
cellar :<%= cellar %>
<% elsif cellar != BottleSpecification::DEFAULT_CELLAR %>
2014-01-18 21:40:52 +00:00
cellar "<%= cellar %>"
2013-09-21 21:21:42 +01:00
<% 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 %>
2014-01-18 21:40:52 +00:00
<%= checksum_type %> "<%= checksum %>" => :<%= osx %>
2013-09-21 21:21:42 +01:00
<% end %>
2013-09-23 17:30:47 +01:00
<% end %>
2013-09-21 21:21:42 +01:00
end
EOS
module Homebrew
def keg_contains(string, keg, ignores)
2015-02-26 19:13:10 +00:00
@put_string_exists_header, @put_filenames = nil
def print_filename(string, filename)
2015-02-26 19:13:10 +00:00
unless @put_string_exists_header
opoo "String '#{string}' still exists in these files:"
@put_string_exists_header = true
end
2015-02-26 19:13:10 +00:00
@put_filenames ||= []
unless @put_filenames.include? filename
puts "#{Tty.red}#{filename}#{Tty.reset}"
@put_filenames << filename
end
end
result = false
keg.each_unique_file_matching(string) do |file|
# skip document file.
next if Metafiles::EXTENSIONS.include? file.extname
# Check dynamic library linkage. Importantly, do not run otool on static
# libraries, which will falsely report "linkage" to themselves.
if file.mach_o_executable? || file.dylib? || file.mach_o_bundle?
2013-12-14 09:35:58 -06:00
linked_libraries = file.dynamically_linked_libraries
linked_libraries = linked_libraries.select { |lib| lib.include? string }
result ||= linked_libraries.any?
2013-12-14 09:35:58 -06:00
else
linked_libraries = []
end
if ARGV.verbose?
print_filename(string, file) if linked_libraries.any?
linked_libraries.each do |lib|
puts " #{Tty.gray}-->#{Tty.reset} links to #{lib}"
end
end
# Use strings to search through the file for each string
Utils.popen_read("strings", "-t", "x", "-", file.to_s) do |io|
until io.eof?
str = io.readline.chomp
next if ignores.any? { |i| i =~ str }
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
result ||= true
if ARGV.verbose?
print_filename string, file
puts " #{Tty.gray}-->#{Tty.reset} match '#{match}' at offset #{Tty.em}0x#{offset}#{Tty.reset}"
end
end
end
end
put_symlink_header = false
keg.find do |pn|
if pn.symlink? && (link = pn.readlink).absolute?
if !put_symlink_header && link.to_s.start_with?(string)
opoo "Absolute symlink starting with #{string}:"
puts " #{pn} -> #{pn.resolved_path}"
put_symlink_header = true
end
result = true
end
end
result
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?
2015-05-27 20:44:51 +08:00
return ofail "Formula not installed or up-to-date: #{f.full_name}"
end
unless built_as_bottle? f
2015-05-27 20:44:51 +08:00
return ofail "Formula not installed with '--build-bottle': #{f.full_name}"
end
unless f.stable
2015-05-27 20:44:51 +08:00
return ofail "Formula has no stable version: #{f.full_name}"
end
if ARGV.include? "--no-revision"
bottle_revision = 0
elsif ARGV.include? "--keep-old"
bottle_revision = f.bottle_specification.revision
else
2015-05-27 20:44:51 +08:00
ohai "Determining #{f.full_name} bottle revision..."
versions = FormulaVersions.new(f)
bottle_revisions = versions.bottle_version_map("origin/master")[f.pkg_version]
bottle_revisions.pop if bottle_revisions.last.to_i > 0
bottle_revision = bottle_revisions.any? ? bottle_revisions.max.to_i + 1 : 0
end
filename = Bottle::Filename.create(f, bottle_tag, bottle_revision)
bottle_path = Pathname.pwd/filename
prefix = HOMEBREW_PREFIX.to_s
cellar = HOMEBREW_CELLAR.to_s
ohai "Bottling #{filename}..."
2013-09-21 21:21:42 +01:00
keg = Keg.new(f.prefix)
relocatable = false
keg.lock do
begin
keg.relocate_install_names prefix, Keg::PREFIX_PLACEHOLDER,
cellar, Keg::CELLAR_PLACEHOLDER, :keg_only => f.keg_only?
keg.delete_pyc_files!
cd cellar do
# 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.pkg_version}"
end
2014-07-17 14:56:38 -05:00
if bottle_path.size > 1*1024*1024
ohai "Detecting if #{filename} is relocatable..."
end
if prefix == "/usr/local"
prefix_check = File.join(prefix, "opt")
else
prefix_check = prefix
end
ignores = []
if f.deps.any? { |dep| dep.name == "go" }
ignores << %r{#{HOMEBREW_CELLAR}/go/[\d\.]+/libexec}
end
relocatable = !keg_contains(prefix_check, keg, ignores)
relocatable = !keg_contains(cellar, keg, ignores) && relocatable
puts if !relocatable && ARGV.verbose?
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?
end
end
end
root_url = ARGV.value("root-url")
# Use underscored version for legacy reasons. Remove at some point.
root_url ||= ARGV.value("root_url")
bottle = BottleSpecification.new
bottle.root_url(root_url) if root_url
bottle.prefix prefix
bottle.cellar relocatable ? :any : cellar
bottle.revision bottle_revision
bottle.sha256 bottle_path.sha256 => bottle_tag
old_spec = f.bottle_specification
if ARGV.include?("--keep-old") && !old_spec.checksums.empty?
bad_fields = [:root_url, :prefix, :cellar, :revision].select do |field|
old_spec.send(field) != bottle.send(field)
end
if bad_fields.any?
bottle_path.unlink if bottle_path.exist?
odie "--keep-old is passed but there are changes in: #{bad_fields.join ", "}"
end
end
output = bottle_output bottle
puts "./#{filename}"
puts output
if ARGV.include? "--rb"
File.open("#{filename.prefix}.bottle.rb", "w") do |file|
file.write("\# #{f.full_name}\n")
file.write(output)
end
end
end
2014-03-11 09:16:14 -05:00
module BottleMerger
def bottle(&block)
instance_eval(&block)
end
end
def merge
write = ARGV.include? "--write"
keep_old = ARGV.include? "--keep-old"
merge_hash = {}
ARGV.named.each do |argument|
bottle_block = IO.read(argument)
formula_name = bottle_block.lines.first.sub(/^# /, "").chomp
merge_hash[formula_name] ||= []
merge_hash[formula_name] << bottle_block
end
merge_hash.each do |formula_name, bottle_blocks|
ohai formula_name
f = Formulary.factory(formula_name)
bottle = if keep_old
f.bottle_specification.dup
else
BottleSpecification.new
end
bottle.extend(BottleMerger)
bottle_blocks.each { |block| bottle.instance_eval(block) }
old_spec = f.bottle_specification
if keep_old && !old_spec.checksums.empty?
bad = [:root_url, :prefix, :cellar, :revision].any? do |field|
old_spec.send(field) != bottle.send(field)
end
if bad
ofail "--keep-old is passed but at least one of fields are not the same, skip it"
next
end
end
output = bottle_output bottle
puts output
if write
2014-01-31 19:07:49 +01:00
update_or_add = nil
2013-12-27 16:43:34 -06:00
2014-06-18 19:23:42 -05:00
Utils::Inreplace.inreplace(f.path) do |s|
if s.include? "bottle do"
update_or_add = "update"
2014-01-31 19:07:49 +01:00
string = s.sub!(/ bottle do.+?end\n/m, output)
odie "Bottle block update failed!" unless string
else
update_or_add = "add"
if s.include? "stable do"
indent = s.slice(/^ +stable do/).length - "stable do".length
string = s.sub!(/^ {#{indent}}stable do(.|\n)+?^ {#{indent}}end\n/m, '\0' + output + "\n")
else
string = s.sub!(
/(
\ {2}( # two spaces at the beginning
url\ ['"][\S\ ]+['"] # url with a string
(
,[\S\ ]*$ # url may have options
(\n^\ {3}[\S\ ]+$)* # options can be in multiple lines
)?|
(homepage|desc|sha1|sha256|head|version|mirror)\ ['"][\S\ ]+['"]| # specs with a string
revision\ \d+ # revision with a number
)\n+ # multiple empty lines
)+
/mx, '\0' + output + "\n")
end
odie "Bottle block addition failed!" unless string
end
end
HOMEBREW_REPOSITORY.cd do
safe_system "git", "commit", "--no-edit", "--verbose",
2014-07-12 22:37:51 -05:00
"--message=#{f.name}: #{update_or_add} #{f.pkg_version} bottle.",
"--", f.path
end
end
end
end
def bottle
if ARGV.include? "--merge"
merge
else
ARGV.resolved_formulae.each do |f|
bottle_formula f
end
end
end
end