brew/Library/Homebrew/software_spec.rb

418 lines
10 KiB
Ruby
Raw Normal View History

require "resource"
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 "global"
require "os/mac/version"
2013-09-14 10:16:52 -05:00
class SoftwareSpec
extend Forwardable
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
2015-05-27 22:15:35 +08:00
attr_reader :name, :full_name, :owner
attr_reader :build, :resources, :patches, :options
attr_reader :deprecated_flags, :deprecated_options
2013-09-21 19:27:24 -05:00
attr_reader :dependency_collector
attr_reader :bottle_specification
attr_reader :compiler_failures
2013-09-17 21:25:39 -05:00
def_delegators :@resource, :stage, :fetch, :verify_download_integrity, :source_modified_time
def_delegators :@resource, :download_name, :cached_download, :clear_cache
def_delegators :@resource, :checksum, :mirrors, :specs, :using
def_delegators :@resource, :version, :mirror, *Checksum::TYPES
2016-07-13 10:11:59 +03:00
def_delegators :@resource, :downloader
2013-09-14 10:16:52 -05:00
def initialize
@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 = ARGV.flags_only
@deprecated_flags = []
@deprecated_options = []
@build = BuildOptions.new(Options.create(@flags), options)
@compiler_failures = []
@bottle_disable_reason = nil
2013-09-17 21:25:39 -05:00
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
r.version ||= begin
if version.nil?
raise "#{full_name}: version missing for \"#{r.name}\" resource!"
end
if version.head?
Version.create("HEAD")
else
version.dup
end
end
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-09-14 19:51:04 +08:00
def bottle_unneeded?
2016-09-20 22:24:31 +02:00
return false unless @bottle_disable_reason
2018-09-17 02:45:00 +02:00
2016-09-20 22:24:31 +02:00
@bottle_disable_reason.unneeded?
2015-09-14 19:51:04 +08:00
end
def bottle_disabled?
2016-09-20 22:24:31 +02:00
@bottle_disable_reason ? true : false
2015-09-14 19:51:04 +08:00
end
attr_reader :bottle_disable_reason
2015-09-14 19:51:04 +08:00
2015-11-01 20:33:24 +08:00
def bottle_defined?
!bottle_specification.collector.keys.empty?
2015-11-01 20:33:24 +08:00
end
def bottled?
2016-04-25 17:57:51 +01:00
bottle_specification.tag?(Utils::Bottles.tag) && \
(bottle_specification.compatible_cellar? || ARGV.force_bottle?)
end
def bottle(disable_type = nil, disable_reason = nil, &block)
2015-09-14 19:51:04 +08:00
if disable_type
@bottle_disable_reason = BottleDisableReason.new(disable_type, disable_reason)
else
bottle_specification.instance_eval(&block)
end
end
def resource_defined?(name)
resources.key?(name)
end
def resource(name, klass = Resource, &block)
2013-09-17 21:25:39 -05:00
if block_given?
raise DuplicateResourceError, name if resource_defined?(name)
2018-09-17 02:45:00 +02:00
res = klass.new(name, &block)
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?
raise ArgumentError, "option name must be longer than one character: #{name}" unless 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
def deps
dependency_collector.deps
end
def recursive_dependencies
deps_f = []
recursive_dependencies = deps.map do |dep|
begin
deps_f << dep.to_formula
dep
rescue TapFormulaUnavailableError
# Don't complain about missing cross-tap dependencies
next
end
end.compact.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)
dependency_collector.add(p.resource) if p.is_a? ExternalPatch
patches << p
end
def fails_with(compiler, &block)
odisabled "fails_with :gcc_4_0" if compiler == :gcc_4_0
odisabled "fails_with :gcc_4_2" if compiler == :gcc_4_2
odisabled "fails_with :gcc" if compiler == :gcc && !block_given?
compiler_failures << CompilerFailure.create(compiler, &block)
end
def needs(*standards)
odisabled "needs :cxx11" if standards.include?(:cxx11)
odisabled "needs :cxx14" if standards.include?(:cxx14)
standards.each do |standard|
compiler_failures.concat CompilerFailure.for_standard(standard)
end
end
def add_legacy_patches(list)
list = Patch.normalize_legacy_patches(list)
list.each { |p| p.owner = self }
patches.concat(list)
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
2013-09-14 10:16:52 -05:00
super
@resource.version = Version.create("HEAD")
2013-09-14 10:16:52 -05:00
end
def verify_download_integrity(_fn)
nil
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
2016-08-18 17:32:35 +01:00
def self.create(formula, tag, rebuild)
new(formula.name, formula.pkg_version, tag, rebuild)
end
2016-08-18 17:32:35 +01:00
def initialize(name, version, tag, rebuild)
@name = File.basename name
@version = version
@tag = tag.to_s.gsub(/_or_later$/, "")
2016-08-18 17:32:35 +01:00
@rebuild = rebuild
end
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
2018-08-06 15:02:52 +02:00
def json
"#{name}--#{version}.#{tag}.bottle.json"
2014-07-18 15:14:42 -05:00
end
2018-08-06 15:02:52 +02:00
def bintray
"#{name}-#{version}#{extname}"
end
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
2016-08-18 17:32:35 +01:00
attr_reader :name, :resource, :prefix, :cellar, :rebuild
def_delegators :resource, :url, :fetch, :verify_download_integrity
def_delegators :resource, :cached_download, :clear_cache
2013-09-14 10:16:52 -05:00
2014-07-17 20:44:56 -05:00
def initialize(formula, spec)
@name = formula.name
@resource = Resource.new
2014-07-17 20:44:56 -05:00
@resource.owner = formula
@resource.specs[:bottle] = true
@spec = spec
2016-04-25 17:57:51 +01:00
checksum, tag = spec.checksum_for(Utils::Bottles.tag)
2016-08-18 17:32:35 +01:00
filename = Filename.create(formula, tag, spec.rebuild)
2018-08-08 07:33:54 +02:00
@resource.url("#{spec.root_url}/#{filename.bintray}",
select_download_strategy(spec.root_url_specs))
2014-07-17 20:44:56 -05:00
@resource.version = formula.pkg_version
@resource.checksum = checksum
@prefix = spec.prefix
@cellar = spec.cellar
2016-08-18 17:32:35 +01:00
@rebuild = spec.rebuild
end
def compatible_cellar?
@spec.compatible_cellar?
end
# Does the bottle need to be relocated?
def skip_relocation?
@spec.skip_relocation?
end
def stage
resource.downloader.stage
end
private
def select_download_strategy(specs)
specs[:using] ||= DownloadStrategyDetector.detect(@spec.root_url)
specs
end
end
class BottleSpecification
DEFAULT_PREFIX = Homebrew::DEFAULT_PREFIX
2016-08-18 17:32:35 +01:00
attr_rw :prefix, :cellar, :rebuild
attr_accessor :tap
attr_reader :checksum, :collector, :root_url_specs
2013-09-14 10:16:52 -05:00
def initialize
2016-08-18 17:32:35 +01:00
@rebuild = 0
@prefix = Homebrew::DEFAULT_PREFIX
@cellar = Homebrew::DEFAULT_CELLAR
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?
@root_url ||= "#{HOMEBREW_BOTTLE_DOMAIN}/#{Utils::Bottles::Bintray.repository(tap)}"
else
@root_url = var
@root_url_specs.merge!(specs)
end
end
def compatible_cellar?
cellar == :any || cellar == :any_skip_relocation || cellar == HOMEBREW_CELLAR.to_s
end
# Does the {Bottle} this BottleSpecification belongs to need to be relocated?
def skip_relocation?
cellar == :any_skip_relocation
end
def tag?(tag)
2016-09-20 22:24:31 +02:00
checksum_for(tag) ? true : false
end
2013-09-14 10:16:52 -05:00
# Checksum methods in the DSL's bottle block optionally take
# a Hash, which indicates the platform the checksum applies on.
Checksum::TYPES.each do |cksum|
define_method(cksum) do |val|
digest, tag = val.shift
2014-07-15 21:55:14 -05:00
collector[tag] = Checksum.new(cksum, digest)
end
end
def checksum_for(tag)
collector.fetch_checksum_for(tag)
2013-09-14 10:16:52 -05:00
end
def checksums
2016-11-18 16:45:13 -08:00
tags = collector.keys.sort_by do |tag|
# Sort non-MacOS tags below MacOS tags.
begin
OS::Mac::Version.from_symbol tag
rescue ArgumentError
2016-11-18 16:45:13 -08:00
"0.#{tag}"
2016-09-17 17:01:04 +01:00
end
2016-11-18 16:45:13 -08:00
end
checksums = {}
tags.reverse_each do |tag|
checksum = collector[tag]
checksums[checksum.hash_type] ||= []
2016-11-18 16:45:13 -08:00
checksums[checksum.hash_type] << { checksum => tag }
end
checksums
end
2013-09-14 10:16:52 -05:00
end
class PourBottleCheck
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"