538 lines
18 KiB
Ruby
Raw Normal View History

#: * `bottle` [`--verbose`] [`--no-rebuild`|`--keep-old`] [`--skip-relocation`] [`--or-later`] [`--root-url=`<URL>] [`--force-core-tap`] [`--json`] <formulae>:
#: Generate a bottle (binary package) from a formula installed with
#: `--build-bottle`.
#:
#: If the formula specifies a rebuild version, it will be incremented in the
#: generated DSL. Passing `--keep-old` will attempt to keep it at its
#: original value, while `--no-rebuild` will remove it.
#:
2017-04-02 10:14:21 +01:00
#: If `--verbose` (or `-v`) is passed, print the bottling commands and any warnings
#: encountered.
#:
#: If `--skip-relocation` is passed, do not check if the bottle can be marked
#: as relocatable.
#:
#: If `--root-url` is passed, use the specified <URL> as the root of the
#: bottle's URL instead of Homebrew's default.
#:
#: If `--or-later` is passed, append _or_later to the bottle tag.
#:
#: If `--force-core-tap` is passed, build a bottle even if <formula> is not
#: in homebrew/core or any installed taps.
#:
#: If `--json` is passed, write bottle information to a JSON file, which can
#: be used as the argument for `--merge`.
#:
#: * `bottle` `--merge` [`--keep-old`] [`--write` [`--no-commit`]] <bottle_json_files>:
#: Generate a bottle from a `--json` output file and print the new DSL merged
#: into the existing formula.
#:
#: If `--write` is passed, write the changes to the formula file. A new
#: commit will then be generated unless `--no-commit` is passed.
require "formula"
2016-04-25 17:57:51 +01:00
require "utils/bottles"
require "tab"
require "keg"
require "formula_versions"
2018-03-25 12:22:29 +05:30
require "cli_parser"
require "utils/inreplace"
require "erb"
2016-09-11 17:41:51 +01:00
BOTTLE_ERB = <<-EOS.freeze
2013-09-21 21:21:42 +01:00
bottle do
<% if !root_url.start_with?(HOMEBREW_BOTTLE_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 %>
2017-09-24 20:12:58 +01:00
<% if rebuild.positive? %>
2016-08-18 17:32:35 +01:00
rebuild <%= rebuild %>
2013-09-21 21:21:42 +01:00
<% end %>
2013-09-23 17:30:47 +01:00
<% checksums.each do |checksum_type, checksum_values| %>
<% checksum_values.each do |checksum_value| %>
<% checksum, macos = checksum_value.shift %>
<%= checksum_type %> "<%= checksum %>" => :<%= macos %><%= "_or_later" if Homebrew.args.or_later? %>
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
MAXIMUM_STRING_MATCHES = 100
module Homebrew
2016-09-26 01:44:51 +02:00
module_function
2018-03-25 12:22:29 +05:30
def bottle
Homebrew::CLI::Parser.parse do
2018-03-25 12:22:29 +05:30
switch "--merge"
switch "--skip-relocation"
switch "--force-core-tap"
switch "--no-rebuild"
switch "--keep-old"
switch "--write"
switch "--no-commit"
switch "--json"
switch "--or-later"
switch :verbose
switch :debug
2018-03-25 12:22:29 +05:30
flag "--root-url"
end
return merge if args.merge?
ensure_relocation_formulae_installed!
2018-03-25 12:22:29 +05:30
ARGV.resolved_formulae.each do |f|
bottle_formula f
end
end
def ensure_relocation_formulae_installed!
Keg.relocation_formulae.each do |f|
next if Formula[f].installed?
ohai "Installing #{f}..."
safe_system HOMEBREW_BREW_FILE, "install", f
end
end
def keg_contain?(string, keg, ignores)
@put_string_exists_header, @put_filenames = nil
2015-02-26 19:13:10 +00:00
print_filename = lambda do |str, filename|
2015-02-26 19:13:10 +00:00
unless @put_string_exists_header
opoo "String '#{str}' still exists in these files:"
2015-02-26 19:13:10 +00:00
@put_string_exists_header = true
end
2015-02-26 19:13:10 +00:00
@put_filenames ||= []
2016-09-22 20:12:28 +02:00
return if @put_filenames.include?(filename)
2016-09-22 20:12:28 +02:00
2016-08-30 21:38:13 +02:00
puts Formatter.error(filename.to_s)
2016-09-22 20:12:28 +02:00
@put_filenames << filename
end
result = false
keg.each_unique_file_matching(string) do |file|
next if Metafiles::EXTENSIONS.include?(file.extname) # Skip document files.
linked_libraries = Keg.file_linked_libraries(file, string)
result ||= !linked_libraries.empty?
if args.verbose?
print_filename.call(string, file) unless linked_libraries.empty?
linked_libraries.each do |lib|
2016-10-02 08:40:38 +02:00
puts " #{Tty.bold}-->#{Tty.reset} links to #{lib}"
end
end
text_matches = []
# 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
text_matches << [match, offset]
end
end
next unless args.verbose? && !text_matches.empty?
print_filename.call(string, file)
2016-09-11 17:41:51 +01:00
text_matches.first(MAXIMUM_STRING_MATCHES).each do |match, offset|
2016-10-02 08:40:38 +02:00
puts " #{Tty.bold}-->#{Tty.reset} match '#{match}' at offset #{Tty.bold}0x#{offset}#{Tty.reset}"
2016-09-11 17:41:51 +01:00
end
2016-09-11 17:41:51 +01:00
if text_matches.size > MAXIMUM_STRING_MATCHES
puts "Only the first #{MAXIMUM_STRING_MATCHES} matches were output"
end
end
keg_contain_absolute_symlink_starting_with?(string, keg) || result
2016-07-26 21:50:00 -07:00
end
def keg_contain_absolute_symlink_starting_with?(string, keg)
absolute_symlinks_start_with_string = []
keg.find do |pn|
2016-09-11 17:41:51 +01:00
next unless pn.symlink? && (link = pn.readlink).absolute?
2016-09-23 11:01:40 +02:00
absolute_symlinks_start_with_string << pn if link.to_s.start_with?(string)
end
if args.verbose?
2016-07-26 21:50:00 -07:00
unless absolute_symlinks_start_with_string.empty?
opoo "Absolute symlink starting with #{string}:"
absolute_symlinks_start_with_string.each do |pn|
puts " #{pn} -> #{pn.resolved_path}"
end
end
end
!absolute_symlinks_start_with_string.empty?
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 tap = f.tap
unless args.force_core_tap?
return ofail "Formula not from core or any taps: #{f.full_name}"
end
2016-09-22 20:12:28 +02:00
tap = CoreTap.instance
end
2015-09-14 20:06:27 +08:00
if f.bottle_disabled?
ofail "Formula has disabled bottle: #{f.full_name}"
puts f.bottle_disable_reason
return
end
2016-09-11 17:41:51 +01:00
unless Utils::Bottles.built_as? f
2015-05-27 20:44:51 +08:00
return ofail "Formula not installed with '--build-bottle': #{f.full_name}"
end
2016-09-23 11:01:40 +02:00
return ofail "Formula has no stable version: #{f.full_name}" unless f.stable
if args.no_rebuild? || !f.tap
2016-08-18 17:32:35 +01:00
rebuild = 0
elsif args.keep_old?
2016-08-18 17:32:35 +01:00
rebuild = f.bottle_specification.rebuild
else
2016-08-18 17:32:35 +01:00
ohai "Determining #{f.full_name} bottle rebuild..."
versions = FormulaVersions.new(f)
2016-08-18 17:32:35 +01:00
rebuilds = versions.bottle_version_map("origin/master")[f.pkg_version]
2017-09-24 20:12:58 +01:00
rebuilds.pop if rebuilds.last.to_i.positive?
2016-08-18 17:32:35 +01:00
rebuild = rebuilds.empty? ? 0 : rebuilds.max.to_i + 1
end
2016-08-18 17:32:35 +01:00
filename = Bottle::Filename.create(f, Utils::Bottles.tag, rebuild)
bottle_path = Pathname.pwd/filename
tar_filename = filename.to_s.sub(/.gz$/, "")
tar_path = Pathname.pwd/tar_filename
prefix = HOMEBREW_PREFIX.to_s
repository = HOMEBREW_REPOSITORY.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
skip_relocation = false
keg.lock do
original_tab = nil
changed_files = nil
begin
keg.delete_pyc_files!
unless args.skip_relocation?
changed_files = keg.replace_locations_with_placeholders
end
Tab.clear_cache
tab = Tab.for_keg(keg)
original_tab = tab.dup
tab.poured_from_bottle = false
tab.HEAD = nil
tab.time = nil
tab.changed_files = changed_files
tab.write
keg.find do |file|
if file.symlink?
# Ruby does not support `File.lutime` yet.
# Shellout using `touch` to change modified time of symlink itself.
system "/usr/bin/touch", "-h",
"-t", tab.source_modified_time.strftime("%Y%m%d%H%M.%S"), file
else
file.utime(tab.source_modified_time, tab.source_modified_time)
end
end
cd cellar do
safe_system "tar", "cf", tar_path, "#{f.name}/#{f.pkg_version}"
tar_path.utime(tab.source_modified_time, tab.source_modified_time)
relocatable_tar_path = "#{f}-bottle.tar"
mv tar_path, relocatable_tar_path
# Use gzip, faster to compress than bzip2, faster to uncompress than bzip2
# or an uncompressed tarball (and more bandwidth friendly).
safe_system "gzip", "-f", relocatable_tar_path
mv "#{relocatable_tar_path}.gz", bottle_path
end
2017-06-01 16:06:51 +02: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 = []
any_go_deps = f.deps.any? do |dep|
dep.name =~ Version.formula_optionally_versioned_regex(:go)
end
if any_go_deps
go_regex =
Version.formula_optionally_versioned_regex(:go, full: false)
ignores << %r{#{Regexp.escape(HOMEBREW_CELLAR)}/#{go_regex}/[\d\.]+/libexec}
end
relocatable = true
if args.skip_relocation?
skip_relocation = true
else
relocatable = false if keg_contain?(prefix_check, keg, ignores)
relocatable = false if keg_contain?(repository, keg, ignores)
relocatable = false if keg_contain?(cellar, keg, ignores)
if prefix != prefix_check
relocatable = false if keg_contain_absolute_symlink_starting_with?(prefix, keg)
relocatable = false if keg_contain?("#{prefix}/etc", keg, ignores)
relocatable = false if keg_contain?("#{prefix}/var", keg, ignores)
end
skip_relocation = relocatable && !keg.require_relocation?
end
puts if !relocatable && args.verbose?
rescue Interrupt
ignore_interrupts { bottle_path.unlink if bottle_path.exist? }
raise
ensure
ignore_interrupts do
2017-09-24 19:24:46 +01:00
original_tab&.write
unless args.skip_relocation?
keg.replace_placeholders_with_locations changed_files
end
end
end
end
root_url = args.root_url
bottle = BottleSpecification.new
bottle.tap = tap
bottle.root_url(root_url) if root_url
if relocatable
if skip_relocation
bottle.cellar :any_skip_relocation
else
bottle.cellar :any
end
else
bottle.cellar cellar
bottle.prefix prefix
end
2016-08-18 17:32:35 +01:00
bottle.rebuild rebuild
sha256 = bottle_path.sha256
bottle.sha256 sha256 => Utils::Bottles.tag
old_spec = f.bottle_specification
if args.keep_old? && !old_spec.checksums.empty?
2017-05-29 18:24:52 +01:00
mismatches = [:root_url, :prefix, :cellar, :rebuild].reject do |key|
old_spec.send(key) == bottle.send(key)
end
mismatches.delete(:cellar) if old_spec.cellar == :any && bottle.cellar == :any_skip_relocation
unless mismatches.empty?
bottle_path.unlink if bottle_path.exist?
mismatches.map! do |key|
old_value = old_spec.send(key).inspect
value = bottle.send(key).inspect
"#{key}: old: #{old_value}, new: #{value}"
end
2017-10-15 02:28:32 +02:00
odie <<~EOS
--keep-old was passed but there are changes in:
#{mismatches.join("\n")}
EOS
end
end
output = bottle_output bottle
puts "./#{filename}"
puts output
return unless args.json?
tag = Utils::Bottles.tag.to_s
tag += "_or_later" if args.or_later?
2016-09-22 20:12:28 +02:00
json = {
f.full_name => {
"formula" => {
"pkg_version" => f.pkg_version.to_s,
"path" => f.path.to_s.strip_prefix("#{HOMEBREW_REPOSITORY}/"),
},
"bottle" => {
"root_url" => bottle.root_url,
"prefix" => bottle.prefix,
"cellar" => bottle.cellar.to_s,
"rebuild" => bottle.rebuild,
"tags" => {
tag => {
2018-08-06 15:02:52 +02:00
"filename" => filename.bintray,
2016-09-22 20:12:28 +02:00
"sha256" => sha256,
2016-09-11 17:41:51 +01:00
},
},
},
2016-09-22 20:12:28 +02:00
"bintray" => {
"package" => Utils::Bottles::Bintray.package(f.name),
"repository" => Utils::Bottles::Bintray.repository(tap),
},
},
}
2018-08-06 15:02:52 +02:00
File.open(filename.json, "w") do |file|
file.write JSON.generate json
end
end
def merge
write = args.write?
bottles_hash = ARGV.named.reduce({}) do |hash, json_file|
deep_merge_hashes hash, JSON.parse(IO.read(json_file))
end
bottles_hash.each do |formula_name, bottle_hash|
ohai formula_name
bottle = BottleSpecification.new
bottle.root_url bottle_hash["bottle"]["root_url"]
cellar = bottle_hash["bottle"]["cellar"]
2017-05-29 18:24:52 +01:00
cellar = cellar.to_sym if ["any", "any_skip_relocation"].include?(cellar)
bottle.cellar cellar
bottle.prefix bottle_hash["bottle"]["prefix"]
2016-08-18 17:32:35 +01:00
bottle.rebuild bottle_hash["bottle"]["rebuild"]
bottle_hash["bottle"]["tags"].each do |tag, tag_hash|
bottle.sha256 tag_hash["sha256"] => tag.to_sym
end
output = bottle_output bottle
if write
2016-09-11 17:41:51 +01:00
path = Pathname.new((HOMEBREW_REPOSITORY/bottle_hash["formula"]["path"]).to_s)
2014-01-31 19:07:49 +01:00
update_or_add = nil
2013-12-27 16:43:34 -06:00
Utils::Inreplace.inreplace(path) do |s|
if s.include? "bottle do"
update_or_add = "update"
if args.keep_old?
mismatches = []
2016-05-28 16:46:34 +01:00
bottle_block_contents = s[/ bottle do(.+?)end\n/m, 1]
bottle_block_contents.lines.each do |line|
line = line.strip
next if line.empty?
key, old_value_original, _, tag = line.split " ", 4
2016-08-18 17:32:35 +01:00
valid_key = %w[root_url prefix cellar rebuild sha1 sha256].include? key
2016-05-28 16:46:34 +01:00
next unless valid_key
old_value = old_value_original.to_s.delete ":'\""
tag = tag.to_s.delete ":"
2016-09-11 17:41:51 +01:00
unless tag.empty?
if !bottle_hash["bottle"]["tags"][tag].to_s.empty?
mismatches << "#{key} => #{tag}"
else
bottle.send(key, old_value => tag.to_sym)
end
next
end
value_original = bottle_hash["bottle"][key]
value = value_original.to_s
next if key == "cellar" && old_value == "any" && value == "any_skip_relocation"
2016-09-11 17:41:51 +01:00
next unless old_value.empty? || value != old_value
old_value = old_value_original.inspect
value = value_original.inspect
mismatches << "#{key}: old: #{old_value}, new: #{value}"
end
unless mismatches.empty?
2017-10-15 02:28:32 +02:00
odie <<~EOS
--keep-old was passed but there are changes in:
#{mismatches.join("\n")}
EOS
end
output = bottle_output bottle
end
puts output
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
if args.keep_old?
2016-05-28 16:46:34 +01:00
odie "--keep-old was passed but there was no existing bottle block!"
end
puts output
update_or_add = "add"
if s.include? "stable do"
2016-09-21 08:49:04 +02:00
indent = s.slice(/^( +)stable do/, 1).length
string = s.sub!(/^ {#{indent}}stable do(.|\n)+?^ {#{indent}}end\n/m, '\0' + output + "\n")
else
2018-06-06 23:34:19 -04:00
pattern = /(
(\ {2}\#[^\n]*\n)* # comments
\ {2}( # two spaces at the beginning
(url|head)\ ['"][\S\ ]+['"] # url or head with a string
(
,[\S\ ]*$ # url may have options
(\n^\ {3}[\S\ ]+$)* # options can be in multiple lines
)?|
(homepage|desc|sha1|sha256|version|mirror)\ ['"][\S\ ]+['"]| # specs with a string
revision\ \d+ # revision with a number
)\n+ # multiple empty lines
)+
2018-06-06 23:34:19 -04:00
/mx
string = s.sub!(pattern, '\0' + output + "\n")
end
odie "Bottle block addition failed!" unless string
end
end
unless args.no_commit?
if ENV["HOMEBREW_GIT_NAME"]
ENV["GIT_AUTHOR_NAME"] =
ENV["GIT_COMMITTER_NAME"] =
ENV["HOMEBREW_GIT_NAME"]
end
if ENV["HOMEBREW_GIT_EMAIL"]
ENV["GIT_AUTHOR_EMAIL"] =
ENV["GIT_COMMITTER_EMAIL"] =
ENV["HOMEBREW_GIT_EMAIL"]
end
short_name = formula_name.split("/", -1).last
pkg_version = bottle_hash["formula"]["pkg_version"]
path.parent.cd do
safe_system "git", "commit", "--no-edit", "--verbose",
"--message=#{short_name}: #{update_or_add} #{pkg_version} bottle.",
"--", path
end
end
else
puts output
end
end
end
end