brew/Library/Homebrew/software_spec.rb

663 lines
18 KiB
Ruby
Raw Normal View History

2021-09-11 01:00:23 +01:00
# typed: true
# frozen_string_literal: true
require "attrable"
require "resource"
require "download_strategy"
require "checksum"
require "version"
require "options"
require "build_options"
require "dependency_collector"
2016-04-25 17:57:51 +01:00
require "utils/bottles"
require "patch"
require "compilers"
require "macos_version"
require "extend/on_system"
2013-09-14 10:16:52 -05:00
class SoftwareSpec
extend Forwardable
include OnSystem::MacOSAndLinux
2013-09-14 10:16:52 -05:00
PREDEFINED_OPTIONS = {
universal: Option.new("universal", "Build a universal binary"),
cxx11: Option.new("c++11", "Build using C++11 mode"),
}.freeze
2020-07-07 11:29:33 +01:00
attr_reader :name, :full_name, :owner, :build, :resources, :patches, :options, :deprecated_flags,
:deprecated_options, :dependency_collector, :bottle_specification, :compiler_failures
2013-09-17 21:25:39 -05:00
2020-07-07 11:29:33 +01:00
def_delegators :@resource, :stage, :fetch, :verify_download_integrity, :source_modified_time, :download_name,
:cached_download, :clear_cache, :checksum, :mirrors, :specs, :using, :version, :mirror,
2020-11-24 15:46:47 +01:00
:downloader
def_delegators :@resource, :sha256
2013-09-14 10:16:52 -05:00
def initialize(flags: [])
# Ensure this is synced with `initialize_dup` and `freeze` (excluding simple objects like integers and booleans)
@resource = Resource.new
2013-09-17 21:25:39 -05:00
@resources = {}
2013-09-21 19:27:24 -05:00
@dependency_collector = DependencyCollector.new
@bottle_specification = BottleSpecification.new
@patches = []
@options = Options.new
@flags = flags
@deprecated_flags = []
@deprecated_options = []
@build = BuildOptions.new(Options.create(@flags), options)
@compiler_failures = []
2013-09-17 21:25:39 -05:00
end
def initialize_dup(other)
super
@resource = @resource.dup
@resources = @resources.dup
@dependency_collector = @dependency_collector.dup
@bottle_specification = @bottle_specification.dup
@patches = @patches.dup
@options = @options.dup
@flags = @flags.dup
@deprecated_flags = @deprecated_flags.dup
@deprecated_options = @deprecated_options.dup
@build = @build.dup
@compiler_failures = @compiler_failures.dup
end
def freeze
@resource.freeze
@resources.freeze
@dependency_collector.freeze
@bottle_specification.freeze
@patches.freeze
@options.freeze
@flags.freeze
@deprecated_flags.freeze
@deprecated_options.freeze
@build.freeze
@compiler_failures.freeze
super
end
def owner=(owner)
2013-09-23 21:39:19 -05:00
@name = owner.name
2015-05-27 22:15:35 +08:00
@full_name = owner.full_name
@bottle_specification.tap = owner.tap
2014-02-27 14:50:22 -06:00
@owner = owner
@resource.owner = self
resources.each_value do |r|
r.owner = self
next if r.version
raise "#{full_name}: version missing for \"#{r.name}\" resource!" if version.nil?
r.version(version.head? ? Version.new("HEAD") : version.dup)
end
patches.each { |p| p.owner = self }
2013-09-17 21:25:39 -05:00
end
def url(val = nil, specs = {})
return @resource.url if val.nil?
2018-09-17 02:45:00 +02:00
@resource.url(val, **specs)
dependency_collector.add(@resource)
end
2015-11-01 20:33:24 +08:00
def bottle_defined?
!bottle_specification.collector.tags.empty?
2015-11-01 20:33:24 +08:00
end
def bottle_tag?(tag = nil)
bottle_specification.tag?(Utils::Bottles.tag(tag))
end
def bottled?(tag = nil)
2023-04-04 15:37:24 +01:00
bottle_tag?(tag) &&
(tag.present? || bottle_specification.compatible_locations? || owner.force_bottle)
end
def bottle(&block)
bottle_specification.instance_eval(&block)
end
def resource_defined?(name)
resources.key?(name)
end
def resource(name, klass = Resource, &block)
2020-11-16 22:18:56 +01:00
if block
raise DuplicateResourceError, name if resource_defined?(name)
2018-09-17 02:45:00 +02:00
res = klass.new(name, &block)
software_spec: do not add empty resources Empty resources are allowed to exist under the following form: resource "filelock" do on_linux do url "https://files.pythonhosted.org/packages/14/ec/6ee2168387ce0154632f856d5cc5592328e9cf93127c5c9aeca92c8c16cb/filelock-3.0.12.tar.gz" sha256 "18d82244ee114f543149c66a6e0c14e9c4f8a1044b5cdaadd0f82159d6a6ff59" end end In this case (or for the on_macos only resource case), just ignore the resource block. Fixes: /usr/local/Homebrew/Library/Homebrew/dependency_collector.rb:148:in `resource_dep' /usr/local/Homebrew/Library/Homebrew/dependency_collector.rb:99:in `parse_spec' /usr/local/Homebrew/Library/Homebrew/dependency_collector.rb:53:in `build' /usr/local/Homebrew/Library/Homebrew/dependency_collector.rb:40:in `block in fetch' /usr/local/Homebrew/Library/Homebrew/dependency_collector.rb:40:in `fetch' /usr/local/Homebrew/Library/Homebrew/dependency_collector.rb:40:in `fetch' /usr/local/Homebrew/Library/Homebrew/dependency_collector.rb:30:in `add' /usr/local/Homebrew/Library/Homebrew/software_spec.rb:120:in `resource' /usr/local/Homebrew/Library/Homebrew/formula.rb:2439:in `block in resource' /usr/local/Homebrew/Library/Homebrew/formula.rb:2438:in `each' /usr/local/Homebrew/Library/Homebrew/formula.rb:2438:in `resource' /usr/local/Homebrew/Library/Taps/homebrew/homebrew-core/Formula/aws-google-auth.rb:149:in `<class:AwsGoogleAuth>' /usr/local/Homebrew/Library/Taps/homebrew/homebrew-core/Formula/aws-google-auth.rb:1:in `load_formula' /usr/local/Homebrew/Library/Homebrew/formulary.rb:38:in `rescue in load_formula' /usr/local/Homebrew/Library/Homebrew/formulary.rb:35:in `load_formula' /usr/local/Homebrew/Library/Homebrew/formulary.rb:56:in `load_formula_from_path' /usr/local/Homebrew/Library/Homebrew/formulary.rb:138:in `load_file' /usr/local/Homebrew/Library/Homebrew/formulary.rb:128:in `klass' /usr/local/Homebrew/Library/Homebrew/formulary.rb:124:in `get_formula' /usr/local/Homebrew/Library/Homebrew/formulary.rb:333:in `factory' /usr/local/Homebrew/Library/Homebrew/cli/args.rb:87:in `block in formulae' /usr/local/Homebrew/Library/Homebrew/cli/args.rb:86:in `map' /usr/local/Homebrew/Library/Homebrew/cli/args.rb:86:in `formulae' /usr/local/Homebrew/Library/Homebrew/cmd/install.rb:133:in `install' /usr/local/Homebrew/Library/Homebrew/brew.rb:111:in `<main>'
2020-06-25 22:24:46 +02:00
return unless res.url
resources[name] = res
dependency_collector.add(res)
2013-09-17 21:25:39 -05:00
else
resources.fetch(name) { raise ResourceMissingError.new(owner, name) }
end
2013-09-14 10:16:52 -05:00
end
2013-09-21 19:27:24 -05:00
def go_resource(name, &block)
resource name, Resource::Go, &block
end
2014-07-31 19:37:39 -05:00
def option_defined?(name)
options.include?(name)
end
def option(name, description = "")
opt = PREDEFINED_OPTIONS.fetch(name) do
2016-09-20 22:03:08 +02:00
unless name.is_a?(String)
raise ArgumentError, "option name must be string or symbol; got a #{name.class}: #{name}"
end
raise ArgumentError, "option name is required" if name.empty?
2023-04-18 15:06:50 -07:00
raise ArgumentError, "option name must be longer than one character: #{name}" if name.length <= 1
raise ArgumentError, "option name must not start with dashes: #{name}" if name.start_with?("-")
2018-09-17 02:45:00 +02:00
Option.new(name, description)
end
options << opt
2013-09-21 19:27:24 -05:00
end
2013-09-21 19:27:24 -05:00
def deprecated_option(hash)
raise ArgumentError, "deprecated_option hash must not be empty" if hash.empty?
2018-09-17 02:45:00 +02:00
hash.each do |old_options, new_options|
Array(old_options).each do |old_option|
Array(new_options).each do |new_option|
deprecated_option = DeprecatedOption.new(old_option, new_option)
deprecated_options << deprecated_option
old_flag = deprecated_option.old_flag
new_flag = deprecated_option.current_flag
next unless @flags.include? old_flag
2018-09-17 02:45:00 +02:00
@flags -= [old_flag]
@flags |= [new_flag]
@deprecated_flags << deprecated_option
end
end
end
@build = BuildOptions.new(Options.create(@flags), options)
end
def depends_on(spec)
2013-09-21 19:27:24 -05:00
dep = dependency_collector.add(spec)
add_dep_option(dep) if dep
2013-09-21 19:27:24 -05:00
end
sig {
params(
2024-02-13 00:42:54 +01:00
dep: T.any(String, T::Hash[T.any(String, Symbol), T.any(Symbol, T::Array[Symbol])]),
bounds: T::Hash[Symbol, Symbol],
).void
}
2024-02-13 00:42:54 +01:00
def uses_from_macos(dep, bounds = {})
if dep.is_a?(Hash)
bounds = dep.dup
dep, tags = bounds.shift
2024-02-13 00:42:54 +01:00
dep = T.cast(dep, String)
tags = [*tags]
2024-02-13 00:42:54 +01:00
bounds = T.cast(bounds, T::Hash[Symbol, Symbol])
else
tags = []
end
depends_on UsesFromMacOSDependency.new(dep, tags, bounds: bounds)
end
# @deprecated
def uses_from_macos_elements
# TODO: Remember to remove the delegate from `Formula`.
odisabled "#uses_from_macos_elements", "#declared_deps"
end
# @deprecated
def uses_from_macos_names
# TODO: Remember to remove the delegate from `Formula`.
odisabled "#uses_from_macos_names", "#declared_deps"
end
2013-09-21 19:27:24 -05:00
def deps
dependency_collector.deps.dup_without_system_deps
end
def declared_deps
2013-09-21 19:27:24 -05:00
dependency_collector.deps
end
def recursive_dependencies
deps_f = []
recursive_dependencies = deps.filter_map do |dep|
deps_f << dep.to_formula
dep
rescue TapFormulaUnavailableError
# Don't complain about missing cross-tap dependencies
next
end.uniq
deps_f.compact.each do |f|
f.recursive_dependencies.each do |dep|
recursive_dependencies << dep unless recursive_dependencies.include?(dep)
end
end
recursive_dependencies
end
2013-09-21 19:27:24 -05:00
def requirements
dependency_collector.requirements
end
def recursive_requirements
Requirement.expand(self)
end
def patch(strip = :p1, src = nil, &block)
p = Patch.create(strip, src, &block)
2022-07-15 18:25:57 +02:00
return if p.is_a?(ExternalPatch) && p.url.blank?
dependency_collector.add(p.resource) if p.is_a? ExternalPatch
patches << p
end
def fails_with(compiler, &block)
compiler_failures << CompilerFailure.create(compiler, &block)
end
def needs(*standards)
standards.each do |standard|
compiler_failures.concat CompilerFailure.for_standard(standard)
end
end
def add_dep_option(dep)
dep.option_names.each do |name|
if dep.optional? && !option_defined?("with-#{name}")
options << Option.new("with-#{name}", "Build with #{name} support")
elsif dep.recommended? && !option_defined?("without-#{name}")
options << Option.new("without-#{name}", "Build without #{name} support")
end
end
end
2013-09-14 10:16:52 -05:00
end
class HeadSoftwareSpec < SoftwareSpec
def initialize(flags: [])
2013-09-14 10:16:52 -05:00
super
@resource.version(Version.new("HEAD"))
2013-09-14 10:16:52 -05:00
end
def verify_download_integrity(_filename)
# no-op
2013-09-14 10:16:52 -05:00
end
end
class Bottle
class Filename
2016-08-18 17:32:35 +01:00
attr_reader :name, :version, :tag, :rebuild
2024-01-01 17:46:48 +00:00
sig { params(formula: Formula, tag: Utils::Bottles::Tag, rebuild: Integer).returns(T.attached_class) }
2016-08-18 17:32:35 +01:00
def self.create(formula, tag, rebuild)
new(formula.name, formula.pkg_version, tag, rebuild)
end
2024-01-01 17:46:48 +00:00
sig { params(name: String, version: PkgVersion, tag: Utils::Bottles::Tag, rebuild: Integer).void }
2016-08-18 17:32:35 +01:00
def initialize(name, version, tag, rebuild)
@name = File.basename name
2024-01-01 17:46:48 +00:00
raise ArgumentError, "Invalid bottle name" unless Utils.safe_filename?(@name)
raise ArgumentError, "Invalid bottle version" unless Utils.safe_filename?(version.to_s)
@version = version
@tag = tag.to_s
2016-08-18 17:32:35 +01:00
@rebuild = rebuild
end
2020-10-20 12:03:48 +02:00
sig { returns(String) }
def to_s
2018-08-06 15:02:52 +02:00
"#{name}--#{version}#{extname}"
end
2016-09-23 18:13:48 +02:00
alias to_str to_s
2014-07-18 15:14:42 -05:00
2020-10-20 12:03:48 +02:00
sig { returns(String) }
2018-08-06 15:02:52 +02:00
def json
"#{name}--#{version}.#{tag}.bottle.json"
2014-07-18 15:14:42 -05:00
end
def url_encode
2020-05-10 00:30:32 +01:00
ERB::Util.url_encode("#{name}-#{version}#{extname}")
2018-08-06 15:02:52 +02:00
end
def github_packages
"#{name}--#{version}#{extname}"
end
2020-10-20 12:03:48 +02:00
sig { returns(String) }
2018-08-06 15:02:52 +02:00
def extname
2017-09-24 20:12:58 +01:00
s = rebuild.positive? ? ".#{rebuild}" : ""
2018-08-06 15:02:52 +02:00
".#{tag}.bottle#{s}.tar.gz"
2014-07-18 15:14:42 -05:00
end
end
extend Forwardable
attr_reader :name, :resource, :cellar, :rebuild
def_delegators :resource, :url, :verify_download_integrity
2021-07-29 21:15:40 +01:00
def_delegators :resource, :cached_download
2013-09-14 10:16:52 -05:00
2021-07-10 21:17:33 +02:00
def initialize(formula, spec, tag = nil)
2014-07-17 20:44:56 -05:00
@name = formula.name
@resource = Resource.new
@resource.owner = formula
@spec = spec
tag_spec = spec.tag_specification_for(Utils::Bottles.tag(tag))
@tag = tag_spec.tag
@cellar = tag_spec.cellar
2016-08-18 17:32:35 +01:00
@rebuild = spec.rebuild
@resource.version(formula.pkg_version.to_s)
@resource.checksum = tag_spec.checksum
@fetch_tab_retried = false
root_url(spec.root_url, spec.root_url_specs)
end
2021-07-29 21:15:40 +01:00
def fetch(verify_download_integrity: true)
@resource.fetch(verify_download_integrity: verify_download_integrity)
rescue DownloadError
raise unless fallback_on_error
fetch_tab
retry
end
def clear_cache
@resource.clear_cache
github_packages_manifest_resource&.clear_cache
@fetch_tab_retried = false
end
def compatible_locations?
@spec.compatible_locations?(tag: @tag)
end
# Does the bottle need to be relocated?
def skip_relocation?
@spec.skip_relocation?(tag: @tag)
end
def stage
resource.downloader.stage
end
def fetch_tab
return if github_packages_manifest_resource.blank?
# a checksum is used later identifying the correct tab but we do not have the checksum for the manifest/tab
github_packages_manifest_resource.fetch(verify_download_integrity: false)
begin
github_packages_manifest_resource_tab(github_packages_manifest_resource)
rescue RuntimeError => e
raise DownloadError.new(github_packages_manifest_resource, e)
end
rescue DownloadError
raise unless fallback_on_error
retry
rescue ArgumentError
raise if @fetch_tab_retried
@fetch_tab_retried = true
github_packages_manifest_resource.clear_cache
retry
end
def tab_attributes
return {} unless github_packages_manifest_resource&.downloaded?
github_packages_manifest_resource_tab(github_packages_manifest_resource)
end
private
def github_packages_manifest_resource_tab(github_packages_manifest_resource)
manifest_json = github_packages_manifest_resource.cached_download.read
json = begin
JSON.parse(manifest_json)
rescue JSON::ParserError
2022-06-28 10:09:59 +01:00
raise "The downloaded GitHub Packages manifest was corrupted or modified (it is not valid JSON): " \
"\n#{github_packages_manifest_resource.cached_download}"
end
manifests = json["manifests"]
raise ArgumentError, "Missing 'manifests' section." if manifests.blank?
manifests_annotations = manifests.filter_map { |m| m["annotations"] }
raise ArgumentError, "Missing 'annotations' section." if manifests_annotations.blank?
bottle_digest = @resource.checksum.hexdigest
image_ref = GitHubPackages.version_rebuild(@resource.version, rebuild, @tag.to_s)
manifest_annotations = manifests_annotations.find do |m|
next if m["sh.brew.bottle.digest"] != bottle_digest
m["org.opencontainers.image.ref.name"] == image_ref
end
raise ArgumentError, "Couldn't find manifest matching bottle checksum." if manifest_annotations.blank?
tab = manifest_annotations["sh.brew.tab"]
raise ArgumentError, "Couldn't find tab from manifest." if tab.blank?
begin
JSON.parse(tab)
rescue JSON::ParserError
raise ArgumentError, "Couldn't parse tab JSON."
end
end
def github_packages_manifest_resource
return if @resource.download_strategy != CurlGitHubPackagesDownloadStrategy
@github_packages_manifest_resource ||= begin
resource = Resource.new("#{name}_bottle_manifest")
version_rebuild = GitHubPackages.version_rebuild(@resource.version, rebuild)
resource.version(version_rebuild)
image_name = GitHubPackages.image_formula_name(@name)
image_tag = GitHubPackages.image_version_rebuild(version_rebuild)
resource.url(
"#{root_url}/#{image_name}/manifests/#{image_tag}",
using: CurlGitHubPackagesDownloadStrategy,
headers: ["Accept: application/vnd.oci.image.index.v1+json"],
)
T.cast(resource.downloader, CurlGitHubPackagesDownloadStrategy).resolved_basename =
"#{name}-#{version_rebuild}.bottle_manifest.json"
resource
end
end
def select_download_strategy(specs)
specs[:using] ||= DownloadStrategyDetector.detect(@root_url)
specs[:bottle] = true
specs
end
2021-07-29 21:15:40 +01:00
def fallback_on_error
# Use the default bottle domain as a fallback mirror
if @resource.url.start_with?(Homebrew::EnvConfig.bottle_domain) &&
Homebrew::EnvConfig.bottle_domain != HOMEBREW_BOTTLE_DEFAULT_DOMAIN
opoo "Bottle missing, falling back to the default domain..."
root_url(HOMEBREW_BOTTLE_DEFAULT_DOMAIN)
@github_packages_manifest_resource = nil
true
else
false
end
end
def root_url(val = nil, specs = {})
return @root_url if val.nil?
@root_url = val
filename = Filename.create(resource.owner, @tag, @spec.rebuild)
path, resolved_basename = Utils::Bottles.path_resolved_basename(val, name, resource.checksum, filename)
@resource.url("#{val}/#{path}", **select_download_strategy(specs))
@resource.downloader.resolved_basename = resolved_basename if resolved_basename.present?
end
end
class BottleSpecification
extend Attrable
RELOCATABLE_CELLARS = [:any, :any_skip_relocation].freeze
attr_rw :rebuild
attr_accessor :tap
attr_reader :collector, :root_url_specs, :repository
2020-10-20 12:03:48 +02:00
sig { void }
2013-09-14 10:16:52 -05:00
def initialize
2016-08-18 17:32:35 +01:00
@rebuild = 0
@repository = Homebrew::DEFAULT_REPOSITORY
2016-04-25 17:57:51 +01:00
@collector = Utils::Bottles::Collector.new
@root_url_specs = {}
2013-09-14 10:16:52 -05:00
end
def root_url(var = nil, specs = {})
if var.nil?
2021-07-29 21:15:40 +01:00
@root_url ||= if (github_packages_url = GitHubPackages.root_url_if_match(Homebrew::EnvConfig.bottle_domain))
github_packages_url
else
Homebrew::EnvConfig.bottle_domain
end
else
@root_url = if (github_packages_url = GitHubPackages.root_url_if_match(var))
github_packages_url
else
var
end
@root_url_specs.merge!(specs)
end
end
def ==(other)
self.class == other.class && rebuild == other.rebuild && collector == other.collector &&
root_url == other.root_url && root_url_specs == other.root_url_specs && tap == other.tap
end
alias eql? ==
2023-11-05 12:23:42 -08:00
sig { params(tag: Utils::Bottles::Tag).returns(T.any(Symbol, String)) }
def tag_to_cellar(tag = Utils::Bottles.tag)
spec = collector.specification_for(tag)
if spec.present?
2023-11-05 12:23:42 -08:00
spec.cellar
else
tag.default_cellar
end
end
sig { params(tag: Utils::Bottles::Tag).returns(T::Boolean) }
def compatible_locations?(tag: Utils::Bottles.tag)
cellar = tag_to_cellar(tag)
2020-11-29 17:51:47 +01:00
return true if RELOCATABLE_CELLARS.include?(cellar)
2023-11-05 12:23:42 -08:00
prefix = Pathname(cellar.to_s).parent.to_s
2022-05-30 04:37:09 +01:00
cellar_relocatable = cellar.size >= HOMEBREW_CELLAR.to_s.size && ENV["HOMEBREW_RELOCATE_BUILD_PREFIX"].present?
prefix_relocatable = prefix.size >= HOMEBREW_PREFIX.to_s.size && ENV["HOMEBREW_RELOCATE_BUILD_PREFIX"].present?
compatible_cellar = cellar == HOMEBREW_CELLAR.to_s || cellar_relocatable
compatible_prefix = prefix == HOMEBREW_PREFIX.to_s || prefix_relocatable
compatible_cellar && compatible_prefix
end
# Does the {Bottle} this {BottleSpecification} belongs to need to be relocated?
sig { params(tag: Utils::Bottles::Tag).returns(T::Boolean) }
def skip_relocation?(tag: Utils::Bottles.tag)
spec = collector.specification_for(tag)
spec&.cellar == :any_skip_relocation
end
sig { params(tag: T.any(Symbol, Utils::Bottles::Tag), no_older_versions: T::Boolean).returns(T::Boolean) }
def tag?(tag, no_older_versions: false)
collector.tag?(tag, no_older_versions: no_older_versions)
end
2020-11-29 17:51:47 +01:00
# Checksum methods in the DSL's bottle block take
2013-09-14 10:16:52 -05:00
# a Hash, which indicates the platform the checksum applies on.
2020-11-29 17:51:47 +01:00
# Example bottle block syntax:
# bottle do
# sha256 cellar: :any_skip_relocation, big_sur: "69489ae397e4645..."
# sha256 cellar: :any, catalina: "449de5ea35d0e94..."
2020-11-29 17:51:47 +01:00
# end
def sha256(hash)
sha256_regex = /^[a-f0-9]{64}$/i
# find new `sha256 big_sur: "69489ae397e4645..."` format
tag, digest = hash.find do |key, value|
key.is_a?(Symbol) && value.is_a?(String) && value.match?(sha256_regex)
end
cellar = hash[:cellar] if digest && tag
tag = Utils::Bottles::Tag.from_symbol(tag)
cellar ||= tag.default_cellar
collector.add(tag, checksum: Checksum.new(digest), cellar: cellar)
end
sig {
params(tag: Utils::Bottles::Tag, no_older_versions: T::Boolean)
.returns(T.nilable(Utils::Bottles::TagSpecification))
}
def tag_specification_for(tag, no_older_versions: false)
collector.specification_for(tag, no_older_versions: no_older_versions)
2013-09-14 10:16:52 -05:00
end
def checksums
tags = collector.tags.sort_by do |tag|
version = tag.to_macos_version
2021-02-04 13:30:16 -05:00
# Give arm64 bottles a higher priority so they are first
priority = (tag.arch == :arm64) ? "2" : "1"
"#{priority}.#{version}_#{tag}"
rescue MacOSVersion::Error
# Sort non-MacOS tags below MacOS tags.
"0.#{tag}"
2016-11-18 16:45:13 -08:00
end
tags.reverse.map do |tag|
spec = collector.specification_for(tag)
2020-11-29 17:51:47 +01:00
{
"tag" => spec.tag.to_sym,
"digest" => spec.checksum,
"cellar" => spec.cellar,
2020-11-29 17:51:47 +01:00
}
end
end
2013-09-14 10:16:52 -05:00
end
class PourBottleCheck
include OnSystem::MacOSAndLinux
def initialize(formula)
@formula = formula
end
def reason(reason)
@formula.pour_bottle_check_unsatisfied_reason = reason
end
def satisfy(&block)
@formula.send(:define_method, :pour_bottle?, &block)
end
end
require "extend/os/software_spec"