2019-11-04 21:00:20 +11:00
|
|
|
# frozen_string_literal: true
|
2019-04-19 15:38:03 +09:00
|
|
|
|
2015-08-03 13:09:07 +01:00
|
|
|
require "formula"
|
2016-04-25 17:57:51 +01:00
|
|
|
require "utils/bottles"
|
2015-08-03 13:09:07 +01:00
|
|
|
require "tab"
|
|
|
|
require "keg"
|
|
|
|
require "formula_versions"
|
2019-04-17 18:25:08 +09:00
|
|
|
require "cli/parser"
|
2015-08-03 13:09:07 +01:00
|
|
|
require "utils/inreplace"
|
|
|
|
require "erb"
|
2012-03-07 17:29:05 -05:00
|
|
|
|
2019-11-04 21:00:20 +11:00
|
|
|
BOTTLE_ERB = <<-EOS
|
2013-09-21 21:21:42 +01:00
|
|
|
bottle do
|
2020-01-26 22:01:02 -08:00
|
|
|
<% if root_url != "#{HOMEBREW_BOTTLE_DEFAULT_DOMAIN}/bottles" %>
|
2014-05-09 17:38:12 +09:00
|
|
|
root_url "<%= root_url %>"
|
|
|
|
<% end %>
|
2019-09-30 16:09:03 +02:00
|
|
|
<% if ![HOMEBREW_DEFAULT_PREFIX, LINUXBREW_DEFAULT_PREFIX].include?(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 %>
|
2018-09-11 14:16:27 -07:00
|
|
|
<% elsif ![Homebrew::DEFAULT_CELLAR, "/usr/local/Cellar"].include?(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| %>
|
2016-09-18 19:57:19 +01:00
|
|
|
<% checksum, macos = checksum_value.shift %>
|
2019-11-06 14:31:03 +00:00
|
|
|
<%= checksum_type %> "<%= checksum %>" => :<%= macos %>
|
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
|
|
|
|
|
2015-09-23 14:55:22 +01:00
|
|
|
MAXIMUM_STRING_MATCHES = 100
|
|
|
|
|
2014-06-18 22:41:47 -05:00
|
|
|
module Homebrew
|
2016-09-26 01:44:51 +02:00
|
|
|
module_function
|
|
|
|
|
2018-07-30 18:25:38 +05:30
|
|
|
def bottle_args
|
|
|
|
Homebrew::CLI::Parser.new do
|
2019-11-04 21:00:20 +11:00
|
|
|
usage_banner <<~EOS
|
2019-01-30 21:33:03 +00:00
|
|
|
`bottle` [<options>] <formula>
|
2018-09-28 21:39:52 +05:30
|
|
|
|
2018-10-08 22:49:03 -04:00
|
|
|
Generate a bottle (binary package) from a formula that was installed with
|
2018-09-28 21:39:52 +05:30
|
|
|
`--build-bottle`.
|
|
|
|
If the formula specifies a rebuild version, it will be incremented in the
|
2018-10-08 22:49:03 -04:00
|
|
|
generated DSL. Passing `--keep-old` will attempt to keep it at its original
|
|
|
|
value, while `--no-rebuild` will remove it.
|
2018-09-28 21:39:52 +05:30
|
|
|
EOS
|
|
|
|
switch "--skip-relocation",
|
2019-04-30 08:44:35 +01:00
|
|
|
description: "Do not check if the bottle can be marked as relocatable."
|
2018-09-28 21:39:52 +05:30
|
|
|
switch "--force-core-tap",
|
2019-04-30 08:44:35 +01:00
|
|
|
description: "Build a bottle even if <formula> is not in `homebrew/core` or any installed taps."
|
2018-09-28 21:39:52 +05:30
|
|
|
switch "--no-rebuild",
|
2019-04-30 08:44:35 +01:00
|
|
|
description: "If the formula specifies a rebuild version, remove it from the generated DSL."
|
2018-09-28 21:39:52 +05:30
|
|
|
switch "--keep-old",
|
2019-04-30 08:44:35 +01:00
|
|
|
description: "If the formula specifies a rebuild version, attempt to preserve its value in the "\
|
|
|
|
"generated DSL."
|
2018-10-08 22:49:03 -04:00
|
|
|
switch "--json",
|
2019-08-19 22:44:50 -04:00
|
|
|
description: "Write bottle information to a JSON file, which can be used as the value for "\
|
2019-04-30 08:44:35 +01:00
|
|
|
"`--merge`."
|
2018-09-28 21:39:52 +05:30
|
|
|
switch "--merge",
|
2019-04-30 08:44:35 +01:00
|
|
|
description: "Generate an updated bottle block for a formula and optionally merge it into the "\
|
2019-08-19 22:44:50 -04:00
|
|
|
"formula file. Instead of a formula name, requires the path to a JSON file generated with "\
|
2019-04-30 08:44:35 +01:00
|
|
|
"`brew bottle --json` <formula>."
|
2018-09-28 21:39:52 +05:30
|
|
|
switch "--write",
|
2019-04-30 08:44:35 +01:00
|
|
|
depends_on: "--merge",
|
2019-08-20 00:04:14 -04:00
|
|
|
description: "Write changes to the formula file. A new commit will be generated unless "\
|
2019-04-30 08:44:35 +01:00
|
|
|
"`--no-commit` is passed."
|
2018-09-28 21:39:52 +05:30
|
|
|
switch "--no-commit",
|
2019-04-30 08:44:35 +01:00
|
|
|
depends_on: "--write",
|
|
|
|
description: "When passed with `--write`, a new commit will not generated after writing changes "\
|
|
|
|
"to the formula file."
|
2018-10-29 14:33:25 -07:00
|
|
|
flag "--root-url=",
|
2019-04-30 08:44:35 +01:00
|
|
|
description: "Use the specified <URL> as the root of the bottle's URL instead of Homebrew's default."
|
2020-07-30 18:40:10 +02:00
|
|
|
|
2018-10-08 22:49:03 -04:00
|
|
|
conflicts "--no-rebuild", "--keep-old"
|
2020-03-04 17:28:24 +00:00
|
|
|
min_named 1
|
2018-03-25 12:22:29 +05:30
|
|
|
end
|
2018-07-30 18:25:38 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def bottle
|
2020-07-30 18:40:10 +02:00
|
|
|
args = bottle_args.parse
|
2018-03-25 12:22:29 +05:30
|
|
|
|
2020-07-31 18:58:07 +02:00
|
|
|
return merge(args: args) if args.merge?
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2019-10-16 14:29:53 +08:00
|
|
|
ensure_relocation_formulae_installed! unless args.skip_relocation?
|
2020-08-19 10:34:48 -04:00
|
|
|
args.named.to_resolved_formulae.each do |f|
|
2020-07-31 18:58:07 +02:00
|
|
|
bottle_formula f, args: args
|
2018-03-25 12:22:29 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-06-22 16:56:37 -07:00
|
|
|
def ensure_relocation_formulae_installed!
|
|
|
|
Keg.relocation_formulae.each do |f|
|
2019-12-03 11:42:09 +00:00
|
|
|
next if Formula[f].latest_version_installed?
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2018-06-22 16:56:37 -07:00
|
|
|
ohai "Installing #{f}..."
|
|
|
|
safe_system HOMEBREW_BREW_FILE, "install", f
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-07-31 18:58:07 +02:00
|
|
|
def keg_contain?(string, keg, ignores, formula_and_runtime_deps_names = nil, args:)
|
2015-12-26 23:00:38 +01:00
|
|
|
@put_string_exists_header, @put_filenames = nil
|
2015-02-26 19:13:10 +00:00
|
|
|
|
2016-09-27 19:11:56 +02:00
|
|
|
print_filename = lambda do |str, filename|
|
2015-02-26 19:13:10 +00:00
|
|
|
unless @put_string_exists_header
|
2016-09-27 19:11:56 +02:00
|
|
|
opoo "String '#{str}' still exists in these files:"
|
2015-02-26 19:13:10 +00:00
|
|
|
@put_string_exists_header = true
|
|
|
|
end
|
2013-10-31 01:20:28 -07:00
|
|
|
|
2015-02-26 19:13:10 +00:00
|
|
|
@put_filenames ||= []
|
2016-09-22 20:12:28 +02:00
|
|
|
|
2017-10-07 00:31: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
|
2015-02-20 14:29:43 +00:00
|
|
|
end
|
|
|
|
|
2013-12-17 20:46:42 -06:00
|
|
|
result = false
|
2013-10-31 01:20:28 -07:00
|
|
|
|
2013-12-17 20:46:42 -06:00
|
|
|
keg.each_unique_file_matching(string) do |file|
|
2017-10-07 00:31:28 +02:00
|
|
|
next if Metafiles::EXTENSIONS.include?(file.extname) # Skip document files.
|
2015-07-10 19:51:43 +08:00
|
|
|
|
2016-07-09 13:52:05 +01:00
|
|
|
linked_libraries = Keg.file_linked_libraries(file, string)
|
2016-08-05 22:01:32 +08:00
|
|
|
result ||= !linked_libraries.empty?
|
2013-10-31 01:20:28 -07:00
|
|
|
|
2018-05-05 18:40:01 +05:30
|
|
|
if args.verbose?
|
2016-09-27 19:11:56 +02:00
|
|
|
print_filename.call(string, file) unless linked_libraries.empty?
|
2014-01-20 15:26:18 -08:00
|
|
|
linked_libraries.each do |lib|
|
2016-10-02 08:40:38 +02:00
|
|
|
puts " #{Tty.bold}-->#{Tty.reset} links to #{lib}"
|
2014-01-20 15:26:18 -08:00
|
|
|
end
|
2013-10-31 01:20:28 -07:00
|
|
|
end
|
|
|
|
|
2015-09-23 14:55:22 +01:00
|
|
|
text_matches = []
|
|
|
|
|
2013-10-31 01:20:28 -07:00
|
|
|
# Use strings to search through the file for each string
|
2014-07-05 13:50:54 -05:00
|
|
|
Utils.popen_read("strings", "-t", "x", "-", file.to_s) do |io|
|
2013-12-14 15:43:15 -06:00
|
|
|
until io.eof?
|
|
|
|
str = io.readline.chomp
|
2015-08-03 13:09:07 +01:00
|
|
|
next if ignores.any? { |i| i =~ str }
|
2013-12-14 15:43:15 -06:00
|
|
|
next unless str.include? string
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2013-12-14 15:43:15 -06:00
|
|
|
offset, match = str.split(" ", 2)
|
|
|
|
next if linked_libraries.include? match # Don't bother reporting a string if it was found by otool
|
2015-09-11 16:49:39 +08:00
|
|
|
|
2018-12-02 11:00:28 -08:00
|
|
|
# Do not report matches to files that do not exist.
|
|
|
|
next unless File.exist? match
|
|
|
|
|
|
|
|
# Do not report matches to build dependencies.
|
|
|
|
if formula_and_runtime_deps_names.present?
|
|
|
|
begin
|
|
|
|
keg_name = Keg.for(Pathname.new(match)).name
|
|
|
|
next unless formula_and_runtime_deps_names.include? keg_name
|
|
|
|
rescue NotAKegError
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-09-11 16:49:39 +08:00
|
|
|
result = true
|
2015-09-23 14:55:22 +01:00
|
|
|
text_matches << [match, offset]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-05-05 18:40:01 +05:30
|
|
|
next unless args.verbose? && !text_matches.empty?
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2016-09-27 19:11:56 +02:00
|
|
|
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
|
2015-02-20 14:29:43 +00:00
|
|
|
|
2016-09-11 17:41:51 +01:00
|
|
|
if text_matches.size > MAXIMUM_STRING_MATCHES
|
2019-11-29 14:53:01 -05:00
|
|
|
puts "Only the first #{MAXIMUM_STRING_MATCHES} matches were output."
|
2013-10-31 01:20:28 -07:00
|
|
|
end
|
|
|
|
end
|
2013-12-17 20:46:42 -06:00
|
|
|
|
2020-08-01 14:56:11 -04:00
|
|
|
keg_contain_absolute_symlink_starting_with?(string, keg, args: args) || result
|
2016-07-26 21:50:00 -07:00
|
|
|
end
|
|
|
|
|
2020-08-01 14:56:11 -04:00
|
|
|
def keg_contain_absolute_symlink_starting_with?(string, keg, args:)
|
2015-09-11 16:49:39 +08:00
|
|
|
absolute_symlinks_start_with_string = []
|
2014-03-27 17:05:17 -05:00
|
|
|
keg.find do |pn|
|
2016-09-11 17:41:51 +01:00
|
|
|
next unless pn.symlink? && (link = pn.readlink).absolute?
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2016-09-23 11:01:40 +02:00
|
|
|
absolute_symlinks_start_with_string << pn if link.to_s.start_with?(string)
|
2014-03-18 19:03:24 -05:00
|
|
|
end
|
|
|
|
|
2020-09-01 14:05:52 +01:00
|
|
|
if args.verbose? && absolute_symlinks_start_with_string.present?
|
|
|
|
opoo "Absolute symlink starting with #{string}:"
|
|
|
|
absolute_symlinks_start_with_string.each do |pn|
|
|
|
|
puts " #{pn} -> #{pn.resolved_path}"
|
2015-09-11 16:49:39 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-07-27 08:54:56 -07:00
|
|
|
!absolute_symlinks_start_with_string.empty?
|
2013-03-11 18:56:26 +00:00
|
|
|
end
|
|
|
|
|
2015-08-03 13:09:07 +01:00
|
|
|
def bottle_output(bottle)
|
2013-09-21 21:21:42 +01:00
|
|
|
erb = ERB.new BOTTLE_ERB
|
2015-08-03 13:09:07 +01:00
|
|
|
erb.result(bottle.instance_eval { binding }).gsub(/^\s*$\n/, "")
|
2013-06-04 20:39:09 +01:00
|
|
|
end
|
|
|
|
|
2020-01-14 15:09:13 +00:00
|
|
|
def sudo_purge
|
|
|
|
return unless ENV["HOMEBREW_BOTTLE_SUDO_PURGE"]
|
|
|
|
|
2020-01-15 10:48:13 +00:00
|
|
|
system "/usr/bin/sudo", "--non-interactive", "/usr/sbin/purge"
|
2020-01-14 15:09:13 +00:00
|
|
|
end
|
|
|
|
|
2020-07-31 18:58:07 +02:00
|
|
|
def bottle_formula(f, args:)
|
2019-12-03 11:42:09 +00:00
|
|
|
return ofail "Formula not installed or up-to-date: #{f.full_name}" unless f.latest_version_installed?
|
2012-03-15 10:57:34 +13:00
|
|
|
|
2017-10-07 00:31:28 +02:00
|
|
|
unless tap = f.tap
|
2019-02-19 13:11:32 +00:00
|
|
|
return ofail "Formula not from core or any installed taps: #{f.full_name}" unless args.force_core_tap?
|
2016-09-22 20:12:28 +02:00
|
|
|
|
|
|
|
tap = CoreTap.instance
|
2015-12-26 13:15:29 +08:00
|
|
|
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
|
|
|
|
|
2019-12-13 15:39:55 -05:00
|
|
|
return ofail "Formula was not installed with --build-bottle: #{f.full_name}" unless Utils::Bottles.built_as? f
|
2012-03-07 17:29:05 -05:00
|
|
|
|
2016-09-23 11:01:40 +02:00
|
|
|
return ofail "Formula has no stable version: #{f.full_name}" unless f.stable
|
2014-02-15 11:28:48 +00:00
|
|
|
|
2018-05-05 18:40:01 +05:30
|
|
|
if args.no_rebuild? || !f.tap
|
2016-08-18 17:32:35 +01:00
|
|
|
rebuild = 0
|
2018-05-05 18:40:01 +05:30
|
|
|
elsif args.keep_old?
|
2016-08-18 17:32:35 +01:00
|
|
|
rebuild = f.bottle_specification.rebuild
|
2015-05-28 23:57:02 -04:00
|
|
|
else
|
2016-08-18 17:32:35 +01:00
|
|
|
ohai "Determining #{f.full_name} bottle rebuild..."
|
2014-05-28 16:23:33 -05:00
|
|
|
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
|
2013-12-10 15:16:22 -06:00
|
|
|
end
|
|
|
|
|
2016-08-18 17:32:35 +01:00
|
|
|
filename = Bottle::Filename.create(f, Utils::Bottles.tag, rebuild)
|
2013-02-09 19:06:54 -08:00
|
|
|
bottle_path = Pathname.pwd/filename
|
2012-03-07 17:29:05 -05:00
|
|
|
|
2015-12-15 14:21:27 +00:00
|
|
|
tar_filename = filename.to_s.sub(/.gz$/, "")
|
|
|
|
tar_path = Pathname.pwd/tar_filename
|
|
|
|
|
2013-03-11 18:56:26 +00:00
|
|
|
prefix = HOMEBREW_PREFIX.to_s
|
2016-09-18 16:31:58 +01:00
|
|
|
repository = HOMEBREW_REPOSITORY.to_s
|
2013-03-11 18:56:26 +00:00
|
|
|
cellar = HOMEBREW_CELLAR.to_s
|
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
ohai "Bottling #{filename}..."
|
2013-09-21 21:21:42 +01:00
|
|
|
|
2018-12-02 11:00:28 -08:00
|
|
|
formula_and_runtime_deps_names = [f.name] + f.runtime_dependencies.map(&:name)
|
2013-12-12 19:46:37 -06:00
|
|
|
keg = Keg.new(f.prefix)
|
|
|
|
relocatable = false
|
2015-09-11 19:13:52 +08:00
|
|
|
skip_relocation = false
|
2013-12-04 22:37:57 -06:00
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
keg.lock do
|
2015-12-15 14:21:27 +00:00
|
|
|
original_tab = nil
|
2016-10-09 19:43:55 -04:00
|
|
|
changed_files = nil
|
2015-12-15 14:21:27 +00:00
|
|
|
|
2013-12-12 19:46:37 -06:00
|
|
|
begin
|
2016-10-25 01:44:40 -04:00
|
|
|
keg.delete_pyc_files!
|
|
|
|
|
2019-02-19 13:11:32 +00:00
|
|
|
changed_files = keg.replace_locations_with_placeholders unless args.skip_relocation?
|
2015-09-11 19:13:52 +08:00
|
|
|
|
2019-11-05 20:34:06 +00:00
|
|
|
Formula.clear_cache
|
2019-11-05 20:33:32 +00:00
|
|
|
Keg.clear_cache
|
2016-01-26 15:52:45 +08:00
|
|
|
Tab.clear_cache
|
2015-12-15 14:21:27 +00:00
|
|
|
tab = Tab.for_keg(keg)
|
|
|
|
original_tab = tab.dup
|
2015-12-16 02:12:50 +01:00
|
|
|
tab.poured_from_bottle = false
|
|
|
|
tab.HEAD = nil
|
|
|
|
tab.time = nil
|
2016-10-09 19:43:55 -04:00
|
|
|
tab.changed_files = changed_files
|
2015-12-15 14:21:27 +00:00
|
|
|
tab.write
|
|
|
|
|
2016-02-10 19:29:40 +08:00
|
|
|
keg.find do |file|
|
|
|
|
if file.symlink?
|
2020-09-06 19:43:30 -04:00
|
|
|
File.lutime(tab.source_modified_time, tab.source_modified_time, file)
|
2016-02-10 19:29:40 +08:00
|
|
|
else
|
|
|
|
file.utime(tab.source_modified_time, tab.source_modified_time)
|
|
|
|
end
|
|
|
|
end
|
2015-12-15 14:21:27 +00:00
|
|
|
|
2014-07-17 14:46:05 -05:00
|
|
|
cd cellar do
|
2020-01-14 15:09:13 +00:00
|
|
|
sudo_purge
|
2015-12-15 14:21:27 +00:00
|
|
|
safe_system "tar", "cf", tar_path, "#{f.name}/#{f.pkg_version}"
|
2020-01-14 15:09:13 +00:00
|
|
|
sudo_purge
|
2016-02-10 19:29:40 +08:00
|
|
|
tar_path.utime(tab.source_modified_time, tab.source_modified_time)
|
2015-12-15 14:21:27 +00:00
|
|
|
relocatable_tar_path = "#{f}-bottle.tar"
|
|
|
|
mv tar_path, relocatable_tar_path
|
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).
|
2015-12-15 14:21:27 +00:00
|
|
|
safe_system "gzip", "-f", relocatable_tar_path
|
2020-01-14 15:09:13 +00:00
|
|
|
sudo_purge
|
2015-12-15 14:21:27 +00:00
|
|
|
mv "#{relocatable_tar_path}.gz", bottle_path
|
2013-12-12 19:46:37 -06:00
|
|
|
end
|
|
|
|
|
2019-02-19 13:11:32 +00:00
|
|
|
ohai "Detecting if #{filename} is relocatable..." if bottle_path.size > 1 * 1024 * 1024
|
2013-12-04 22:37:57 -06:00
|
|
|
|
2020-03-13 21:15:06 +00:00
|
|
|
prefix_check = if Homebrew.default_prefix?(prefix)
|
|
|
|
File.join(prefix, "opt")
|
2013-12-12 19:46:37 -06:00
|
|
|
else
|
2020-03-13 21:15:06 +00:00
|
|
|
prefix
|
2013-12-12 19:46:37 -06:00
|
|
|
end
|
|
|
|
|
2018-12-02 18:27:03 -08:00
|
|
|
# Ignore matches to source code, which is not required at run time.
|
|
|
|
# These matches may be caused by debugging symbols.
|
|
|
|
ignores = [%r{/include/|\.(c|cc|cpp|h|hpp)$}]
|
2018-06-18 14:36:51 +01:00
|
|
|
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)
|
2020-06-02 09:49:23 +01:00
|
|
|
ignores << %r{#{Regexp.escape(HOMEBREW_CELLAR)}/#{go_regex}/[\d.]+/libexec}
|
2015-02-20 14:29:43 +00:00
|
|
|
end
|
|
|
|
|
2016-07-27 08:54:56 -07:00
|
|
|
relocatable = true
|
2018-05-05 18:40:01 +05:30
|
|
|
if args.skip_relocation?
|
2016-06-28 15:33:22 +08:00
|
|
|
skip_relocation = true
|
|
|
|
else
|
2020-07-31 18:58:07 +02:00
|
|
|
relocatable = false if keg_contain?(prefix_check, keg, ignores, formula_and_runtime_deps_names, args: args)
|
|
|
|
relocatable = false if keg_contain?(repository, keg, ignores, args: args)
|
|
|
|
relocatable = false if keg_contain?(cellar, keg, ignores, formula_and_runtime_deps_names, args: args)
|
2016-07-27 08:54:56 -07:00
|
|
|
if prefix != prefix_check
|
2020-08-01 14:56:11 -04:00
|
|
|
relocatable = false if keg_contain_absolute_symlink_starting_with?(prefix, keg, args: args)
|
2020-07-31 18:58:07 +02:00
|
|
|
relocatable = false if keg_contain?("#{prefix}/etc", keg, ignores, args: args)
|
|
|
|
relocatable = false if keg_contain?("#{prefix}/var", keg, ignores, args: args)
|
|
|
|
relocatable = false if keg_contain?("#{prefix}/share/vim", keg, ignores, args: args)
|
2016-07-27 08:54:56 -07:00
|
|
|
end
|
2016-09-22 14:36:24 -04:00
|
|
|
skip_relocation = relocatable && !keg.require_relocation?
|
2016-06-28 15:33:22 +08:00
|
|
|
end
|
2018-05-05 18:40:01 +05:30
|
|
|
puts if !relocatable && args.verbose?
|
2013-12-12 19:46:37 -06:00
|
|
|
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
|
2019-02-19 13:11:32 +00:00
|
|
|
keg.replace_placeholders_with_locations changed_files unless args.skip_relocation?
|
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
|
|
|
|
2018-05-05 18:40:01 +05:30
|
|
|
root_url = args.root_url
|
2014-05-09 17:38:12 +09:00
|
|
|
|
2014-03-10 14:56:02 -05:00
|
|
|
bottle = BottleSpecification.new
|
2016-09-13 08:57:55 +01:00
|
|
|
bottle.tap = tap
|
2014-05-09 17:38:12 +09:00
|
|
|
bottle.root_url(root_url) if root_url
|
2015-09-11 19:13:52 +08:00
|
|
|
if relocatable
|
|
|
|
if skip_relocation
|
|
|
|
bottle.cellar :any_skip_relocation
|
|
|
|
else
|
|
|
|
bottle.cellar :any
|
|
|
|
end
|
|
|
|
else
|
|
|
|
bottle.cellar cellar
|
2015-09-11 19:15:22 +08:00
|
|
|
bottle.prefix prefix
|
2015-09-11 19:13:52 +08:00
|
|
|
end
|
2016-08-18 17:32:35 +01:00
|
|
|
bottle.rebuild rebuild
|
2016-05-28 15:54:05 +01:00
|
|
|
sha256 = bottle_path.sha256
|
2016-09-04 13:22:08 +01:00
|
|
|
bottle.sha256 sha256 => Utils::Bottles.tag
|
2013-03-11 18:56:26 +00:00
|
|
|
|
2016-09-04 13:22:08 +01:00
|
|
|
old_spec = f.bottle_specification
|
2018-05-05 18:40:01 +05:30
|
|
|
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)
|
2015-09-10 19:47:22 +08:00
|
|
|
end
|
2019-10-15 20:13:04 +02:00
|
|
|
if (old_spec.cellar == :any && bottle.cellar == :any_skip_relocation) ||
|
|
|
|
(old_spec.cellar == cellar &&
|
|
|
|
[:any, :any_skip_relocation].include?(bottle.cellar))
|
2018-12-01 10:06:12 +01:00
|
|
|
mismatches.delete(:cellar)
|
2019-10-21 14:03:50 +02:00
|
|
|
bottle.cellar old_spec.cellar
|
2018-12-01 10:06:12 +01:00
|
|
|
end
|
2016-09-10 16:11:01 +01:00
|
|
|
unless mismatches.empty?
|
2015-09-10 19:47:22 +08:00
|
|
|
bottle_path.unlink if bottle_path.exist?
|
2016-09-10 10:13:33 +01:00
|
|
|
|
2016-09-10 19:43:49 +01:00
|
|
|
mismatches.map! do |key|
|
|
|
|
old_value = old_spec.send(key).inspect
|
|
|
|
value = bottle.send(key).inspect
|
|
|
|
"#{key}: old: #{old_value}, new: #{value}"
|
2016-09-10 10:13:33 +01:00
|
|
|
end
|
|
|
|
|
2019-11-04 21:00:20 +11:00
|
|
|
odie <<~EOS
|
2016-09-10 10:13:33 +01:00
|
|
|
--keep-old was passed but there are changes in:
|
2016-09-10 16:11:01 +01:00
|
|
|
#{mismatches.join("\n")}
|
2016-09-10 10:13:33 +01:00
|
|
|
EOS
|
2015-09-10 19:47:22 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
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
|
|
|
|
2018-05-05 18:40:01 +05:30
|
|
|
return unless args.json?
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2016-09-22 20:12:28 +02:00
|
|
|
json = {
|
|
|
|
f.full_name => {
|
|
|
|
"formula" => {
|
|
|
|
"pkg_version" => f.pkg_version.to_s,
|
2018-11-02 17:18:07 +00:00
|
|
|
"path" => f.path.to_s.delete_prefix("#{HOMEBREW_REPOSITORY}/"),
|
2016-09-22 20:12:28 +02:00
|
|
|
},
|
2018-11-02 17:18:07 +00:00
|
|
|
"bottle" => {
|
2016-09-22 20:12:28 +02:00
|
|
|
"root_url" => bottle.root_url,
|
2018-11-02 17:18:07 +00:00
|
|
|
"prefix" => bottle.prefix,
|
|
|
|
"cellar" => bottle.cellar.to_s,
|
|
|
|
"rebuild" => bottle.rebuild,
|
|
|
|
"tags" => {
|
2019-11-06 14:31:03 +00:00
|
|
|
Utils::Bottles.tag.to_s => {
|
2018-11-02 17:18:07 +00:00
|
|
|
"filename" => filename.bintray,
|
2018-08-22 01:30:18 -07:00
|
|
|
"local_filename" => filename.to_s,
|
2018-11-02 17:18:07 +00:00
|
|
|
"sha256" => sha256,
|
2016-09-11 17:41:51 +01:00
|
|
|
},
|
2016-05-28 15:54:05 +01:00
|
|
|
},
|
|
|
|
},
|
2016-09-22 20:12:28 +02:00
|
|
|
"bintray" => {
|
2018-11-02 17:18:07 +00:00
|
|
|
"package" => Utils::Bottles::Bintray.package(f.name),
|
2016-09-22 20:12:28 +02:00
|
|
|
"repository" => Utils::Bottles::Bintray.repository(tap),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2018-08-06 15:02:52 +02:00
|
|
|
File.open(filename.json, "w") do |file|
|
2016-11-20 13:00:01 -05:00
|
|
|
file.write JSON.generate json
|
2013-09-21 21:24:50 +01:00
|
|
|
end
|
2012-03-07 17:29:05 -05:00
|
|
|
end
|
|
|
|
|
2020-07-31 18:58:07 +02:00
|
|
|
def merge(args:)
|
2020-03-04 17:28:24 +00:00
|
|
|
bottles_hash = args.named.reduce({}) do |hash, json_file|
|
2020-01-27 19:31:10 +01:00
|
|
|
hash.deep_merge(JSON.parse(IO.read(json_file))) do |key, first, second|
|
|
|
|
if key == "cellar"
|
|
|
|
# Prioritize HOMEBREW_CELLAR over :any over :any_skip_relocation
|
|
|
|
cellars = [first, second]
|
|
|
|
if cellars.include?(HOMEBREW_CELLAR)
|
|
|
|
HOMEBREW_CELLAR
|
|
|
|
elsif first.start_with?("/")
|
|
|
|
first
|
|
|
|
elsif second.start_with?("/")
|
|
|
|
second
|
2020-07-09 21:10:25 -07:00
|
|
|
elsif cellars.include?("any")
|
|
|
|
"any"
|
|
|
|
elsif cellars.include?("any_skip_relocation")
|
|
|
|
"any_skip_relocation"
|
2020-01-27 19:31:10 +01:00
|
|
|
else
|
|
|
|
second
|
|
|
|
end
|
|
|
|
else
|
|
|
|
second
|
|
|
|
end
|
|
|
|
end
|
2013-09-21 15:16:16 +01:00
|
|
|
end
|
2014-03-10 14:56:02 -05:00
|
|
|
|
2016-05-28 15:54:05 +01:00
|
|
|
bottles_hash.each do |formula_name, bottle_hash|
|
2013-09-21 15:16:16 +01:00
|
|
|
ohai formula_name
|
2014-03-10 14:56:02 -05:00
|
|
|
|
2016-05-28 15:54:05 +01:00
|
|
|
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)
|
2016-05-28 15:54:05 +01:00
|
|
|
bottle.cellar cellar
|
|
|
|
bottle.prefix bottle_hash["bottle"]["prefix"]
|
2016-08-18 17:32:35 +01:00
|
|
|
bottle.rebuild bottle_hash["bottle"]["rebuild"]
|
2016-05-28 15:54:05 +01:00
|
|
|
bottle_hash["bottle"]["tags"].each do |tag, tag_hash|
|
|
|
|
bottle.sha256 tag_hash["sha256"] => tag.to_sym
|
2015-09-10 19:47:22 +08:00
|
|
|
end
|
|
|
|
|
2013-09-21 21:30:57 +01:00
|
|
|
output = bottle_output bottle
|
|
|
|
|
2020-03-04 17:28:24 +00:00
|
|
|
if args.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
|
|
|
|
2016-05-28 15:54:05 +01:00
|
|
|
Utils::Inreplace.inreplace(path) do |s|
|
2020-07-25 00:48:15 +05:30
|
|
|
if s.inreplace_string.include? "bottle do"
|
2015-08-03 13:09:07 +01:00
|
|
|
update_or_add = "update"
|
2018-05-05 18:40:01 +05:30
|
|
|
if args.keep_old?
|
2016-05-28 15:54:05 +01:00
|
|
|
mismatches = []
|
2020-07-27 17:23:08 +02:00
|
|
|
bottle_block_contents = s.inreplace_string[/ bottle do(.+?)end\n/m, 1]
|
2016-05-28 15:54:05 +01:00
|
|
|
bottle_block_contents.lines.each do |line|
|
|
|
|
line = line.strip
|
|
|
|
next if line.empty?
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2016-09-10 21:15:28 +01:00
|
|
|
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
|
|
|
|
|
2018-09-12 14:01:03 -07:00
|
|
|
old_value = old_value_original.to_s.delete "'\""
|
|
|
|
old_value = old_value.to_s.delete ":" if key != "root_url"
|
2016-05-28 15:54:05 +01:00
|
|
|
tag = tag.to_s.delete ":"
|
|
|
|
|
2016-09-11 17:41:51 +01:00
|
|
|
unless tag.empty?
|
2018-09-14 17:02:19 +01:00
|
|
|
if bottle_hash["bottle"]["tags"][tag].present?
|
2016-09-04 13:22:06 +01:00
|
|
|
mismatches << "#{key} => #{tag}"
|
|
|
|
else
|
2016-09-10 21:15:28 +01:00
|
|
|
bottle.send(key, old_value => tag.to_sym)
|
2016-05-28 15:54:05 +01:00
|
|
|
end
|
|
|
|
next
|
|
|
|
end
|
|
|
|
|
2016-09-10 21:15:28 +01:00
|
|
|
value_original = bottle_hash["bottle"][key]
|
|
|
|
value = value_original.to_s
|
2016-05-28 15:54:05 +01:00
|
|
|
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
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2016-09-11 17:41:51 +01:00
|
|
|
old_value = old_value_original.inspect
|
|
|
|
value = value_original.inspect
|
|
|
|
mismatches << "#{key}: old: #{old_value}, new: #{value}"
|
2016-05-28 15:54:05 +01:00
|
|
|
end
|
2016-09-10 16:11:01 +01:00
|
|
|
|
2016-08-05 22:01:32 +08:00
|
|
|
unless mismatches.empty?
|
2019-11-04 21:00:20 +11:00
|
|
|
odie <<~EOS
|
2016-09-10 16:11:01 +01:00
|
|
|
--keep-old was passed but there are changes in:
|
|
|
|
#{mismatches.join("\n")}
|
|
|
|
EOS
|
2016-05-28 15:54:05 +01:00
|
|
|
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)
|
2015-08-03 13:09:07 +01:00
|
|
|
odie "Bottle block update failed!" unless string
|
2013-09-21 21:30:57 +01:00
|
|
|
else
|
2019-02-19 13:11:32 +00:00
|
|
|
odie "--keep-old was passed but there was no existing bottle block!" if args.keep_old?
|
2016-05-28 15:54:05 +01:00
|
|
|
puts output
|
2015-08-03 13:09:07 +01:00
|
|
|
update_or_add = "add"
|
2020-07-01 22:08:12 +01: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|sha256|version|mirror|license)\ ['"][\S\ ]+['"]| # specs with a string
|
|
|
|
(revision|version_scheme)\ \d+| # revision with a number
|
|
|
|
(stable|livecheck)\ do(\n+^\ {4}[\S\ ]+$)*\n+^\ {2}end # components with blocks
|
|
|
|
)\n+ # multiple empty lines
|
|
|
|
)+
|
|
|
|
/mx
|
2020-08-19 17:12:32 +01:00
|
|
|
string = s.sub!(pattern, "\\0#{output}\n")
|
2015-08-03 13:09:07 +01:00
|
|
|
odie "Bottle block addition failed!" unless string
|
2013-09-21 21:30:57 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-05-05 18:40:01 +05:30
|
|
|
unless args.no_commit?
|
2020-08-23 06:32:26 +02:00
|
|
|
Utils::Git.set_name_email!
|
2017-11-18 11:22:29 +00:00
|
|
|
|
2016-06-02 17:42:44 +02:00
|
|
|
short_name = formula_name.split("/", -1).last
|
2016-05-28 15:54:05 +01:00
|
|
|
pkg_version = bottle_hash["formula"]["pkg_version"]
|
|
|
|
|
|
|
|
path.parent.cd do
|
2020-04-01 16:20:35 +01:00
|
|
|
safe_system "git", "commit", "--no-edit", "--verbose",
|
2019-04-30 08:44:35 +01:00
|
|
|
"--message=#{short_name}: #{update_or_add} #{pkg_version} bottle.",
|
|
|
|
"--", path
|
2015-10-15 15:12:02 +08:00
|
|
|
end
|
2014-06-18 19:31:18 -05:00
|
|
|
end
|
2016-05-28 15:54:05 +01:00
|
|
|
else
|
|
|
|
puts output
|
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
|
|
|
end
|
2012-03-07 17:29:05 -05:00
|
|
|
end
|