2024-07-01 23:36:18 +01:00
|
|
|
# typed: strict
|
2019-11-04 21:00:20 +11:00
|
|
|
# frozen_string_literal: true
|
2019-04-19 15:38:03 +09:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
require "abstract_command"
|
2024-03-21 21:31:25 -07:00
|
|
|
require "fileutils"
|
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"
|
2024-02-05 17:42:27 +01:00
|
|
|
require "sbom"
|
2015-08-03 13:09:07 +01:00
|
|
|
require "keg"
|
|
|
|
require "formula_versions"
|
|
|
|
require "utils/inreplace"
|
|
|
|
require "erb"
|
2023-01-04 22:37:35 -05:00
|
|
|
require "utils/gzip"
|
2021-08-06 11:42:55 -04:00
|
|
|
require "api"
|
2024-01-11 19:22:16 -08:00
|
|
|
require "extend/hash/deep_merge"
|
2012-03-07 17:29:05 -05:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
module Homebrew
|
|
|
|
module DevCmd
|
|
|
|
class Bottle < AbstractCommand
|
|
|
|
include FileUtils
|
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
BOTTLE_ERB = T.let(<<-EOS.freeze, String)
|
2013-09-21 21:21:42 +01:00
|
|
|
bottle do
|
2021-04-08 17:58:24 +01:00
|
|
|
<% if [HOMEBREW_BOTTLE_DEFAULT_DOMAIN.to_s,
|
|
|
|
"#{HOMEBREW_BOTTLE_DEFAULT_DOMAIN}/bottles"].exclude?(root_url) %>
|
2021-04-28 14:57:41 -04:00
|
|
|
root_url "<%= root_url %>"<% if root_url_using.present? %>,
|
|
|
|
using: <%= root_url_using %>
|
2021-04-27 09:15:53 -04:00
|
|
|
<% end %>
|
2014-05-09 17:38:12 +09: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 %>
|
2021-01-20 23:26:41 +01:00
|
|
|
<% sha256_lines.each do |line| %>
|
|
|
|
<%= line %>
|
2013-09-23 17:30:47 +01:00
|
|
|
<% end %>
|
2013-09-21 21:21:42 +01:00
|
|
|
end
|
2018-09-28 21:39:52 +05:30
|
|
|
EOS
|
2018-07-30 18:25:38 +05:30
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
MAXIMUM_STRING_MATCHES = 100
|
2018-03-25 12:22:29 +05:30
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
ALLOWABLE_HOMEBREW_REPOSITORY_LINKS = T.let([
|
2024-03-18 12:31:44 -07:00
|
|
|
%r{#{Regexp.escape(HOMEBREW_LIBRARY)}/Homebrew/os/(mac|linux)/pkgconfig},
|
2024-07-01 23:36:18 +01:00
|
|
|
].freeze, T::Array[Regexp])
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
cmd_args do
|
|
|
|
description <<~EOS
|
|
|
|
Generate a bottle (binary package) from a formula that was 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.
|
|
|
|
EOS
|
|
|
|
switch "--skip-relocation",
|
|
|
|
description: "Do not check if the bottle can be marked as relocatable."
|
|
|
|
switch "--force-core-tap",
|
|
|
|
description: "Build a bottle even if <formula> is not in `homebrew/core` or any installed taps."
|
|
|
|
switch "--no-rebuild",
|
|
|
|
description: "If the formula specifies a rebuild version, remove it from the generated DSL."
|
|
|
|
switch "--keep-old",
|
|
|
|
description: "If the formula specifies a rebuild version, attempt to preserve its value in the " \
|
|
|
|
"generated DSL."
|
|
|
|
switch "--json",
|
|
|
|
description: "Write bottle information to a JSON file, which can be used as the value for " \
|
|
|
|
"`--merge`."
|
|
|
|
switch "--merge",
|
|
|
|
description: "Generate an updated bottle block for a formula and optionally merge it into the " \
|
|
|
|
"formula file. Instead of a formula name, requires the path to a JSON file generated " \
|
|
|
|
"with `brew bottle --json` <formula>."
|
|
|
|
switch "--write",
|
|
|
|
depends_on: "--merge",
|
|
|
|
description: "Write changes to the formula file. A new commit will be generated unless " \
|
|
|
|
"`--no-commit` is passed."
|
|
|
|
switch "--no-commit",
|
|
|
|
depends_on: "--write",
|
|
|
|
description: "When passed with `--write`, a new commit will not generated after writing changes " \
|
|
|
|
"to the formula file."
|
|
|
|
switch "--only-json-tab",
|
|
|
|
depends_on: "--json",
|
|
|
|
description: "When passed with `--json`, the tab will be written to the JSON file but not the bottle."
|
|
|
|
switch "--no-all-checks",
|
|
|
|
depends_on: "--merge",
|
|
|
|
description: "Don't try to create an `all` bottle or stop a no-change upload."
|
|
|
|
flag "--committer=",
|
|
|
|
description: "Specify a committer name and email in `git`'s standard author format."
|
|
|
|
flag "--root-url=",
|
|
|
|
description: "Use the specified <URL> as the root of the bottle's URL instead of Homebrew's default."
|
|
|
|
flag "--root-url-using=",
|
|
|
|
description: "Use the specified download strategy class for downloading the bottle's URL instead of " \
|
|
|
|
"Homebrew's default."
|
|
|
|
|
|
|
|
conflicts "--no-rebuild", "--keep-old"
|
|
|
|
|
|
|
|
named_args [:installed_formula, :file], min: 1, without_api: true
|
|
|
|
end
|
2023-08-18 11:03:22 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
sig { override.void }
|
|
|
|
def run
|
|
|
|
if args.merge?
|
|
|
|
Homebrew.install_bundler_gems!(groups: ["ast"])
|
|
|
|
return merge
|
|
|
|
end
|
2018-03-25 12:22:29 +05:30
|
|
|
|
2024-05-09 13:10:35 +01:00
|
|
|
Homebrew.install_bundler_gems!(groups: ["bottle"])
|
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
gnu_tar_formula_ensure_installed_if_needed!
|
2015-02-26 19:13:10 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
args.named.to_resolved_formulae(uniq: false).each do |formula|
|
|
|
|
bottle_formula formula
|
|
|
|
end
|
2015-02-26 19:13:10 +00:00
|
|
|
end
|
2013-10-31 01:20:28 -07:00
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig {
|
2024-07-02 00:03:28 +01:00
|
|
|
params(tag: Symbol, digest: T.any(Checksum, String), cellar: T.nilable(T.any(String, Symbol)),
|
|
|
|
tag_column: Integer, digest_column: Integer).returns(String)
|
2024-07-01 23:36:18 +01:00
|
|
|
}
|
2024-03-30 16:31:13 -07:00
|
|
|
def generate_sha256_line(tag, digest, cellar, tag_column, digest_column)
|
|
|
|
line = "sha256 "
|
|
|
|
tag_column += line.length
|
|
|
|
digest_column += line.length
|
|
|
|
if cellar.is_a?(Symbol)
|
|
|
|
line += "cellar: :#{cellar},"
|
|
|
|
elsif cellar_parameter_needed?(cellar)
|
|
|
|
line += %Q(cellar: "#{cellar}",)
|
|
|
|
end
|
|
|
|
line += " " * (tag_column - line.length)
|
|
|
|
line += "#{tag}:"
|
|
|
|
line += " " * (digest_column - line.length)
|
|
|
|
%Q(#{line}"#{digest}")
|
|
|
|
end
|
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig { params(bottle: BottleSpecification, root_url_using: T.nilable(String)).returns(String) }
|
2024-03-30 16:31:13 -07:00
|
|
|
def bottle_output(bottle, root_url_using)
|
|
|
|
cellars = bottle.checksums.filter_map do |checksum|
|
|
|
|
cellar = checksum["cellar"]
|
|
|
|
next unless cellar_parameter_needed? cellar
|
|
|
|
|
|
|
|
case cellar
|
|
|
|
when String
|
|
|
|
%Q("#{cellar}")
|
|
|
|
when Symbol
|
|
|
|
":#{cellar}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
tag_column = cellars.empty? ? 0 : "cellar: #{cellars.max_by(&:length)}, ".length
|
|
|
|
|
|
|
|
tags = bottle.checksums.map { |checksum| checksum["tag"] }
|
|
|
|
# Start where the tag ends, add the max length of the tag, add two for the `: `
|
|
|
|
digest_column = tag_column + tags.max_by(&:length).length + 2
|
|
|
|
|
|
|
|
sha256_lines = bottle.checksums.map do |checksum|
|
|
|
|
generate_sha256_line(checksum["tag"], checksum["digest"], checksum["cellar"], tag_column, digest_column)
|
|
|
|
end
|
|
|
|
erb_binding = bottle.instance_eval { binding }
|
|
|
|
erb_binding.local_variable_set(:sha256_lines, sha256_lines)
|
|
|
|
erb_binding.local_variable_set(:root_url_using, root_url_using)
|
|
|
|
erb = ERB.new BOTTLE_ERB
|
|
|
|
erb.result(erb_binding).gsub(/^\s*$\n/, "")
|
|
|
|
end
|
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig { params(filenames: T::Array[String]).returns(T::Array[T::Hash[String, T.untyped]]) }
|
2024-03-30 16:31:13 -07:00
|
|
|
def parse_json_files(filenames)
|
|
|
|
filenames.map do |filename|
|
|
|
|
JSON.parse(File.read(filename))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig { params(json_files: T::Array[T::Hash[String, T.untyped]]).returns(T::Hash[String, T.untyped]) }
|
2024-03-30 16:31:13 -07:00
|
|
|
def merge_json_files(json_files)
|
|
|
|
json_files.reduce({}) do |hash, json_file|
|
|
|
|
json_file.each_value do |json_hash|
|
|
|
|
json_bottle = json_hash["bottle"]
|
|
|
|
cellar = json_bottle.delete("cellar")
|
|
|
|
json_bottle["tags"].each_value do |json_platform|
|
|
|
|
json_platform["cellar"] ||= cellar
|
|
|
|
end
|
|
|
|
end
|
|
|
|
hash.deep_merge(json_file)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig {
|
|
|
|
params(old_keys: T::Array[String], old_bottle_spec: BottleSpecification,
|
|
|
|
new_bottle_hash: T::Hash[String, T.untyped]).returns(T::Array[T::Array[String]])
|
|
|
|
}
|
2024-03-30 16:31:13 -07:00
|
|
|
def merge_bottle_spec(old_keys, old_bottle_spec, new_bottle_hash)
|
|
|
|
mismatches = []
|
|
|
|
checksums = []
|
|
|
|
|
|
|
|
new_values = {
|
|
|
|
root_url: new_bottle_hash["root_url"],
|
|
|
|
rebuild: new_bottle_hash["rebuild"],
|
|
|
|
}
|
|
|
|
|
|
|
|
skip_keys = [:sha256, :cellar]
|
|
|
|
old_keys.each do |key|
|
|
|
|
next if skip_keys.include?(key)
|
|
|
|
|
|
|
|
old_value = old_bottle_spec.send(key).to_s
|
|
|
|
new_value = new_values[key].to_s
|
|
|
|
|
|
|
|
next if old_value.present? && new_value == old_value
|
|
|
|
|
|
|
|
mismatches << "#{key}: old: #{old_value.inspect}, new: #{new_value.inspect}"
|
|
|
|
end
|
|
|
|
|
|
|
|
return [mismatches, checksums] if old_keys.exclude? :sha256
|
|
|
|
|
|
|
|
old_bottle_spec.collector.each_tag do |tag|
|
|
|
|
old_tag_spec = old_bottle_spec.collector.specification_for(tag)
|
|
|
|
old_hexdigest = old_tag_spec.checksum.hexdigest
|
|
|
|
old_cellar = old_tag_spec.cellar
|
|
|
|
new_value = new_bottle_hash.dig("tags", tag.to_s)
|
|
|
|
if new_value.present? && new_value["sha256"] != old_hexdigest
|
|
|
|
mismatches << "sha256 #{tag}: old: #{old_hexdigest.inspect}, new: #{new_value["sha256"].inspect}"
|
|
|
|
elsif new_value.present? && new_value["cellar"] != old_cellar.to_s
|
|
|
|
mismatches << "cellar #{tag}: old: #{old_cellar.to_s.inspect}, new: #{new_value["cellar"].inspect}"
|
|
|
|
else
|
|
|
|
checksums << { cellar: old_cellar, tag.to_sym => old_hexdigest }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
[mismatches, checksums]
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig {
|
|
|
|
params(string: String, keg: Keg, ignores: T::Array[String],
|
|
|
|
formula_and_runtime_deps_names: T.nilable(T::Array[String])).returns(T::Boolean)
|
|
|
|
}
|
2024-03-18 12:31:44 -07:00
|
|
|
def keg_contain?(string, keg, ignores, formula_and_runtime_deps_names = nil)
|
|
|
|
@put_string_exists_header, @put_filenames = nil
|
2016-09-22 20:12:28 +02:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
print_filename = lambda do |str, filename|
|
|
|
|
unless @put_string_exists_header
|
|
|
|
opoo "String '#{str}' still exists in these files:"
|
2024-07-01 23:36:18 +01:00
|
|
|
@put_string_exists_header = T.let(true, T.nilable(T::Boolean))
|
2024-03-18 12:31:44 -07:00
|
|
|
end
|
2013-10-31 01:20:28 -07:00
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
@put_filenames ||= T.let([], T.nilable(T::Array[T.any(String, Pathname)]))
|
2015-07-10 19:51:43 +08:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
return false if @put_filenames.include?(filename)
|
2013-10-31 01:20:28 -07:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
puts Formatter.error(filename.to_s)
|
|
|
|
@put_filenames << filename
|
2014-01-20 15:26:18 -08:00
|
|
|
end
|
2013-10-31 01:20:28 -07:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
result = T.let(false, T::Boolean)
|
2015-09-23 14:55:22 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
keg.each_unique_file_matching(string) do |file|
|
|
|
|
next if Metafiles::EXTENSIONS.include?(file.extname) # Skip document files.
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
linked_libraries = Keg.file_linked_libraries(file, string)
|
|
|
|
result ||= !linked_libraries.empty?
|
2015-02-20 14:29:43 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
if args.verbose?
|
|
|
|
print_filename.call(string, file) unless linked_libraries.empty?
|
|
|
|
linked_libraries.each do |lib|
|
|
|
|
puts " #{Tty.bold}-->#{Tty.reset} links to #{lib}"
|
|
|
|
end
|
|
|
|
end
|
2013-12-17 20:46:42 -06:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
text_matches = Keg.text_matches_in_file(file, string, ignores, linked_libraries,
|
|
|
|
formula_and_runtime_deps_names)
|
|
|
|
result = true if text_matches.any?
|
2016-07-26 21:50:00 -07:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
next if !args.verbose? || text_matches.empty?
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
print_filename.call(string, file)
|
|
|
|
text_matches.first(MAXIMUM_STRING_MATCHES).each do |match, offset|
|
|
|
|
puts " #{Tty.bold}-->#{Tty.reset} match '#{match}' at offset #{Tty.bold}0x#{offset}#{Tty.reset}"
|
|
|
|
end
|
|
|
|
|
|
|
|
if text_matches.size > MAXIMUM_STRING_MATCHES
|
|
|
|
puts "Only the first #{MAXIMUM_STRING_MATCHES} matches were output."
|
|
|
|
end
|
|
|
|
end
|
2014-03-18 19:03:24 -05:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
keg_contain_absolute_symlink_starting_with?(string, keg) || result
|
2015-09-11 16:49:39 +08:00
|
|
|
end
|
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig { params(string: String, keg: Keg).returns(T::Boolean) }
|
2024-03-18 12:31:44 -07:00
|
|
|
def keg_contain_absolute_symlink_starting_with?(string, keg)
|
|
|
|
absolute_symlinks_start_with_string = []
|
|
|
|
keg.find do |pn|
|
|
|
|
next if !pn.symlink? || !(link = pn.readlink).absolute?
|
2013-03-11 18:56:26 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
absolute_symlinks_start_with_string << pn if link.to_s.start_with?(string)
|
|
|
|
end
|
2021-02-02 15:09:39 -05:00
|
|
|
|
2024-03-18 12:31:44 -07: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}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
!absolute_symlinks_start_with_string.empty?
|
|
|
|
end
|
2021-01-20 23:26:41 +01:00
|
|
|
|
2024-07-02 11:21:29 +01:00
|
|
|
sig { params(cellar: T.nilable(T.any(String, Symbol))).returns(T::Boolean) }
|
2024-03-18 12:31:44 -07:00
|
|
|
def cellar_parameter_needed?(cellar)
|
|
|
|
default_cellars = [
|
|
|
|
Homebrew::DEFAULT_MACOS_CELLAR,
|
|
|
|
Homebrew::DEFAULT_MACOS_ARM_CELLAR,
|
|
|
|
Homebrew::DEFAULT_LINUX_CELLAR,
|
|
|
|
]
|
|
|
|
cellar.present? && default_cellars.exclude?(cellar)
|
|
|
|
end
|
2021-02-02 15:09:39 -05:00
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig { returns(T.nilable(T::Boolean)) }
|
2024-03-18 12:31:44 -07:00
|
|
|
def sudo_purge
|
|
|
|
return unless ENV["HOMEBREW_BOTTLE_SUDO_PURGE"]
|
2023-02-06 21:15:38 +09:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
system "/usr/bin/sudo", "--non-interactive", "/usr/sbin/purge"
|
|
|
|
end
|
2023-08-04 10:02:44 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
sig { returns(T::Array[String]) }
|
|
|
|
def tar_args
|
|
|
|
[].freeze
|
|
|
|
end
|
2023-02-10 17:01:22 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
sig { params(gnu_tar_formula: Formula).returns(String) }
|
|
|
|
def gnu_tar(gnu_tar_formula)
|
|
|
|
"#{gnu_tar_formula.opt_bin}/tar"
|
|
|
|
end
|
2023-08-18 11:03:22 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
sig { params(mtime: String).returns(T::Array[String]) }
|
|
|
|
def reproducible_gnutar_args(mtime)
|
|
|
|
# Ensure gnu tar is set up for reproducibility.
|
|
|
|
# https://reproducible-builds.org/docs/archives/
|
|
|
|
[
|
|
|
|
# File modification times
|
|
|
|
"--mtime=#{mtime}",
|
|
|
|
# File ordering
|
|
|
|
"--sort=name",
|
|
|
|
# Users, groups and numeric ids
|
|
|
|
"--owner=0", "--group=0", "--numeric-owner",
|
|
|
|
# PAX headers
|
|
|
|
"--format=pax",
|
|
|
|
# Set exthdr names to exclude PID (for GNU tar <1.33). Also don't store atime and ctime.
|
|
|
|
"--pax-option=globexthdr.name=/GlobalHead.%n,exthdr.name=%d/PaxHeaders/%f,delete=atime,delete=ctime"
|
|
|
|
].freeze
|
|
|
|
end
|
2023-08-18 11:03:22 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
sig { returns(T.nilable(Formula)) }
|
|
|
|
def gnu_tar_formula_ensure_installed_if_needed!
|
|
|
|
gnu_tar_formula = begin
|
|
|
|
Formula["gnu-tar"]
|
|
|
|
rescue FormulaUnavailableError
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
return if gnu_tar_formula.blank?
|
2023-08-18 11:03:22 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
ensure_formula_installed!(gnu_tar_formula, reason: "bottling")
|
2023-02-10 17:01:22 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
gnu_tar_formula
|
|
|
|
end
|
2023-02-10 17:01:22 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
sig { params(mtime: String).returns([String, T::Array[String]]) }
|
|
|
|
def setup_tar_and_args!(mtime)
|
|
|
|
# Without --only-json-tab bottles are never reproducible
|
|
|
|
default_tar_args = ["tar", tar_args].freeze
|
|
|
|
return default_tar_args unless args.only_json_tab?
|
2023-02-07 19:17:00 +09:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
# Use gnu-tar as it can be set up for reproducibility better than libarchive
|
|
|
|
# and to be consistent between macOS and Linux.
|
|
|
|
gnu_tar_formula = gnu_tar_formula_ensure_installed_if_needed!
|
|
|
|
return default_tar_args if gnu_tar_formula.blank?
|
2021-08-21 16:22:36 -07:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
[gnu_tar(gnu_tar_formula), reproducible_gnutar_args(mtime)].freeze
|
|
|
|
end
|
2021-08-21 16:22:36 -07:00
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig { params(formula: T.untyped).returns(T::Array[T.untyped]) }
|
2024-03-18 12:31:44 -07:00
|
|
|
def formula_ignores(formula)
|
|
|
|
ignores = []
|
|
|
|
cellar_regex = Regexp.escape(HOMEBREW_CELLAR)
|
|
|
|
prefix_regex = Regexp.escape(HOMEBREW_PREFIX)
|
2023-02-10 17:01:22 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
# Ignore matches to go keg, because all go binaries are statically linked.
|
|
|
|
any_go_deps = formula.deps.any? do |dep|
|
|
|
|
Version.formula_optionally_versioned_regex(:go).match?(dep.name)
|
|
|
|
end
|
|
|
|
if any_go_deps
|
|
|
|
go_regex = Version.formula_optionally_versioned_regex(:go, full: false)
|
|
|
|
ignores << %r{#{cellar_regex}/#{go_regex}/[\d.]+/libexec}
|
|
|
|
end
|
2021-08-21 16:22:36 -07:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
# TODO: Refactor and move to extend/os
|
|
|
|
# rubocop:disable Homebrew/MoveToExtendOS
|
|
|
|
ignores << case formula.name
|
|
|
|
# On Linux, GCC installation can be moved so long as the whole directory tree is moved together:
|
|
|
|
# https://gcc-help.gcc.gnu.narkive.com/GnwuCA7l/moving-gcc-from-the-installation-path-is-it-allowed.
|
|
|
|
when Version.formula_optionally_versioned_regex(:gcc)
|
|
|
|
Regexp.union(%r{#{cellar_regex}/gcc}, %r{#{prefix_regex}/opt/gcc}) if OS.linux?
|
|
|
|
# binutils is relocatable for the same reason: https://github.com/Homebrew/brew/pull/11899#issuecomment-906804451.
|
|
|
|
when Version.formula_optionally_versioned_regex(:binutils)
|
|
|
|
%r{#{cellar_regex}/binutils} if OS.linux?
|
|
|
|
end
|
|
|
|
# rubocop:enable Homebrew/MoveToExtendOS
|
2012-03-15 10:57:34 +13:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
ignores.compact
|
2023-03-10 23:46:07 +00:00
|
|
|
end
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig { params(formula: Formula).void }
|
2024-03-18 12:31:44 -07:00
|
|
|
def bottle_formula(formula)
|
|
|
|
local_bottle_json = args.json? && formula.local_bottle_path.present?
|
2016-09-22 20:12:28 +02:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
unless local_bottle_json
|
|
|
|
unless formula.latest_version_installed?
|
|
|
|
return ofail "Formula not installed or up-to-date: #{formula.full_name}"
|
|
|
|
end
|
|
|
|
unless Utils::Bottles.built_as? formula
|
|
|
|
return ofail "Formula was not installed with `--build-bottle`: #{formula.full_name}"
|
|
|
|
end
|
|
|
|
end
|
2015-12-26 13:15:29 +08:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
tap = formula.tap
|
|
|
|
if tap.nil?
|
|
|
|
return ofail "Formula not from core or any installed taps: #{formula.full_name}" unless args.force_core_tap?
|
2014-02-15 11:28:48 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
tap = CoreTap.instance
|
|
|
|
end
|
|
|
|
raise TapUnavailableError, tap.name unless tap.installed?
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
return ofail "Formula has no stable version: #{formula.full_name}" unless formula.stable
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
bottle_tag, rebuild = if local_bottle_json
|
|
|
|
_, tag_string, rebuild_string = Utils::Bottles.extname_tag_rebuild(formula.local_bottle_path.to_s)
|
|
|
|
[tag_string.to_sym, rebuild_string.to_i]
|
2021-04-30 14:06:37 -04:00
|
|
|
end
|
2013-12-10 15:16:22 -06:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
bottle_tag = if bottle_tag
|
|
|
|
Utils::Bottles::Tag.from_symbol(bottle_tag)
|
|
|
|
else
|
|
|
|
Utils::Bottles.tag
|
|
|
|
end
|
2012-03-07 17:29:05 -05:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
rebuild ||= if args.no_rebuild? || !tap
|
|
|
|
0
|
|
|
|
elsif args.keep_old?
|
|
|
|
formula.bottle_specification.rebuild
|
|
|
|
else
|
|
|
|
ohai "Determining #{formula.full_name} bottle rebuild..."
|
|
|
|
FormulaVersions.new(formula).formula_at_revision("origin/HEAD") do |upstream_formula|
|
|
|
|
if formula.pkg_version == upstream_formula.pkg_version
|
|
|
|
upstream_formula.bottle_specification.rebuild + 1
|
|
|
|
else
|
|
|
|
0
|
|
|
|
end
|
|
|
|
end || 0
|
|
|
|
end
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
filename = ::Bottle::Filename.create(formula, bottle_tag, rebuild)
|
|
|
|
local_filename = filename.to_s
|
|
|
|
bottle_path = Pathname.pwd/local_filename
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
tab = nil
|
|
|
|
keg = nil
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
tap_path = tap.path
|
|
|
|
tap_git_revision = tap.git_head
|
|
|
|
tap_git_remote = tap.remote
|
2015-12-15 14:21:27 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
root_url = args.root_url
|
2013-03-11 18:56:26 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
relocatable = T.let(false, T::Boolean)
|
|
|
|
skip_relocation = T.let(false, T::Boolean)
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
prefix = HOMEBREW_PREFIX.to_s
|
|
|
|
cellar = HOMEBREW_CELLAR.to_s
|
2021-04-13 14:26:31 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
if local_bottle_json
|
|
|
|
bottle_path = formula.local_bottle_path
|
2024-07-01 23:36:18 +01:00
|
|
|
local_filename = bottle_path&.basename&.to_s
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
tab_path = Utils::Bottles.receipt_path(bottle_path)
|
|
|
|
raise "This bottle does not contain the file INSTALL_RECEIPT.json: #{bottle_path}" unless tab_path
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
tab_json = Utils::Bottles.file_from_bottle(bottle_path, tab_path)
|
|
|
|
tab = Tab.from_file_content(tab_json, tab_path)
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-19 12:36:30 -07:00
|
|
|
tag_spec = Formula[formula.name].bottle_specification
|
|
|
|
.tag_specification_for(bottle_tag, no_older_versions: true)
|
2024-03-18 12:31:44 -07:00
|
|
|
relocatable = [:any, :any_skip_relocation].include?(tag_spec.cellar)
|
|
|
|
skip_relocation = tag_spec.cellar == :any_skip_relocation
|
2021-03-30 17:35:13 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
prefix = bottle_tag.default_prefix
|
|
|
|
cellar = bottle_tag.default_cellar
|
2021-03-30 17:35:13 +01:00
|
|
|
else
|
2024-03-18 12:31:44 -07:00
|
|
|
tar_filename = filename.to_s.sub(/.gz$/, "")
|
|
|
|
tar_path = Pathname.pwd/tar_filename
|
2015-12-15 14:21:27 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
keg = Keg.new(formula.prefix)
|
2013-12-12 19:46:37 -06:00
|
|
|
end
|
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
ohai "Bottling #{local_filename}..."
|
|
|
|
|
|
|
|
formula_and_runtime_deps_names = [formula.name] + formula.runtime_dependencies.map(&:name)
|
|
|
|
|
|
|
|
# this will be nil when using a local bottle
|
|
|
|
keg&.lock do
|
|
|
|
original_tab = nil
|
|
|
|
changed_files = nil
|
|
|
|
|
|
|
|
begin
|
|
|
|
keg.delete_pyc_files!
|
|
|
|
|
|
|
|
changed_files = keg.replace_locations_with_placeholders unless args.skip_relocation?
|
|
|
|
|
|
|
|
Formula.clear_cache
|
|
|
|
Keg.clear_cache
|
|
|
|
Tab.clear_cache
|
|
|
|
Dependency.clear_cache
|
|
|
|
Requirement.clear_cache
|
2024-02-05 17:42:27 +01:00
|
|
|
|
2024-04-28 03:23:21 +02:00
|
|
|
tab = keg.tab
|
2024-03-18 12:31:44 -07:00
|
|
|
original_tab = tab.dup
|
|
|
|
tab.poured_from_bottle = false
|
|
|
|
tab.time = nil
|
|
|
|
tab.changed_files = changed_files.dup
|
|
|
|
if args.only_json_tab?
|
|
|
|
tab.changed_files.delete(Pathname.new(Tab::FILENAME))
|
|
|
|
tab.tabfile.unlink
|
|
|
|
else
|
|
|
|
tab.write
|
|
|
|
end
|
|
|
|
|
2024-05-09 13:10:35 +01:00
|
|
|
sbom = SBOM.create(formula, tab)
|
2024-05-13 07:36:51 +01:00
|
|
|
sbom.write(bottling: true)
|
2024-02-05 17:42:27 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
keg.consistent_reproducible_symlink_permissions!
|
|
|
|
|
|
|
|
cd cellar do
|
|
|
|
sudo_purge
|
|
|
|
# Tar then gzip for reproducible bottles.
|
|
|
|
tar_mtime = tab.source_modified_time.strftime("%Y-%m-%d %H:%M:%S")
|
|
|
|
tar, tar_args = setup_tar_and_args!(tar_mtime)
|
|
|
|
safe_system tar, "--create", "--numeric-owner",
|
|
|
|
*tar_args,
|
|
|
|
"--file", tar_path, "#{formula.name}/#{formula.pkg_version}"
|
|
|
|
sudo_purge
|
|
|
|
# Set filename as it affects the tarball checksum.
|
|
|
|
relocatable_tar_path = "#{formula}-bottle.tar"
|
|
|
|
mv T.must(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).
|
|
|
|
Utils::Gzip.compress_with_options(relocatable_tar_path,
|
|
|
|
mtime: tab.source_modified_time,
|
|
|
|
orig_name: relocatable_tar_path,
|
2024-07-01 23:36:18 +01:00
|
|
|
output: T.must(bottle_path))
|
2024-03-18 12:31:44 -07:00
|
|
|
sudo_purge
|
|
|
|
end
|
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
if bottle_path && bottle_path.size > 1 * 1024 * 1024
|
|
|
|
ohai "Detecting if #{local_filename} is relocatable..."
|
|
|
|
end
|
2024-03-18 12:31:44 -07:00
|
|
|
|
|
|
|
prefix_check = if Homebrew.default_prefix?(prefix)
|
|
|
|
File.join(prefix, "opt")
|
|
|
|
else
|
|
|
|
prefix
|
|
|
|
end
|
|
|
|
|
|
|
|
# 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)$}]
|
|
|
|
|
|
|
|
# Add additional workarounds to ignore
|
|
|
|
ignores += formula_ignores(formula)
|
|
|
|
|
|
|
|
repository_reference = if HOMEBREW_PREFIX == HOMEBREW_REPOSITORY
|
|
|
|
HOMEBREW_LIBRARY
|
|
|
|
else
|
|
|
|
HOMEBREW_REPOSITORY
|
|
|
|
end.to_s
|
|
|
|
if keg_contain?(repository_reference, keg, ignores + ALLOWABLE_HOMEBREW_REPOSITORY_LINKS)
|
|
|
|
odie "Bottle contains non-relocatable reference to #{repository_reference}!"
|
|
|
|
end
|
|
|
|
|
|
|
|
relocatable = true
|
|
|
|
if args.skip_relocation?
|
|
|
|
skip_relocation = true
|
|
|
|
else
|
|
|
|
relocatable = false if keg_contain?(prefix_check, keg, ignores, formula_and_runtime_deps_names)
|
|
|
|
relocatable = false if keg_contain?(cellar, keg, ignores, formula_and_runtime_deps_names)
|
|
|
|
relocatable = false if keg_contain?(HOMEBREW_LIBRARY.to_s, keg, ignores, formula_and_runtime_deps_names)
|
|
|
|
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)
|
|
|
|
relocatable = false if keg_contain?("#{prefix}/share/vim", keg, ignores)
|
|
|
|
end
|
|
|
|
skip_relocation = relocatable && !keg.require_relocation?
|
|
|
|
end
|
|
|
|
puts if !relocatable && args.verbose?
|
|
|
|
rescue Interrupt
|
2024-07-01 23:36:18 +01:00
|
|
|
ignore_interrupts { bottle_path.unlink if bottle_path&.exist? }
|
2024-03-18 12:31:44 -07:00
|
|
|
raise
|
|
|
|
ensure
|
|
|
|
ignore_interrupts do
|
|
|
|
original_tab&.write
|
|
|
|
keg.replace_placeholders_with_locations changed_files unless args.skip_relocation?
|
|
|
|
end
|
|
|
|
end
|
2013-12-12 19:46:37 -06:00
|
|
|
end
|
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
bottle = BottleSpecification.new
|
|
|
|
bottle.tap = tap
|
|
|
|
bottle.root_url(root_url) if root_url
|
|
|
|
bottle_cellar = if relocatable
|
|
|
|
if skip_relocation
|
|
|
|
:any_skip_relocation
|
|
|
|
else
|
|
|
|
:any
|
|
|
|
end
|
2020-12-17 09:42:08 +00:00
|
|
|
else
|
2024-03-18 12:31:44 -07:00
|
|
|
cellar
|
2020-12-17 09:42:08 +00:00
|
|
|
end
|
2024-03-18 12:31:44 -07:00
|
|
|
bottle.rebuild rebuild
|
2024-07-01 23:36:18 +01:00
|
|
|
sha256 = bottle_path&.sha256
|
2024-03-18 12:31:44 -07:00
|
|
|
bottle.sha256 cellar: bottle_cellar, bottle_tag.to_sym => sha256
|
|
|
|
|
|
|
|
old_spec = formula.bottle_specification
|
|
|
|
if args.keep_old? && !old_spec.checksums.empty?
|
|
|
|
mismatches = [:root_url, :rebuild].reject do |key|
|
|
|
|
old_spec.send(key) == bottle.send(key)
|
2020-12-17 09:42:08 +00:00
|
|
|
end
|
2024-03-18 12:31:44 -07:00
|
|
|
unless mismatches.empty?
|
2024-07-01 23:36:18 +01:00
|
|
|
bottle_path.unlink if bottle_path&.exist?
|
2024-03-18 12:31:44 -07:00
|
|
|
|
|
|
|
mismatches.map! do |key|
|
|
|
|
old_value = old_spec.send(key).inspect
|
|
|
|
value = bottle.send(key).inspect
|
|
|
|
"#{key}: old: #{old_value}, new: #{value}"
|
|
|
|
end
|
|
|
|
|
|
|
|
odie <<~EOS
|
|
|
|
`--keep-old` was passed but there are changes in:
|
|
|
|
#{mismatches.join("\n")}
|
|
|
|
EOS
|
2016-07-27 08:54:56 -07:00
|
|
|
end
|
2016-09-10 10:13:33 +01:00
|
|
|
end
|
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
output = bottle_output(bottle, args.root_url_using)
|
|
|
|
|
|
|
|
puts "./#{local_filename}"
|
|
|
|
puts output
|
|
|
|
|
|
|
|
return unless args.json?
|
|
|
|
|
2024-06-19 01:27:23 +02:00
|
|
|
if keg
|
|
|
|
keg_prefix = "#{keg}/"
|
|
|
|
path_exec_files = [keg/"bin", keg/"sbin"].select(&:exist?)
|
|
|
|
.flat_map(&:children)
|
|
|
|
.select(&:executable?)
|
|
|
|
.map { |path| path.to_s.delete_prefix(keg_prefix) }
|
|
|
|
all_files = keg.find
|
|
|
|
.select(&:file?)
|
|
|
|
.map { |path| path.to_s.delete_prefix(keg_prefix) }
|
|
|
|
end
|
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
json = {
|
|
|
|
formula.full_name => {
|
|
|
|
"formula" => {
|
|
|
|
"name" => formula.name,
|
|
|
|
"pkg_version" => formula.pkg_version.to_s,
|
|
|
|
"path" => formula.path.to_s.delete_prefix("#{HOMEBREW_REPOSITORY}/"),
|
|
|
|
"tap_git_path" => formula.path.to_s.delete_prefix("#{tap_path}/"),
|
|
|
|
"tap_git_revision" => tap_git_revision,
|
|
|
|
"tap_git_remote" => tap_git_remote,
|
|
|
|
# descriptions can contain emoji. sigh.
|
|
|
|
"desc" => formula.desc.to_s.encode(
|
|
|
|
Encoding.find("ASCII"),
|
|
|
|
invalid: :replace, undef: :replace, replace: "",
|
|
|
|
).strip,
|
|
|
|
"license" => SPDX.license_expression_to_string(formula.license),
|
|
|
|
"homepage" => formula.homepage,
|
|
|
|
},
|
|
|
|
"bottle" => {
|
|
|
|
"root_url" => bottle.root_url,
|
|
|
|
"cellar" => bottle_cellar.to_s,
|
|
|
|
"rebuild" => bottle.rebuild,
|
|
|
|
"date" => Pathname(filename.to_s).mtime.strftime("%F"),
|
|
|
|
"tags" => {
|
|
|
|
bottle_tag.to_s => {
|
2024-06-19 01:27:23 +02:00
|
|
|
"filename" => filename.url_encode,
|
|
|
|
"local_filename" => filename.to_s,
|
|
|
|
"sha256" => sha256,
|
|
|
|
"tab" => tab.to_bottle_hash,
|
|
|
|
"path_exec_files" => path_exec_files,
|
|
|
|
"all_files" => all_files,
|
2024-03-18 12:31:44 -07:00
|
|
|
},
|
|
|
|
},
|
2016-09-11 17:41:51 +01:00
|
|
|
},
|
2016-05-28 15:54:05 +01:00
|
|
|
},
|
2024-03-18 12:31:44 -07:00
|
|
|
}
|
2012-03-07 17:29:05 -05:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
puts "Writing #{filename.json}" if args.verbose?
|
|
|
|
json_path = Pathname(filename.json)
|
|
|
|
json_path.unlink if json_path.exist?
|
|
|
|
json_path.write(JSON.pretty_generate(json))
|
|
|
|
end
|
2020-12-01 19:08:59 +01:00
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig { returns(T::Hash[String, T.untyped]) }
|
2024-03-18 12:31:44 -07:00
|
|
|
def merge
|
|
|
|
bottles_hash = merge_json_files(parse_json_files(args.named))
|
2014-03-10 14:56:02 -05:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
any_cellars = ["any", "any_skip_relocation"]
|
|
|
|
bottles_hash.each do |formula_name, bottle_hash|
|
|
|
|
ohai formula_name
|
2021-04-21 17:10:05 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
bottle = BottleSpecification.new
|
|
|
|
bottle.root_url bottle_hash["bottle"]["root_url"]
|
|
|
|
bottle.rebuild bottle_hash["bottle"]["rebuild"]
|
2021-11-08 03:39:18 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
path = HOMEBREW_REPOSITORY/bottle_hash["formula"]["path"]
|
|
|
|
formula = Formulary.factory(path)
|
2021-11-08 03:39:18 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
old_bottle_spec = formula.bottle_specification
|
|
|
|
old_pkg_version = formula.pkg_version
|
|
|
|
FormulaVersions.new(formula).formula_at_revision("origin/HEAD") do |upstream_formula|
|
|
|
|
old_pkg_version = upstream_formula.pkg_version
|
|
|
|
end
|
2021-04-21 17:10:05 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
old_bottle_spec_matches = old_bottle_spec &&
|
|
|
|
bottle_hash["formula"]["pkg_version"] == old_pkg_version.to_s &&
|
|
|
|
bottle.root_url == old_bottle_spec.root_url &&
|
|
|
|
old_bottle_spec.collector.tags.present?
|
|
|
|
|
|
|
|
# if all the cellars and checksums are the same: we can create an
|
|
|
|
# `all: $SHA256` bottle.
|
|
|
|
tag_hashes = bottle_hash["bottle"]["tags"].values
|
|
|
|
all_bottle = !args.no_all_checks? &&
|
|
|
|
(!old_bottle_spec_matches || bottle.rebuild != old_bottle_spec.rebuild) &&
|
|
|
|
tag_hashes.count > 1 &&
|
|
|
|
tag_hashes.uniq { |tag_hash| "#{tag_hash["cellar"]}-#{tag_hash["sha256"]}" }.count == 1
|
|
|
|
|
|
|
|
bottle_hash["bottle"]["tags"].each do |tag, tag_hash|
|
|
|
|
cellar = tag_hash["cellar"]
|
|
|
|
cellar = cellar.to_sym if any_cellars.include?(cellar)
|
|
|
|
|
|
|
|
tag_sym = if all_bottle
|
|
|
|
:all
|
|
|
|
else
|
|
|
|
tag.to_sym
|
|
|
|
end
|
|
|
|
|
|
|
|
sha256_hash = { cellar:, tag_sym => tag_hash["sha256"] }
|
|
|
|
bottle.sha256 sha256_hash
|
|
|
|
|
|
|
|
break if all_bottle
|
|
|
|
end
|
2021-04-21 17:10:05 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
unless args.write?
|
|
|
|
puts bottle_output(bottle, args.root_url_using)
|
|
|
|
next
|
|
|
|
end
|
2015-09-10 19:47:22 +08:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
no_bottle_changes = if !args.no_all_checks? && old_bottle_spec_matches &&
|
|
|
|
bottle.rebuild != old_bottle_spec.rebuild
|
|
|
|
bottle.collector.tags.all? do |tag|
|
|
|
|
tag_spec = bottle.collector.specification_for(tag)
|
|
|
|
next false if tag_spec.blank?
|
2021-04-21 17:10:05 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
old_tag_spec = old_bottle_spec.collector.specification_for(tag)
|
|
|
|
next false if old_tag_spec.blank?
|
2021-01-06 09:11:34 -08:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
next false if tag_spec.cellar != old_tag_spec.cellar
|
2021-04-29 16:00:22 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
tag_spec.checksum.hexdigest == old_tag_spec.checksum.hexdigest
|
|
|
|
end
|
|
|
|
end
|
2021-04-29 16:00:22 +01:00
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
all_bottle_hash = T.let(nil, T.nilable(T::Hash[String, T.untyped]))
|
2024-03-18 12:31:44 -07:00
|
|
|
bottle_hash["bottle"]["tags"].each do |tag, tag_hash|
|
|
|
|
filename = ::Bottle::Filename.new(
|
|
|
|
formula_name,
|
|
|
|
PkgVersion.parse(bottle_hash["formula"]["pkg_version"]),
|
|
|
|
Utils::Bottles::Tag.from_symbol(tag.to_sym),
|
|
|
|
bottle_hash["bottle"]["rebuild"],
|
|
|
|
)
|
|
|
|
|
|
|
|
if all_bottle && all_bottle_hash.nil?
|
|
|
|
all_bottle_tag_hash = tag_hash.dup
|
|
|
|
|
|
|
|
all_filename = ::Bottle::Filename.new(
|
|
|
|
formula_name,
|
|
|
|
PkgVersion.parse(bottle_hash["formula"]["pkg_version"]),
|
|
|
|
Utils::Bottles::Tag.from_symbol(:all),
|
|
|
|
bottle_hash["bottle"]["rebuild"],
|
|
|
|
)
|
|
|
|
|
|
|
|
all_bottle_tag_hash["filename"] = all_filename.url_encode
|
|
|
|
all_bottle_tag_hash["local_filename"] = all_filename.to_s
|
|
|
|
cellar = all_bottle_tag_hash.delete("cellar")
|
|
|
|
|
|
|
|
all_bottle_formula_hash = bottle_hash.dup
|
|
|
|
all_bottle_formula_hash["bottle"]["cellar"] = cellar
|
|
|
|
all_bottle_formula_hash["bottle"]["tags"] = { all: all_bottle_tag_hash }
|
|
|
|
|
|
|
|
all_bottle_hash = { formula_name => all_bottle_formula_hash }
|
|
|
|
|
|
|
|
puts "Copying #{filename} to #{all_filename}" if args.verbose?
|
|
|
|
FileUtils.cp filename.to_s, all_filename.to_s
|
|
|
|
|
|
|
|
puts "Writing #{all_filename.json}" if args.verbose?
|
|
|
|
all_local_json_path = Pathname(all_filename.json)
|
|
|
|
all_local_json_path.unlink if all_local_json_path.exist?
|
|
|
|
all_local_json_path.write(JSON.pretty_generate(all_bottle_hash))
|
|
|
|
end
|
|
|
|
|
|
|
|
if all_bottle || no_bottle_changes
|
|
|
|
puts "Removing #{filename} and #{filename.json}" if args.verbose?
|
|
|
|
FileUtils.rm_f [filename.to_s, filename.json]
|
|
|
|
end
|
|
|
|
end
|
2021-01-01 15:23:09 -08:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
next if no_bottle_changes
|
2013-12-27 16:43:34 -06:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
require "utils/ast"
|
|
|
|
formula_ast = Utils::AST::FormulaAST.new(path.read)
|
|
|
|
checksums = old_checksums(formula, formula_ast, bottle_hash)
|
|
|
|
update_or_add = checksums.nil? ? "add" : "update"
|
2013-09-21 21:30:57 +01:00
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
checksums&.each { |checksum| bottle.sha256(checksum) }
|
2024-03-18 12:31:44 -07:00
|
|
|
output = bottle_output(bottle, args.root_url_using)
|
|
|
|
puts output
|
2017-11-18 11:22:29 +00:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
case update_or_add
|
|
|
|
when "update"
|
|
|
|
formula_ast.replace_bottle_block(output)
|
|
|
|
when "add"
|
|
|
|
formula_ast.add_bottle_block(output)
|
|
|
|
end
|
|
|
|
path.atomic_write(formula_ast.process)
|
2016-05-28 15:54:05 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
next if args.no_commit?
|
2021-04-29 14:48:45 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
Utils::Git.set_name_email!(committer: args.committer.blank?)
|
|
|
|
Utils::Git.setup_gpg!
|
2021-04-29 14:48:45 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
if (committer = args.committer)
|
|
|
|
committer = Utils.parse_author!(committer)
|
|
|
|
ENV["GIT_COMMITTER_NAME"] = committer[:name]
|
|
|
|
ENV["GIT_COMMITTER_EMAIL"] = committer[:email]
|
|
|
|
end
|
2021-04-29 14:48:45 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
short_name = formula_name.split("/", -1).last
|
|
|
|
pkg_version = bottle_hash["formula"]["pkg_version"]
|
2021-04-29 14:48:45 +01:00
|
|
|
|
2024-03-18 12:31:44 -07:00
|
|
|
path.parent.cd do
|
|
|
|
safe_system "git", "commit", "--no-edit", "--verbose",
|
|
|
|
"--message=#{short_name}: #{update_or_add} #{pkg_version} bottle.",
|
|
|
|
"--", path
|
|
|
|
end
|
|
|
|
end
|
2021-04-29 14:48:45 +01:00
|
|
|
end
|
|
|
|
|
2024-07-01 23:36:18 +01:00
|
|
|
sig {
|
|
|
|
params(formula: Formula, formula_ast: Utils::AST::FormulaAST,
|
|
|
|
bottle_hash: T::Hash[String, T.untyped]).returns(T.nilable(T::Array[String]))
|
|
|
|
}
|
2024-03-18 12:31:44 -07:00
|
|
|
def old_checksums(formula, formula_ast, bottle_hash)
|
2024-07-01 23:36:18 +01:00
|
|
|
bottle_node = T.cast(formula_ast.bottle_block, T.nilable(RuboCop::AST::BlockNode))
|
2024-03-18 12:31:44 -07:00
|
|
|
return if bottle_node.nil?
|
|
|
|
return [] unless args.keep_old?
|
|
|
|
|
2024-03-19 12:36:30 -07:00
|
|
|
old_keys = T.cast(Utils::AST.body_children(bottle_node.body), T::Array[RuboCop::AST::SendNode])
|
|
|
|
.map(&:method_name)
|
2024-03-18 12:31:44 -07:00
|
|
|
old_bottle_spec = formula.bottle_specification
|
|
|
|
mismatches, checksums = merge_bottle_spec(old_keys, old_bottle_spec, bottle_hash["bottle"])
|
|
|
|
if mismatches.present?
|
|
|
|
odie <<~EOS
|
|
|
|
`--keep-old` was passed but there are changes in:
|
|
|
|
#{mismatches.join("\n")}
|
|
|
|
EOS
|
|
|
|
end
|
|
|
|
checksums
|
|
|
|
end
|
2021-01-01 15:23:09 -08:00
|
|
|
end
|
|
|
|
end
|
2012-03-07 17:29:05 -05:00
|
|
|
end
|
2023-04-10 16:18:56 -04:00
|
|
|
|
|
|
|
require "extend/os/dev-cmd/bottle"
|