2009-10-15 09:07:12 +01:00
|
|
|
require 'download_strategy'
|
2013-01-26 20:05:39 -06:00
|
|
|
require 'dependency_collector'
|
2012-02-04 18:45:08 -08:00
|
|
|
require 'formula_support'
|
2013-02-09 18:19:50 -06:00
|
|
|
require 'formula_lock'
|
2013-03-11 16:41:08 +01:00
|
|
|
require 'formula_pin'
|
2012-02-25 15:22:32 -08:00
|
|
|
require 'hardware'
|
2012-03-07 21:30:03 -05:00
|
|
|
require 'bottles'
|
2012-03-09 10:56:45 -08:00
|
|
|
require 'patches'
|
2012-03-18 13:58:13 -05:00
|
|
|
require 'compilers'
|
2012-09-14 07:54:14 -07:00
|
|
|
require 'build_environment'
|
2013-01-23 00:26:23 -06:00
|
|
|
require 'build_options'
|
2011-03-21 14:23:28 -07:00
|
|
|
|
2012-04-05 21:09:24 -05:00
|
|
|
|
2009-08-21 20:20:44 +01:00
|
|
|
class Formula
|
2012-03-06 19:23:51 -06:00
|
|
|
include FileUtils
|
2013-01-19 20:45:57 -06:00
|
|
|
extend BuildEnvironmentDSL
|
2010-07-04 14:17:03 -07:00
|
|
|
|
2012-04-05 21:09:24 -05:00
|
|
|
attr_reader :name, :path, :homepage, :downloader
|
|
|
|
attr_reader :stable, :bottle, :devel, :head, :active_spec
|
2010-06-15 12:35:55 -07:00
|
|
|
|
2012-11-01 14:40:59 -05:00
|
|
|
# The current working directory during builds and tests.
|
|
|
|
# Will only be non-nil inside #stage and #test.
|
|
|
|
attr_reader :buildpath, :testpath
|
2012-02-24 12:50:21 -08:00
|
|
|
|
2009-08-21 20:20:44 +01:00
|
|
|
# Homebrew determines the name
|
2010-09-22 07:54:53 -07:00
|
|
|
def initialize name='__UNKNOWN__', path=nil
|
2012-03-09 10:18:12 -08:00
|
|
|
@name = name
|
2013-04-13 17:40:12 -05:00
|
|
|
# If we got an explicit path, use that, else determine from the name
|
|
|
|
@path = path.nil? ? self.class.path(name) : Pathname.new(path)
|
|
|
|
@homepage = self.class.homepage
|
|
|
|
|
|
|
|
set_spec :stable
|
|
|
|
set_spec :devel
|
|
|
|
set_spec :head
|
|
|
|
set_spec :bottle do |bottle|
|
|
|
|
# Ensure the bottle URL is set. If it does not have a checksum,
|
|
|
|
# then a bottle is not available for the current platform.
|
|
|
|
# TODO: push this down into Bottle; we can pass the formula instance
|
|
|
|
# into a validation method on the bottle instance.
|
|
|
|
unless bottle.checksum.nil? || bottle.checksum.empty?
|
|
|
|
@bottle = bottle
|
|
|
|
bottle.url ||= bottle_url(self)
|
2013-01-27 18:41:06 +00:00
|
|
|
end
|
2012-04-05 21:09:24 -05:00
|
|
|
end
|
|
|
|
|
2013-04-13 17:40:12 -05:00
|
|
|
@active_spec = determine_active_spec
|
|
|
|
validate_attributes :url, :name, :version
|
2012-06-26 01:35:37 -05:00
|
|
|
@downloader = download_strategy.new(name, @active_spec)
|
2012-07-30 11:32:56 -07:00
|
|
|
|
|
|
|
# Combine DSL `option` and `def options`
|
2012-08-11 16:51:05 -05:00
|
|
|
options.each do |opt, desc|
|
|
|
|
# make sure to strip "--" from the start of options
|
|
|
|
self.class.build.add opt[/--(.+)$/, 1], desc
|
|
|
|
end
|
2013-03-11 16:41:08 +01:00
|
|
|
|
|
|
|
@pin = FormulaPin.new(self)
|
2012-04-05 21:09:24 -05:00
|
|
|
end
|
|
|
|
|
2013-04-13 17:40:12 -05:00
|
|
|
def set_spec(name)
|
|
|
|
spec = self.class.send(name)
|
|
|
|
return if spec.nil?
|
|
|
|
if block_given? && yield(spec) || !spec.url.nil?
|
|
|
|
instance_variable_set("@#{name}", spec)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def determine_active_spec
|
|
|
|
case
|
|
|
|
when @head && ARGV.build_head? then @head # --HEAD
|
|
|
|
when @devel && ARGV.build_devel? then @devel # --devel
|
|
|
|
when @bottle && install_bottle?(self) then @bottle # bottle available
|
2013-04-13 17:40:14 -05:00
|
|
|
when @stable then @stable
|
2013-04-13 17:40:13 -05:00
|
|
|
when @devel && @stable.nil? then @devel # devel-only
|
|
|
|
when @head && @stable.nil? then @head # head-only
|
2013-04-13 17:40:14 -05:00
|
|
|
else
|
2013-04-27 14:44:48 -05:00
|
|
|
raise FormulaSpecificationError, "formulae require at least a URL"
|
2013-04-13 17:40:12 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def validate_attributes(*attrs)
|
|
|
|
attrs.each do |attr|
|
|
|
|
if (value = send(attr).to_s).empty? || value =~ /\s/
|
|
|
|
raise FormulaValidationError.new(attr, value)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def url; active_spec.url; end
|
|
|
|
def version; active_spec.version; end
|
|
|
|
def mirrors; active_spec.mirrors; end
|
2012-04-05 21:09:24 -05:00
|
|
|
|
2009-07-24 15:10:01 +01:00
|
|
|
# if the dir is there, but it's empty we consider it not installed
|
|
|
|
def installed?
|
2013-04-09 19:40:08 -05:00
|
|
|
(dir = installed_prefix).directory? && dir.children.length > 0
|
2009-07-24 15:10:01 +01:00
|
|
|
end
|
|
|
|
|
2012-01-14 20:03:30 -06:00
|
|
|
def linked_keg
|
2013-04-15 15:00:57 -05:00
|
|
|
Pathname.new("#{HOMEBREW_LIBRARY}/LinkedKegs/#{name}")
|
2012-01-14 20:03:30 -06:00
|
|
|
end
|
|
|
|
|
2010-09-24 11:55:21 -05:00
|
|
|
def installed_prefix
|
2013-05-25 18:31:49 -05:00
|
|
|
if head && (head_prefix = prefix(head.version)).directory?
|
2010-09-24 11:55:21 -05:00
|
|
|
head_prefix
|
2013-05-25 18:31:49 -05:00
|
|
|
elsif devel && (devel_prefix = prefix(devel.version)).directory?
|
2012-04-05 21:09:24 -05:00
|
|
|
devel_prefix
|
2010-09-24 11:55:21 -05:00
|
|
|
else
|
|
|
|
prefix
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-08-18 18:37:32 -05:00
|
|
|
def installed_version
|
|
|
|
require 'keg'
|
|
|
|
Keg.new(installed_prefix).version
|
|
|
|
end
|
|
|
|
|
2013-05-25 18:31:49 -05:00
|
|
|
def prefix(v=version)
|
|
|
|
Pathname.new("#{HOMEBREW_CELLAR}/#{name}/#{v}")
|
2010-02-19 21:55:17 -08:00
|
|
|
end
|
2011-09-11 12:57:53 -07:00
|
|
|
def rack; prefix.parent end
|
2010-02-19 21:55:17 -08:00
|
|
|
|
2012-10-28 08:54:54 -07:00
|
|
|
def bin; prefix+'bin' end
|
|
|
|
def doc; share+'doc'+name end
|
|
|
|
def include; prefix+'include' end
|
|
|
|
def info; share+'info' end
|
|
|
|
def lib; prefix+'lib' end
|
|
|
|
def libexec; prefix+'libexec' end
|
|
|
|
def man; share+'man' end
|
|
|
|
def man1; man+'man1' end
|
|
|
|
def man2; man+'man2' end
|
|
|
|
def man3; man+'man3' end
|
|
|
|
def man4; man+'man4' end
|
|
|
|
def man5; man+'man5' end
|
|
|
|
def man6; man+'man6' end
|
|
|
|
def man7; man+'man7' end
|
|
|
|
def man8; man+'man8' end
|
|
|
|
def sbin; prefix+'sbin' end
|
|
|
|
def share; prefix+'share' end
|
2009-10-02 15:55:34 +01:00
|
|
|
|
2013-05-03 09:28:45 -07:00
|
|
|
def frameworks; prefix+'Frameworks' end
|
2013-04-21 10:44:24 +02:00
|
|
|
def kext_prefix; prefix+'Library/Extensions' end
|
|
|
|
|
2009-10-03 15:23:28 +01:00
|
|
|
# configuration needs to be preserved past upgrades
|
2009-10-02 15:55:34 +01:00
|
|
|
def etc; HOMEBREW_PREFIX+'etc' end
|
2010-07-04 14:17:03 -07:00
|
|
|
# generally we don't want var stuff inside the keg
|
|
|
|
def var; HOMEBREW_PREFIX+'var' end
|
2010-04-06 13:13:50 -07:00
|
|
|
|
2013-01-27 22:34:26 +00:00
|
|
|
def bash_completion; prefix+'etc/bash_completion.d' end
|
|
|
|
def zsh_completion; share+'zsh/site-functions' end
|
|
|
|
|
2012-09-08 12:18:52 -07:00
|
|
|
# override this to provide a plist
|
2012-11-25 15:06:41 +00:00
|
|
|
def plist; nil; end
|
|
|
|
alias :startup_plist :plist
|
2011-12-30 23:56:52 -06:00
|
|
|
# plist name, i.e. the name of the launchd service
|
|
|
|
def plist_name; 'homebrew.mxcl.'+name end
|
|
|
|
def plist_path; prefix+(plist_name+'.plist') end
|
2012-11-25 15:06:41 +00:00
|
|
|
def plist_manual; self.class.plist_manual end
|
|
|
|
def plist_startup; self.class.plist_startup end
|
2011-12-30 23:56:52 -06:00
|
|
|
|
2013-02-11 20:24:06 -08:00
|
|
|
# Defined and active build-time options.
|
|
|
|
def build; self.class.build; end
|
2012-07-30 11:32:56 -07:00
|
|
|
|
2013-04-15 15:00:57 -05:00
|
|
|
def opt_prefix
|
|
|
|
Pathname.new("#{HOMEBREW_PREFIX}/opt/#{name}")
|
|
|
|
end
|
2012-08-10 16:33:22 -04:00
|
|
|
|
2010-07-04 14:17:03 -07:00
|
|
|
def download_strategy
|
2013-04-13 17:40:12 -05:00
|
|
|
active_spec.download_strategy
|
2010-07-04 14:17:03 -07:00
|
|
|
end
|
2010-03-17 16:56:06 -07:00
|
|
|
|
2010-07-04 14:17:03 -07:00
|
|
|
def cached_download
|
2013-04-13 17:40:12 -05:00
|
|
|
downloader.cached_location
|
2009-08-11 12:20:55 -07:00
|
|
|
end
|
2009-09-28 14:06:53 -07:00
|
|
|
|
2013-01-27 19:22:56 +00:00
|
|
|
# Can be overridden to selectively disable bottles from formulae.
|
|
|
|
# Defaults to true so overridden version does not have to check if bottles
|
|
|
|
# are supported.
|
2013-02-11 20:24:06 -08:00
|
|
|
def pour_bottle?; true end
|
2013-01-27 19:22:56 +00:00
|
|
|
|
2013-04-01 20:33:12 +01:00
|
|
|
# Can be overridden to run commands on both source and bottle installation.
|
|
|
|
def post_install; end
|
|
|
|
|
2009-09-28 14:06:53 -07:00
|
|
|
# tell the user about any caveats regarding this package, return a string
|
2009-08-10 16:48:30 +01:00
|
|
|
def caveats; nil end
|
2009-09-28 14:06:53 -07:00
|
|
|
|
2011-06-21 22:28:29 +01:00
|
|
|
# any e.g. configure options for this package
|
2011-06-22 17:47:30 +01:00
|
|
|
def options; [] end
|
2011-06-21 22:28:29 +01:00
|
|
|
|
2009-08-10 16:48:30 +01:00
|
|
|
# patches are automatically applied after extracting the tarball
|
2009-11-13 14:27:21 -05:00
|
|
|
# return an array of strings, or if you need a patch level other than -p1
|
2009-08-11 18:16:12 +01:00
|
|
|
# return a Hash eg.
|
|
|
|
# {
|
|
|
|
# :p0 => ['http://foo.com/patch1', 'http://foo.com/patch2'],
|
2013-02-11 20:24:06 -08:00
|
|
|
# :p1 => 'http://bar.com/patch2'
|
2009-08-11 18:16:12 +01:00
|
|
|
# }
|
2009-09-17 21:10:39 +01:00
|
|
|
# The final option is to return DATA, then put a diff after __END__. You
|
2009-09-04 15:28:18 +01:00
|
|
|
# can still return a Hash with DATA as the value for a patch level key.
|
2009-09-21 23:51:31 +01:00
|
|
|
def patches; end
|
2009-09-28 14:06:53 -07:00
|
|
|
|
2009-09-17 21:10:39 +01:00
|
|
|
# rarely, you don't want your library symlinked into the main prefix
|
|
|
|
# see gettext.rb for an example
|
2010-07-18 10:38:45 -07:00
|
|
|
def keg_only?
|
2012-08-09 02:00:58 -05:00
|
|
|
kor = self.class.keg_only_reason
|
|
|
|
not kor.nil? and kor.valid?
|
|
|
|
end
|
|
|
|
|
|
|
|
def keg_only_reason
|
|
|
|
self.class.keg_only_reason
|
2010-07-18 10:38:45 -07:00
|
|
|
end
|
2009-08-10 16:48:30 +01:00
|
|
|
|
2012-03-18 13:58:13 -05:00
|
|
|
def fails_with? cc
|
|
|
|
cc = Compiler.new(cc) unless cc.is_a? Compiler
|
2013-03-13 02:07:01 -05:00
|
|
|
(self.class.cc_failures || []).any? do |failure|
|
2013-03-11 22:54:15 -05:00
|
|
|
failure.compiler == cc.name && failure.build >= cc.build
|
2011-08-26 17:08:42 +01:00
|
|
|
end
|
2011-03-21 14:23:28 -07:00
|
|
|
end
|
|
|
|
|
2009-09-28 14:06:53 -07:00
|
|
|
# sometimes the clean process breaks things
|
|
|
|
# skip cleaning paths in a formula with a class method like this:
|
|
|
|
# skip_clean [bin+"foo", lib+"bar"]
|
2011-12-17 17:12:19 -08:00
|
|
|
# redefining skip_clean? now deprecated
|
2009-09-28 14:06:53 -07:00
|
|
|
def skip_clean? path
|
2010-08-09 00:30:51 -05:00
|
|
|
return true if self.class.skip_clean_all?
|
2012-09-09 10:01:59 -07:00
|
|
|
return true if path.extname == '.la' and self.class.skip_clean_paths.include? :la
|
2009-09-28 14:06:53 -07:00
|
|
|
to_check = path.relative_path_from(prefix).to_s
|
2009-11-23 10:07:23 -08:00
|
|
|
self.class.skip_clean_paths.include? to_check
|
2009-09-28 14:06:53 -07:00
|
|
|
end
|
|
|
|
|
2009-08-10 16:48:30 +01:00
|
|
|
# yields self with current working directory set to the uncompressed tarball
|
|
|
|
def brew
|
2013-04-13 17:40:12 -05:00
|
|
|
validate_attributes :name, :version
|
Dependency resolution
Specify dependencies in your formula's deps function. You can return an Array,
String or Hash, eg:
def deps
{ :optional => 'libogg', :required => %w[flac sdl], :recommended => 'cmake' }
end
Note currently the Hash is flattened and qualifications are ignored. If you
only return an Array or String, the qualification is assumed to be :required.
Other packaging systems have problems when it comes to packages requiring a
specific version of a package, or some patches that may not work well with
other software. With Homebrew we have some options:
1. If the formula is vanilla but an older version we can cherry-pick the old
version and install it in the Cellar in parallel, but just not symlink it
into /usr/local while forcing the formula that depends on it to link to
that one and not any other versions of it.
2. If the dependency requires patches then we shouldn't install this for use
by any other tools, (I guess this needs to be decided on a per-situation
basis). It can be installed into the parent formula's prefix, and not
symlinked into /usr/local. In this case the dependency's Formula
derivation should be saved in the parent formula's file (check git or
flac for an example of this).
Both the above can be done currently with hacks, so I'll flesh out a proper
way sometime this week.
2009-09-07 01:06:08 +01:00
|
|
|
|
2009-08-11 12:20:55 -07:00
|
|
|
stage do
|
2009-08-10 16:48:30 +01:00
|
|
|
begin
|
|
|
|
patch
|
2013-01-26 13:16:55 +00:00
|
|
|
# we allow formulae to do anything they want to the Ruby process
|
2009-09-04 15:28:18 +01:00
|
|
|
# so load any deps before this point! And exit asap afterwards
|
2009-08-10 16:48:30 +01:00
|
|
|
yield self
|
2013-02-17 22:54:27 -06:00
|
|
|
rescue RuntimeError, SystemCallError
|
2012-02-21 01:14:02 -06:00
|
|
|
%w(config.log CMakeCache.txt).each do |fn|
|
|
|
|
(HOMEBREW_LOGS/name).install(fn) if File.file?(fn)
|
2011-08-31 11:07:27 +01:00
|
|
|
end
|
2012-02-21 01:14:02 -06:00
|
|
|
raise
|
2009-08-10 16:48:30 +01:00
|
|
|
end
|
|
|
|
end
|
2009-07-24 15:10:01 +01:00
|
|
|
end
|
2009-08-10 16:48:30 +01:00
|
|
|
|
2013-01-23 00:26:25 -06:00
|
|
|
def lock
|
2013-02-09 18:19:50 -06:00
|
|
|
@lock = FormulaLock.new(name)
|
|
|
|
@lock.lock
|
2013-01-23 00:26:25 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
def unlock
|
2013-02-09 18:19:50 -06:00
|
|
|
@lock.unlock unless @lock.nil?
|
2013-01-23 00:26:25 -06:00
|
|
|
end
|
|
|
|
|
2013-05-21 22:41:21 -05:00
|
|
|
def pinnable?
|
|
|
|
@pin.pinnable?
|
|
|
|
end
|
|
|
|
|
|
|
|
def pinned?
|
|
|
|
@pin.pinned?
|
|
|
|
end
|
|
|
|
|
|
|
|
def pin
|
|
|
|
@pin.pin
|
|
|
|
end
|
|
|
|
|
|
|
|
def unpin
|
|
|
|
@pin.unpin
|
|
|
|
end
|
|
|
|
|
2010-09-11 20:22:54 +01:00
|
|
|
def == b
|
|
|
|
name == b.name
|
|
|
|
end
|
|
|
|
def eql? b
|
|
|
|
self == b and self.class.equal? b.class
|
|
|
|
end
|
|
|
|
def hash
|
|
|
|
name.hash
|
|
|
|
end
|
|
|
|
def <=> b
|
|
|
|
name <=> b.name
|
|
|
|
end
|
|
|
|
def to_s
|
|
|
|
name
|
|
|
|
end
|
2013-03-20 22:23:36 -05:00
|
|
|
def inspect
|
|
|
|
name
|
|
|
|
end
|
2010-09-11 20:22:54 +01:00
|
|
|
|
2010-09-22 08:05:58 -07:00
|
|
|
# Standard parameters for CMake builds.
|
|
|
|
# Using Build Type "None" tells cmake to use our CFLAGS,etc. settings.
|
|
|
|
# Setting it to Release would ignore our flags.
|
2012-05-29 16:58:32 -07:00
|
|
|
# Setting CMAKE_FIND_FRAMEWORK to "LAST" tells CMake to search for our
|
|
|
|
# libraries before trying to utilize Frameworks, many of which will be from
|
|
|
|
# 3rd party installs.
|
2010-09-22 08:05:58 -07:00
|
|
|
# Note: there isn't a std_autotools variant because autotools is a lot
|
|
|
|
# less consistent and the standard parameters are more memorable.
|
2012-05-22 16:32:12 -05:00
|
|
|
def std_cmake_args
|
2012-05-29 16:58:32 -07:00
|
|
|
%W[
|
|
|
|
-DCMAKE_INSTALL_PREFIX=#{prefix}
|
|
|
|
-DCMAKE_BUILD_TYPE=None
|
|
|
|
-DCMAKE_FIND_FRAMEWORK=LAST
|
|
|
|
-Wno-dev
|
|
|
|
]
|
2009-08-21 20:20:44 +01:00
|
|
|
end
|
|
|
|
|
2013-01-21 10:33:56 +01:00
|
|
|
def python(options={:allowed_major_versions => [2, 3]}, &block)
|
|
|
|
require 'python_helper'
|
2013-06-03 16:59:11 -05:00
|
|
|
python_helper(options, &block)
|
2013-01-21 10:33:56 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
# Explicitly only execute the block for 2.x (if a python 2.x is available)
|
|
|
|
def python2 &block
|
|
|
|
python(:allowed_major_versions => [2], &block)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Explicitly only execute the block for 3.x (if a python 3.x is available)
|
|
|
|
def python3 &block
|
|
|
|
python(:allowed_major_versions => [3], &block)
|
|
|
|
end
|
|
|
|
|
2009-09-11 14:22:46 +01:00
|
|
|
def self.class_s name
|
2013-02-11 20:24:06 -08:00
|
|
|
# remove invalid characters and then camelcase it
|
2009-10-19 13:05:47 +01:00
|
|
|
name.capitalize.gsub(/[-_.\s]([a-zA-Z0-9])/) { $1.upcase } \
|
|
|
|
.gsub('+', 'x')
|
2009-08-21 20:20:44 +01:00
|
|
|
end
|
2010-03-09 02:18:08 +00:00
|
|
|
|
|
|
|
# an array of all Formula names
|
|
|
|
def self.names
|
|
|
|
Dir["#{HOMEBREW_REPOSITORY}/Library/Formula/*.rb"].map{ |f| File.basename f, '.rb' }.sort
|
|
|
|
end
|
|
|
|
|
2010-09-11 20:22:54 +01:00
|
|
|
def self.each
|
2012-08-21 11:39:45 -04:00
|
|
|
names.each do |name|
|
2013-04-14 23:29:15 -05:00
|
|
|
begin
|
|
|
|
yield Formula.factory(name)
|
2013-02-17 22:54:27 -06:00
|
|
|
rescue
|
2011-03-11 14:54:26 -08:00
|
|
|
# Don't let one broken formula break commands. But do complain.
|
2012-08-21 11:39:45 -04:00
|
|
|
onoe "Failed to import: #{name}"
|
|
|
|
next
|
2010-07-18 14:07:40 -07:00
|
|
|
end
|
|
|
|
end
|
2010-09-11 20:22:54 +01:00
|
|
|
end
|
2012-08-21 11:39:45 -04:00
|
|
|
class << self
|
|
|
|
include Enumerable
|
|
|
|
end
|
2012-08-10 16:05:30 -04:00
|
|
|
|
|
|
|
def self.installed
|
|
|
|
HOMEBREW_CELLAR.children.map{ |rack| factory(rack.basename) rescue nil }.compact
|
|
|
|
end
|
|
|
|
|
2010-03-09 02:18:08 +00:00
|
|
|
def self.aliases
|
|
|
|
Dir["#{HOMEBREW_REPOSITORY}/Library/Aliases/*"].map{ |f| File.basename f }.sort
|
2009-11-16 15:35:58 -08:00
|
|
|
end
|
2009-08-21 20:20:44 +01:00
|
|
|
|
2011-05-07 19:29:54 -05:00
|
|
|
def self.canonical_name name
|
2011-12-26 11:52:33 -08:00
|
|
|
name = name.to_s if name.kind_of? Pathname
|
|
|
|
|
2013-04-15 15:00:57 -05:00
|
|
|
formula_with_that_name = Pathname.new("#{HOMEBREW_REPOSITORY}/Library/Formula/#{name}.rb")
|
|
|
|
possible_alias = Pathname.new("#{HOMEBREW_REPOSITORY}/Library/Aliases/#{name}")
|
|
|
|
possible_cached_formula = Pathname.new("#{HOMEBREW_CACHE_FORMULA}/#{name}.rb")
|
2011-06-08 11:13:50 -07:00
|
|
|
|
2010-11-05 13:44:24 +00:00
|
|
|
if name.include? "/"
|
2012-03-04 14:17:54 +00:00
|
|
|
if name =~ %r{(.+)/(.+)/(.+)}
|
2013-04-15 15:00:57 -05:00
|
|
|
tap_name = "#$1-#$2".downcase
|
|
|
|
tapd = Pathname.new("#{HOMEBREW_REPOSITORY}/Library/Taps/#{tap_name}")
|
2012-03-04 14:17:54 +00:00
|
|
|
tapd.find_formula do |relative_pathname|
|
|
|
|
return "#{tapd}/#{relative_pathname}" if relative_pathname.stem.to_s == $3
|
|
|
|
end if tapd.directory?
|
|
|
|
end
|
|
|
|
# Otherwise don't resolve paths or URLs
|
2010-11-05 13:44:24 +00:00
|
|
|
name
|
|
|
|
elsif formula_with_that_name.file? and formula_with_that_name.readable?
|
|
|
|
name
|
|
|
|
elsif possible_alias.file?
|
|
|
|
possible_alias.realpath.basename('.rb').to_s
|
2011-06-08 11:13:50 -07:00
|
|
|
elsif possible_cached_formula.file?
|
|
|
|
possible_cached_formula.to_s
|
2010-09-22 12:32:16 -07:00
|
|
|
else
|
|
|
|
name
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2009-08-21 20:20:44 +01:00
|
|
|
def self.factory name
|
2010-09-22 07:54:53 -07:00
|
|
|
# If an instance of Formula is passed, just return it
|
2009-09-18 19:16:39 +01:00
|
|
|
return name if name.kind_of? Formula
|
2010-09-22 07:54:53 -07:00
|
|
|
|
2012-02-17 21:18:37 +01:00
|
|
|
# Otherwise, convert to String in case a Pathname comes in
|
|
|
|
name = name.to_s
|
|
|
|
|
2010-09-22 07:54:53 -07:00
|
|
|
# If a URL is passed, download to the cache and install
|
|
|
|
if name =~ %r[(https?|ftp)://]
|
|
|
|
url = name
|
|
|
|
name = Pathname.new(name).basename
|
2012-08-23 14:33:33 -05:00
|
|
|
path = HOMEBREW_CACHE_FORMULA+name
|
2010-09-22 07:54:53 -07:00
|
|
|
name = name.basename(".rb").to_s
|
|
|
|
|
2012-08-23 14:33:33 -05:00
|
|
|
unless Object.const_defined? self.class_s(name)
|
|
|
|
HOMEBREW_CACHE_FORMULA.mkpath
|
|
|
|
FileUtils.rm path, :force => true
|
|
|
|
curl url, '-o', path
|
|
|
|
end
|
2010-09-22 07:54:53 -07:00
|
|
|
|
|
|
|
install_type = :from_url
|
2012-11-25 14:33:52 +00:00
|
|
|
elsif name.match bottle_regex
|
|
|
|
bottle_filename = Pathname(name).realpath
|
2013-01-30 21:47:08 -08:00
|
|
|
version = Version.parse(bottle_filename).to_s
|
|
|
|
name = bottle_filename.basename.to_s.rpartition("-#{version}").first
|
2012-11-25 14:33:52 +00:00
|
|
|
path = Formula.path(name)
|
|
|
|
install_type = :from_local_bottle
|
2009-09-04 15:28:18 +01:00
|
|
|
else
|
2011-06-08 11:13:50 -07:00
|
|
|
name = Formula.canonical_name(name)
|
2012-05-15 02:27:11 -04:00
|
|
|
|
2012-10-09 14:45:53 -05:00
|
|
|
if name =~ %r{^(\w+)/(\w+)/([^/])+$}
|
|
|
|
# name appears to be a tapped formula, so we don't munge it
|
|
|
|
# in order to provide a useful error message when require fails.
|
|
|
|
path = Pathname.new(name)
|
|
|
|
elsif name.include? "/"
|
|
|
|
# If name was a path or mapped to a cached formula
|
|
|
|
|
2012-05-15 02:27:11 -04:00
|
|
|
# require allows filenames to drop the .rb extension, but everything else
|
|
|
|
# in our codebase will require an exact and fullpath.
|
|
|
|
name = "#{name}.rb" unless name =~ /\.rb$/
|
|
|
|
|
2011-03-12 16:37:56 -08:00
|
|
|
path = Pathname.new(name)
|
2010-09-22 07:54:53 -07:00
|
|
|
name = path.stem
|
|
|
|
install_type = :from_path
|
|
|
|
else
|
|
|
|
# For names, map to the path and then require
|
2012-08-23 14:33:33 -05:00
|
|
|
path = Formula.path(name)
|
2010-09-22 07:54:53 -07:00
|
|
|
install_type = :from_name
|
|
|
|
end
|
2009-09-04 15:28:18 +01:00
|
|
|
end
|
2010-09-22 07:54:53 -07:00
|
|
|
|
2012-08-23 14:33:33 -05:00
|
|
|
klass_name = self.class_s(name)
|
|
|
|
unless Object.const_defined? klass_name
|
|
|
|
puts "#{$0}: loading #{path}" if ARGV.debug?
|
|
|
|
require path
|
|
|
|
end
|
|
|
|
|
2009-09-23 10:06:19 -07:00
|
|
|
begin
|
2011-03-20 22:37:25 +01:00
|
|
|
klass = Object.const_get klass_name
|
2011-06-08 11:34:01 -07:00
|
|
|
rescue NameError
|
2009-09-23 10:06:19 -07:00
|
|
|
# TODO really this text should be encoded into the exception
|
|
|
|
# and only shown if the UI deems it correct to show it
|
|
|
|
onoe "class \"#{klass_name}\" expected but not found in #{name}.rb"
|
|
|
|
puts "Double-check the name of the class in that formula."
|
|
|
|
raise LoadError
|
|
|
|
end
|
2010-09-22 07:54:53 -07:00
|
|
|
|
2012-11-25 14:33:52 +00:00
|
|
|
if install_type == :from_local_bottle
|
|
|
|
formula = klass.new(name)
|
|
|
|
formula.downloader.local_bottle_path = bottle_filename
|
|
|
|
return formula
|
|
|
|
end
|
|
|
|
|
2012-11-28 10:58:29 -06:00
|
|
|
raise NameError if !klass.ancestors.include? Formula
|
2013-02-01 09:25:47 -08:00
|
|
|
raise NameError if klass == Formula
|
2012-11-28 10:00:40 -06:00
|
|
|
|
2010-09-22 07:54:53 -07:00
|
|
|
return klass.new(name) if install_type == :from_name
|
2012-08-23 14:33:33 -05:00
|
|
|
return klass.new(name, path.to_s)
|
2012-08-26 17:06:38 -07:00
|
|
|
rescue NoMethodError
|
|
|
|
# This is a programming error in an existing formula, and should not
|
|
|
|
# have a "no such formula" message.
|
|
|
|
raise
|
2012-08-21 15:51:59 -05:00
|
|
|
rescue LoadError, NameError
|
2012-08-26 17:08:09 -07:00
|
|
|
# Catch NameError so that things that are invalid symbols still get
|
|
|
|
# a useful error message.
|
2009-08-21 20:20:44 +01:00
|
|
|
raise FormulaUnavailableError.new(name)
|
|
|
|
end
|
|
|
|
|
2012-03-16 12:29:47 +00:00
|
|
|
def tap
|
|
|
|
if path.realpath.to_s =~ %r{#{HOMEBREW_REPOSITORY}/Library/Taps/(\w+)-(\w+)}
|
|
|
|
"#$1/#$2"
|
|
|
|
else
|
|
|
|
# remotely installed formula are not mxcl/master but this will do for now
|
|
|
|
"mxcl/master"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2009-08-21 20:20:44 +01:00
|
|
|
def self.path name
|
2013-04-15 15:00:57 -05:00
|
|
|
Pathname.new("#{HOMEBREW_REPOSITORY}/Library/Formula/#{name.downcase}.rb")
|
2009-08-21 20:20:44 +01:00
|
|
|
end
|
|
|
|
|
2012-07-23 17:06:38 -07:00
|
|
|
def deps; self.class.dependencies.deps; end
|
|
|
|
def requirements; self.class.dependencies.requirements; end
|
2010-09-11 20:22:54 +01:00
|
|
|
|
2012-09-14 07:54:14 -07:00
|
|
|
def env
|
2012-12-23 19:43:10 -06:00
|
|
|
@env ||= self.class.env
|
2012-09-14 07:54:14 -07:00
|
|
|
end
|
|
|
|
|
2012-08-07 14:22:20 -05:00
|
|
|
def conflicts
|
2013-05-10 23:45:06 -05:00
|
|
|
requirements.grep(ConflictRequirement)
|
2012-08-07 14:22:20 -05:00
|
|
|
end
|
|
|
|
|
2013-01-23 00:26:26 -06:00
|
|
|
# Returns a list of Dependency objects in an installable order, which
|
|
|
|
# means if a depends on b then b will be ordered before a in this list
|
|
|
|
def recursive_dependencies(&block)
|
|
|
|
Dependency.expand(self, &block)
|
2010-01-13 09:00:24 +00:00
|
|
|
end
|
|
|
|
|
2013-01-23 00:26:24 -06:00
|
|
|
# The full set of Requirements for this formula's dependency tree.
|
2013-01-30 17:55:04 -06:00
|
|
|
def recursive_requirements(&block)
|
|
|
|
Requirement.expand(self, &block)
|
2012-08-14 21:45:08 -05:00
|
|
|
end
|
|
|
|
|
2012-08-15 22:08:40 -05:00
|
|
|
def to_hash
|
|
|
|
hsh = {
|
|
|
|
"name" => name,
|
|
|
|
"homepage" => homepage,
|
|
|
|
"versions" => {
|
|
|
|
"stable" => (stable.version.to_s if stable),
|
2013-05-05 10:55:39 -05:00
|
|
|
"bottle" => bottle ? true : false,
|
2012-08-15 22:08:40 -05:00
|
|
|
"devel" => (devel.version.to_s if devel),
|
|
|
|
"head" => (head.version.to_s if head)
|
|
|
|
},
|
|
|
|
"installed" => [],
|
|
|
|
"linked_keg" => (linked_keg.realpath.basename.to_s if linked_keg.exist?),
|
|
|
|
"keg_only" => keg_only?,
|
|
|
|
"dependencies" => deps.map {|dep| dep.to_s},
|
|
|
|
"conflicts_with" => conflicts.map {|c| c.formula},
|
|
|
|
"options" => [],
|
|
|
|
"caveats" => caveats
|
|
|
|
}
|
|
|
|
|
|
|
|
build.each do |opt|
|
|
|
|
hsh["options"] << {
|
|
|
|
"option" => "--"+opt.name,
|
|
|
|
"description" => opt.description
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
if rack.directory?
|
2013-05-16 17:48:12 -05:00
|
|
|
rack.subdirs.each do |keg|
|
2012-08-15 22:08:40 -05:00
|
|
|
tab = Tab.for_keg keg
|
|
|
|
|
|
|
|
hsh["installed"] << {
|
|
|
|
"version" => keg.basename.to_s,
|
2013-02-16 21:19:35 -06:00
|
|
|
"used_options" => tab.used_options.map(&:flag),
|
2013-03-15 00:28:18 +00:00
|
|
|
"built_as_bottle" => tab.built_bottle,
|
|
|
|
"poured_from_bottle" => tab.poured_from_bottle
|
2012-08-15 22:08:40 -05:00
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
hsh
|
|
|
|
|
|
|
|
end
|
|
|
|
|
2013-04-07 00:49:56 -05:00
|
|
|
protected
|
2012-03-04 16:48:00 -08:00
|
|
|
|
2009-07-24 15:10:01 +01:00
|
|
|
# Pretty titles the command and buffers stdout/stderr
|
|
|
|
# Throws if there's an error
|
2009-08-10 16:48:30 +01:00
|
|
|
def system cmd, *args
|
2011-07-30 11:03:27 +01:00
|
|
|
# remove "boring" arguments so that the important ones are more likely to
|
|
|
|
# be shown considering that we trim long ohai lines to the terminal width
|
|
|
|
pretty_args = args.dup
|
2012-08-31 08:09:35 -04:00
|
|
|
if cmd == "./configure" and not ARGV.verbose?
|
|
|
|
pretty_args.delete "--disable-dependency-tracking"
|
|
|
|
pretty_args.delete "--disable-debug"
|
|
|
|
end
|
2011-07-07 18:06:21 +01:00
|
|
|
ohai "#{cmd} #{pretty_args*' '}".strip
|
2009-11-06 17:09:14 +00:00
|
|
|
|
2011-09-10 11:24:29 +01:00
|
|
|
removed_ENV_variables = case if args.empty? then cmd.split(' ').first else cmd end
|
|
|
|
when "xcodebuild"
|
|
|
|
ENV.remove_cc_etc
|
|
|
|
end
|
|
|
|
|
2009-08-10 16:48:30 +01:00
|
|
|
if ARGV.verbose?
|
|
|
|
safe_system cmd, *args
|
2009-07-24 15:10:01 +01:00
|
|
|
else
|
2012-09-11 20:59:59 -04:00
|
|
|
@exec_count ||= 0
|
|
|
|
@exec_count += 1
|
|
|
|
logd = HOMEBREW_LOGS/name
|
2013-01-21 10:33:56 +01:00
|
|
|
logfn = "#{logd}/%02d.%s" % [@exec_count, File.basename(cmd.to_s).split(' ').first]
|
2012-09-11 20:59:59 -04:00
|
|
|
mkdir_p(logd)
|
|
|
|
|
2009-11-06 17:09:14 +00:00
|
|
|
rd, wr = IO.pipe
|
2013-02-17 22:54:27 -06:00
|
|
|
fork do
|
2009-11-06 17:09:14 +00:00
|
|
|
rd.close
|
|
|
|
$stdout.reopen wr
|
|
|
|
$stderr.reopen wr
|
2012-02-24 17:23:20 -06:00
|
|
|
args.collect!{|arg| arg.to_s}
|
2013-01-21 10:33:56 +01:00
|
|
|
exec(cmd.to_s, *args) rescue nil
|
2012-09-11 20:59:59 -04:00
|
|
|
puts "Failed to execute: #{cmd}"
|
2009-11-11 19:42:35 +00:00
|
|
|
exit! 1 # never gets here unless exec threw or failed
|
2009-11-06 17:09:14 +00:00
|
|
|
end
|
2009-11-09 17:45:22 +00:00
|
|
|
wr.close
|
2012-09-11 20:59:59 -04:00
|
|
|
|
|
|
|
f = File.open(logfn, 'w')
|
|
|
|
f.write(rd.read) until rd.eof?
|
|
|
|
|
2009-11-07 18:23:21 +00:00
|
|
|
Process.wait
|
2011-09-10 11:24:29 +01:00
|
|
|
|
2012-09-28 09:12:15 -04:00
|
|
|
unless $?.success?
|
|
|
|
unless ARGV.verbose?
|
|
|
|
f.flush
|
2012-12-11 13:58:43 -06:00
|
|
|
Kernel.system "/usr/bin/tail", "-n", "5", logfn
|
2012-09-28 09:12:15 -04:00
|
|
|
end
|
|
|
|
f.puts
|
|
|
|
require 'cmd/--config'
|
|
|
|
Homebrew.write_build_config(f)
|
2012-10-31 11:27:30 -04:00
|
|
|
raise ErrorDuringExecution
|
2012-09-28 09:12:15 -04:00
|
|
|
end
|
2012-09-11 20:59:59 -04:00
|
|
|
end
|
2013-02-17 22:54:27 -06:00
|
|
|
rescue ErrorDuringExecution
|
2012-10-31 11:27:30 -04:00
|
|
|
raise BuildError.new(self, cmd, args, $?)
|
2012-09-11 20:59:59 -04:00
|
|
|
ensure
|
2012-11-09 14:04:05 -06:00
|
|
|
f.close if f and not f.closed?
|
2012-09-28 09:12:15 -04:00
|
|
|
removed_ENV_variables.each do |key, value|
|
|
|
|
ENV[key] = value
|
|
|
|
end if removed_ENV_variables
|
2009-08-10 16:48:30 +01:00
|
|
|
end
|
2009-07-24 15:10:01 +01:00
|
|
|
|
2013-04-07 00:49:56 -05:00
|
|
|
public
|
2009-09-07 16:10:50 +02:00
|
|
|
|
2011-09-11 15:23:41 -07:00
|
|
|
# For brew-fetch and others.
|
|
|
|
def fetch
|
2011-09-19 19:05:44 -07:00
|
|
|
# Ensure the cache exists
|
|
|
|
HOMEBREW_CACHE.mkpath
|
2013-05-16 14:06:26 -05:00
|
|
|
downloader.fetch
|
2013-05-16 14:06:26 -05:00
|
|
|
cached_download
|
2011-09-11 15:23:41 -07:00
|
|
|
end
|
|
|
|
|
|
|
|
# For FormulaInstaller.
|
2012-04-05 21:09:24 -05:00
|
|
|
def verify_download_integrity fn
|
2013-04-13 17:40:12 -05:00
|
|
|
active_spec.verify_download_integrity(fn)
|
2009-08-11 12:20:55 -07:00
|
|
|
end
|
|
|
|
|
2012-11-01 14:40:59 -05:00
|
|
|
def test
|
|
|
|
ret = nil
|
|
|
|
mktemp do
|
|
|
|
@testpath = Pathname.pwd
|
2013-01-09 19:25:02 -06:00
|
|
|
ret = instance_eval(&self.class.test)
|
2012-11-01 14:40:59 -05:00
|
|
|
@testpath = nil
|
|
|
|
end
|
|
|
|
ret
|
|
|
|
end
|
|
|
|
|
2013-01-07 17:34:56 -06:00
|
|
|
def test_defined?
|
2013-01-09 19:25:02 -06:00
|
|
|
not self.class.instance_variable_get(:@test_defined).nil?
|
2013-01-07 17:34:56 -06:00
|
|
|
end
|
|
|
|
|
2013-04-07 00:49:56 -05:00
|
|
|
private
|
2012-03-04 16:48:00 -08:00
|
|
|
|
2009-08-11 12:20:55 -07:00
|
|
|
def stage
|
2013-05-16 14:06:26 -05:00
|
|
|
fetched = fetch
|
2013-05-16 14:06:26 -05:00
|
|
|
verify_download_integrity(fetched) if fetched.file?
|
2011-08-24 01:13:15 +01:00
|
|
|
mktemp do
|
2011-09-11 15:23:41 -07:00
|
|
|
downloader.stage
|
2012-02-24 12:50:21 -08:00
|
|
|
# Set path after the downloader changes the working folder.
|
|
|
|
@buildpath = Pathname.pwd
|
2011-08-24 01:13:15 +01:00
|
|
|
yield
|
2012-02-24 12:50:21 -08:00
|
|
|
@buildpath = nil
|
2009-07-31 11:05:23 -07:00
|
|
|
end
|
|
|
|
end
|
2010-07-04 14:17:03 -07:00
|
|
|
|
2009-08-10 16:48:30 +01:00
|
|
|
def patch
|
2012-03-09 10:56:45 -08:00
|
|
|
patch_list = Patches.new(patches)
|
2009-09-04 15:28:18 +01:00
|
|
|
return if patch_list.empty?
|
2009-09-15 20:07:40 +01:00
|
|
|
|
2012-04-29 11:51:04 -07:00
|
|
|
if patch_list.external_patches?
|
2011-08-29 14:55:28 -07:00
|
|
|
ohai "Downloading patches"
|
2012-04-29 11:51:04 -07:00
|
|
|
patch_list.download!
|
2011-08-29 14:55:28 -07:00
|
|
|
end
|
2009-09-15 20:07:40 +01:00
|
|
|
|
2010-05-11 08:34:29 -07:00
|
|
|
ohai "Patching"
|
2009-09-09 11:59:46 -04:00
|
|
|
patch_list.each do |p|
|
2012-03-09 10:56:45 -08:00
|
|
|
case p.compression
|
2012-04-29 11:51:04 -07:00
|
|
|
when :gzip then safe_system "/usr/bin/gunzip", p.compressed_filename
|
|
|
|
when :bzip2 then safe_system "/usr/bin/bunzip2", p.compressed_filename
|
2009-08-11 18:16:12 +01:00
|
|
|
end
|
2012-03-09 10:56:45 -08:00
|
|
|
# -f means don't prompt the user if there are errors; just exit with non-zero status
|
|
|
|
safe_system '/usr/bin/patch', '-f', *(p.patch_args)
|
2009-07-24 15:10:01 +01:00
|
|
|
end
|
|
|
|
end
|
2009-08-11 18:16:12 +01:00
|
|
|
|
2012-04-06 17:28:44 -05:00
|
|
|
def self.method_added method
|
2013-01-07 17:34:56 -06:00
|
|
|
case method
|
|
|
|
when :brew
|
|
|
|
raise "You cannot override Formula#brew"
|
|
|
|
when :test
|
2013-01-09 19:25:02 -06:00
|
|
|
@test_defined = true
|
2013-01-07 17:34:56 -06:00
|
|
|
end
|
2009-07-31 14:17:56 +01:00
|
|
|
end
|
2009-08-21 20:20:44 +01:00
|
|
|
|
2009-09-28 14:06:53 -07:00
|
|
|
class << self
|
2010-07-04 14:17:03 -07:00
|
|
|
# The methods below define the formula DSL.
|
2009-09-28 14:06:53 -07:00
|
|
|
|
2012-04-05 21:09:24 -05:00
|
|
|
attr_rw :homepage, :keg_only_reason, :skip_clean_all, :cc_failures
|
2012-11-25 15:06:41 +00:00
|
|
|
attr_rw :plist_startup, :plist_manual
|
2009-10-17 14:35:24 +02:00
|
|
|
|
2012-06-18 19:58:35 -05:00
|
|
|
Checksum::TYPES.each do |cksum|
|
2013-03-12 00:38:52 -05:00
|
|
|
class_eval <<-EOS, __FILE__, __LINE__ + 1
|
2013-03-18 14:59:10 -05:00
|
|
|
def #{cksum}(val)
|
|
|
|
@stable ||= SoftwareSpec.new
|
|
|
|
@stable.#{cksum}(val)
|
2012-04-05 21:09:24 -05:00
|
|
|
end
|
2013-03-12 00:38:52 -05:00
|
|
|
EOS
|
2010-07-04 14:17:03 -07:00
|
|
|
end
|
|
|
|
|
2012-07-30 11:32:56 -07:00
|
|
|
def build
|
2013-01-23 00:26:28 -06:00
|
|
|
@build ||= BuildOptions.new(ARGV.options_only)
|
2012-07-30 11:32:56 -07:00
|
|
|
end
|
|
|
|
|
2013-02-11 21:02:28 -06:00
|
|
|
def url val, specs={}
|
2012-04-05 21:09:24 -05:00
|
|
|
@stable ||= SoftwareSpec.new
|
|
|
|
@stable.url(val, specs)
|
2009-10-17 14:35:24 +02:00
|
|
|
end
|
2009-09-28 14:06:53 -07:00
|
|
|
|
2012-01-22 22:32:15 -06:00
|
|
|
def stable &block
|
2012-04-05 21:09:24 -05:00
|
|
|
return @stable unless block_given?
|
|
|
|
instance_eval(&block)
|
2012-01-22 22:32:15 -06:00
|
|
|
end
|
|
|
|
|
2013-03-28 17:37:28 -05:00
|
|
|
def bottle *, &block
|
2012-04-05 21:09:24 -05:00
|
|
|
return @bottle unless block_given?
|
|
|
|
@bottle ||= Bottle.new
|
|
|
|
@bottle.instance_eval(&block)
|
|
|
|
end
|
2012-03-18 20:37:10 +13:00
|
|
|
|
2012-04-05 21:09:24 -05:00
|
|
|
def devel &block
|
|
|
|
return @devel unless block_given?
|
|
|
|
@devel ||= SoftwareSpec.new
|
|
|
|
@devel.instance_eval(&block)
|
|
|
|
end
|
2012-03-18 20:37:10 +13:00
|
|
|
|
2013-02-11 20:50:52 -06:00
|
|
|
def head val=nil, specs={}
|
2012-04-05 21:09:24 -05:00
|
|
|
return @head if val.nil?
|
|
|
|
@head ||= HeadSoftwareSpec.new
|
|
|
|
@head.url(val, specs)
|
|
|
|
end
|
2012-03-18 20:37:10 +13:00
|
|
|
|
2012-04-05 21:09:24 -05:00
|
|
|
def version val=nil
|
|
|
|
@stable ||= SoftwareSpec.new
|
|
|
|
@stable.version(val)
|
2012-01-22 22:32:15 -06:00
|
|
|
end
|
|
|
|
|
2012-06-26 01:35:37 -05:00
|
|
|
def mirror val
|
2012-04-05 21:09:24 -05:00
|
|
|
@stable ||= SoftwareSpec.new
|
2012-06-26 01:35:37 -05:00
|
|
|
@stable.mirror(val)
|
2011-09-11 15:23:41 -07:00
|
|
|
end
|
|
|
|
|
2012-02-28 19:56:35 -08:00
|
|
|
def dependencies
|
|
|
|
@dependencies ||= DependencyCollector.new
|
|
|
|
end
|
|
|
|
|
|
|
|
def depends_on dep
|
2013-01-27 15:48:04 -06:00
|
|
|
d = dependencies.add(dep)
|
|
|
|
post_depends_on(d) unless d.nil?
|
2009-09-18 19:16:39 +01:00
|
|
|
end
|
2009-09-28 14:06:53 -07:00
|
|
|
|
2012-07-30 11:32:56 -07:00
|
|
|
def option name, description=nil
|
|
|
|
# Support symbols
|
|
|
|
name = name.to_s
|
|
|
|
raise "Option name is required." if name.empty?
|
|
|
|
raise "Options should not start with dashes." if name[0, 1] == "-"
|
|
|
|
build.add name, description
|
|
|
|
end
|
2012-11-25 15:06:41 +00:00
|
|
|
|
|
|
|
def plist_options options
|
|
|
|
@plist_startup = options[:startup]
|
|
|
|
@plist_manual = options[:manual]
|
|
|
|
end
|
2012-07-30 11:32:56 -07:00
|
|
|
|
2012-07-28 13:02:46 -03:00
|
|
|
def conflicts_with formula, opts={}
|
2012-11-06 17:32:23 -06:00
|
|
|
dependencies.add ConflictRequirement.new(formula, name, opts)
|
2012-07-28 13:02:46 -03:00
|
|
|
end
|
|
|
|
|
2012-10-21 13:03:35 -07:00
|
|
|
def skip_clean *paths
|
|
|
|
paths = [paths].flatten
|
|
|
|
|
|
|
|
# :all is deprecated though
|
|
|
|
if paths.include? :all
|
2012-08-29 15:43:28 -04:00
|
|
|
@skip_clean_all = true
|
2010-07-25 15:53:39 -07:00
|
|
|
return
|
|
|
|
end
|
2012-10-21 13:03:35 -07:00
|
|
|
|
2009-09-28 14:06:53 -07:00
|
|
|
@skip_clean_paths ||= []
|
2012-10-21 13:03:35 -07:00
|
|
|
paths.each do |p|
|
|
|
|
p = p.to_s unless p == :la # Keep :la in paths as a symbol
|
2012-09-09 10:01:59 -07:00
|
|
|
@skip_clean_paths << p unless @skip_clean_paths.include? p
|
2009-09-28 14:06:53 -07:00
|
|
|
end
|
|
|
|
end
|
2010-07-04 14:17:03 -07:00
|
|
|
|
2010-08-09 00:30:51 -05:00
|
|
|
def skip_clean_all?
|
|
|
|
@skip_clean_all
|
|
|
|
end
|
|
|
|
|
2009-09-29 23:51:37 +01:00
|
|
|
def skip_clean_paths
|
|
|
|
@skip_clean_paths or []
|
|
|
|
end
|
2010-07-18 15:18:51 -07:00
|
|
|
|
2011-03-15 22:01:27 -07:00
|
|
|
def keg_only reason, explanation=nil
|
2011-03-15 22:46:10 -07:00
|
|
|
@keg_only_reason = KegOnlyReason.new(reason, explanation.to_s.chomp)
|
2010-07-18 10:38:45 -07:00
|
|
|
end
|
2011-03-21 14:23:28 -07:00
|
|
|
|
2012-03-18 13:58:13 -05:00
|
|
|
def fails_with compiler, &block
|
2013-03-13 02:07:01 -05:00
|
|
|
@cc_failures ||= Set.new
|
|
|
|
@cc_failures << CompilerFailure.new(compiler, &block)
|
2011-03-21 14:23:28 -07:00
|
|
|
end
|
2012-11-01 14:40:59 -05:00
|
|
|
|
|
|
|
def test &block
|
|
|
|
return @test unless block_given?
|
2013-01-09 19:25:02 -06:00
|
|
|
@test_defined = true
|
|
|
|
@test = block
|
2012-11-01 14:40:59 -05:00
|
|
|
end
|
2013-01-23 00:26:26 -06:00
|
|
|
|
2013-04-07 00:49:56 -05:00
|
|
|
private
|
2013-01-23 00:26:26 -06:00
|
|
|
|
2013-01-27 15:48:04 -06:00
|
|
|
def post_depends_on(dep)
|
|
|
|
# Generate with- or without- options for optional and recommended
|
|
|
|
# dependencies and requirements
|
2013-02-16 17:15:14 -06:00
|
|
|
name = dep.name.split("/").last # strip any tap prefix
|
|
|
|
if dep.optional? && !build.has_option?("with-#{name}")
|
|
|
|
build.add("with-#{name}", "Build with #{name} support")
|
|
|
|
elsif dep.recommended? && !build.has_option?("without-#{name}")
|
|
|
|
build.add("without-#{name}", "Build without #{name} support")
|
2013-01-23 00:26:26 -06:00
|
|
|
end
|
2013-01-23 00:26:26 -06:00
|
|
|
end
|
2009-09-28 14:06:53 -07:00
|
|
|
end
|
2009-07-24 15:10:01 +01:00
|
|
|
end
|
2012-03-06 00:26:25 -06:00
|
|
|
|
|
|
|
require 'formula_specialties'
|