| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  | require "dependable" | 
					
						
							|  |  |  | require "dependency" | 
					
						
							|  |  |  | require "dependencies" | 
					
						
							|  |  |  | require "build_environment" | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | # A base class for non-formula requirements needed by formulae. | 
					
						
							|  |  |  | # A "fatal" requirement is one that will fail the build if it is not present. | 
					
						
							|  |  |  | # By default, Requirements are non-fatal. | 
					
						
							|  |  |  | class Requirement | 
					
						
							|  |  |  |   include Dependable | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-06 19:55:31 +08:00
										 |  |  |   attr_reader :tags, :name, :cask, :download, :default_formula | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |   def initialize(tags = []) | 
					
						
							| 
									
										
										
										
											2015-06-06 19:55:31 +08:00
										 |  |  |     @default_formula = self.class.default_formula | 
					
						
							| 
									
										
										
										
											2014-12-25 20:43:40 +00:00
										 |  |  |     @cask ||= self.class.cask | 
					
						
							|  |  |  |     @download ||= self.class.download | 
					
						
							| 
									
										
										
										
											2017-01-09 22:44:01 +00:00
										 |  |  |     @formula = nil | 
					
						
							| 
									
										
										
										
											2014-12-25 20:43:40 +00:00
										 |  |  |     tags.each do |tag| | 
					
						
							|  |  |  |       next unless tag.is_a? Hash | 
					
						
							|  |  |  |       @cask ||= tag[:cask] | 
					
						
							|  |  |  |       @download ||= tag[:download] | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2013-05-06 16:08:50 -05:00
										 |  |  |     @tags = tags | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |     @tags << :build if self.class.build | 
					
						
							| 
									
										
										
										
											2013-02-12 16:24:30 -06:00
										 |  |  |     @name ||= infer_name | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-14 19:47:19 +00:00
										 |  |  |   def option_names | 
					
						
							|  |  |  |     [name] | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   # The message to show when the requirement is not met. | 
					
						
							| 
									
										
										
										
											2014-12-25 20:43:40 +00:00
										 |  |  |   def message | 
					
						
							| 
									
										
										
										
											2016-07-10 13:53:03 +01:00
										 |  |  |     _, _, class_name = self.class.to_s.rpartition "::" | 
					
						
							|  |  |  |     s = "#{class_name} unsatisfied!\n" | 
					
						
							| 
									
										
										
										
											2014-12-25 20:43:40 +00:00
										 |  |  |     if cask | 
					
						
							| 
									
										
										
										
											2016-06-30 18:25:21 +01:00
										 |  |  |       s += <<-EOS.undent
 | 
					
						
							| 
									
										
										
										
											2014-12-25 20:43:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-06 18:34:09 +02:00
										 |  |  |         You can install with Homebrew-Cask: | 
					
						
							| 
									
										
										
										
											2016-06-30 18:25:21 +01:00
										 |  |  |           brew cask install #{cask} | 
					
						
							| 
									
										
										
										
											2014-12-25 20:43:40 +00:00
										 |  |  |       EOS | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if download | 
					
						
							|  |  |  |       s += <<-EOS.undent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         You can download from: | 
					
						
							|  |  |  |           #{download} | 
					
						
							|  |  |  |       EOS | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     s | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 18:58:26 -06:00
										 |  |  |   # Overriding #satisfied? is deprecated. | 
					
						
							| 
									
										
										
										
											2013-06-02 17:14:42 -05:00
										 |  |  |   # Pass a block or boolean to the satisfy DSL method instead. | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   def satisfied? | 
					
						
							| 
									
										
										
										
											2014-07-07 21:32:35 -05:00
										 |  |  |     result = self.class.satisfy.yielder { |p| instance_eval(&p) } | 
					
						
							|  |  |  |     @satisfied_result = result | 
					
						
							| 
									
										
										
										
											2017-01-09 22:44:01 +00:00
										 |  |  |     return false unless result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if parent = satisfied_result_parent | 
					
						
							|  |  |  |       parent.to_s =~ %r{(#{Regexp.escape(HOMEBREW_CELLAR)}|#{Regexp.escape(HOMEBREW_PREFIX)}/opt)/([\w+-.@]+)} | 
					
						
							|  |  |  |       @formula = $2 | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     true | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Overriding #fatal? is deprecated. | 
					
						
							|  |  |  |   # Pass a boolean to the fatal DSL method instead. | 
					
						
							|  |  |  |   def fatal? | 
					
						
							|  |  |  |     self.class.fatal || false | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:47 -05:00
										 |  |  |   def default_formula? | 
					
						
							|  |  |  |     self.class.default_formula || false | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 22:44:01 +00:00
										 |  |  |   def satisfied_result_parent | 
					
						
							|  |  |  |     return unless @satisfied_result.is_a?(Pathname) | 
					
						
							|  |  |  |     @satisfied_result.resolved_path.parent | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   # Overriding #modify_build_environment is deprecated. | 
					
						
							| 
									
										
										
										
											2016-01-13 03:01:26 +09:00
										 |  |  |   # Pass a block to the env DSL method instead. | 
					
						
							| 
									
										
										
										
											2013-06-24 10:39:00 -05:00
										 |  |  |   # Note: #satisfied? should be called before invoking this method | 
					
						
							|  |  |  |   # as the env modifications may depend on its side effects. | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   def modify_build_environment | 
					
						
							| 
									
										
										
										
											2014-07-07 21:32:35 -05:00
										 |  |  |     instance_eval(&env_proc) if env_proc | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # XXX If the satisfy block returns a Pathname, then make sure that it | 
					
						
							|  |  |  |     # remains available on the PATH. This makes requirements like | 
					
						
							|  |  |  |     #   satisfy { which("executable") } | 
					
						
							|  |  |  |     # work, even under superenv where "executable" wouldn't normally be on the | 
					
						
							|  |  |  |     # PATH. | 
					
						
							| 
									
										
										
										
											2017-01-09 22:44:01 +00:00
										 |  |  |     parent = satisfied_result_parent | 
					
						
							|  |  |  |     return unless parent | 
					
						
							| 
									
										
										
										
											2016-09-23 22:02:23 +02:00
										 |  |  |     return if ENV["PATH"].split(File::PATH_SEPARATOR).include?(parent.to_s) | 
					
						
							|  |  |  |     ENV.append_path("PATH", parent) | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def env | 
					
						
							| 
									
										
										
										
											2014-07-07 21:32:35 -05:00
										 |  |  |     self.class.env | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-07 21:32:35 -05:00
										 |  |  |   def env_proc | 
					
						
							|  |  |  |     self.class.env_proc | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-12 17:35:16 -06:00
										 |  |  |   def ==(other) | 
					
						
							| 
									
										
										
										
											2013-06-27 01:18:32 -05:00
										 |  |  |     instance_of?(other.class) && name == other.name && tags == other.tags | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2016-09-23 18:13:48 +02:00
										 |  |  |   alias eql? == | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |   def hash | 
					
						
							| 
									
										
										
										
											2014-11-12 17:33:07 -06:00
										 |  |  |     name.hash ^ tags.hash | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-07 22:24:36 -05:00
										 |  |  |   def inspect | 
					
						
							| 
									
										
										
										
											2014-07-01 15:07:06 -05:00
										 |  |  |     "#<#{self.class.name}: #{name.inspect} #{tags.inspect}>" | 
					
						
							| 
									
										
										
										
											2013-06-07 22:24:36 -05:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 22:44:01 +00:00
										 |  |  |   def formula | 
					
						
							|  |  |  |     @formula || self.class.default_formula | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:47 -05:00
										 |  |  |   def to_dependency | 
					
						
							| 
									
										
										
										
											2017-01-09 22:44:01 +00:00
										 |  |  |     if formula =~ HOMEBREW_TAP_FORMULA_REGEX | 
					
						
							|  |  |  |       TapDependency.new(formula, tags, method(:modify_build_environment), name) | 
					
						
							|  |  |  |     elsif formula | 
					
						
							|  |  |  |       Dependency.new(formula, tags, method(:modify_build_environment), name) | 
					
						
							| 
									
										
										
										
											2015-08-20 16:15:31 +08:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:47 -05:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 00:37:02 -04:00
										 |  |  |   def display_s | 
					
						
							|  |  |  |     name | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   private | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-12 16:24:30 -06:00
										 |  |  |   def infer_name | 
					
						
							| 
									
										
										
										
											2014-06-12 16:12:21 -05:00
										 |  |  |     klass = self.class.name || self.class.to_s | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |     klass.sub!(/(Dependency|Requirement)$/, "") | 
					
						
							|  |  |  |     klass.sub!(/^(\w+::)*/, "") | 
					
						
							| 
									
										
										
										
											2013-02-12 16:24:30 -06:00
										 |  |  |     klass.downcase | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-19 12:33:00 -05:00
										 |  |  |   def which(cmd) | 
					
						
							|  |  |  |     super(cmd, ORIGINAL_PATHS.join(File::PATH_SEPARATOR)) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-02 23:08:51 +08:00
										 |  |  |   def which_all(cmd) | 
					
						
							|  |  |  |     super(cmd, ORIGINAL_PATHS.join(File::PATH_SEPARATOR)) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   class << self | 
					
						
							| 
									
										
										
										
											2014-07-07 20:16:51 -05:00
										 |  |  |     include BuildEnvironmentDSL | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-07 21:32:35 -05:00
										 |  |  |     attr_reader :env_proc | 
					
						
							| 
									
										
										
										
											2014-05-30 18:11:25 -05:00
										 |  |  |     attr_rw :fatal, :default_formula | 
					
						
							| 
									
										
										
										
											2014-12-25 20:43:40 +00:00
										 |  |  |     attr_rw :cask, :download | 
					
						
							| 
									
										
										
										
											2014-05-30 18:11:25 -05:00
										 |  |  |     # build is deprecated, use `depends_on <requirement> => :build` instead | 
					
						
							|  |  |  |     attr_rw :build | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |     def satisfy(options = {}, &block) | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |       @satisfied ||= Requirement::Satisfier.new(options, &block) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2014-07-07 21:32:35 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def env(*settings, &block) | 
					
						
							|  |  |  |       if block_given? | 
					
						
							|  |  |  |         @env_proc = block | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         super | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   class Satisfier | 
					
						
							| 
									
										
										
										
											2014-08-14 19:58:17 -05:00
										 |  |  |     def initialize(options, &block) | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |       case options | 
					
						
							|  |  |  |       when Hash | 
					
						
							| 
									
										
										
										
											2016-09-17 15:32:44 +01:00
										 |  |  |         @options = { build_env: true } | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |         @options.merge!(options) | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         @satisfied = options | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |       @proc = block | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def yielder | 
					
						
							|  |  |  |       if instance_variable_defined?(:@satisfied) | 
					
						
							|  |  |  |         @satisfied | 
					
						
							|  |  |  |       elsif @options[:build_env] | 
					
						
							| 
									
										
										
										
											2014-07-02 21:24:01 -05:00
										 |  |  |         require "extend/ENV" | 
					
						
							| 
									
										
										
										
											2013-08-19 12:33:00 -05:00
										 |  |  |         ENV.with_build_environment { yield @proc } | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  |       else | 
					
						
							|  |  |  |         yield @proc | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2013-01-30 17:55:04 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:46 -05:00
										 |  |  |   class << self | 
					
						
							|  |  |  |     # Expand the requirements of dependent recursively, optionally yielding | 
					
						
							|  |  |  |     # [dependent, req] pairs to allow callers to apply arbitrary filters to | 
					
						
							|  |  |  |     # the list. | 
					
						
							|  |  |  |     # The default filter, which is applied when a block is not given, omits | 
					
						
							|  |  |  |     # optionals and recommendeds based on what the dependent has asked for. | 
					
						
							|  |  |  |     def expand(dependent, &block) | 
					
						
							| 
									
										
										
										
											2014-07-03 14:50:57 -05:00
										 |  |  |       reqs = Requirements.new | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:46 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |       formulae = dependent.recursive_dependencies.map(&:to_formula) | 
					
						
							|  |  |  |       formulae.unshift(dependent) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-25 10:12:14 -05:00
										 |  |  |       formulae.each do |f| | 
					
						
							|  |  |  |         f.requirements.each do |req| | 
					
						
							| 
									
										
										
										
											2016-09-23 22:02:23 +02:00
										 |  |  |           next if prune?(f, req, &block) | 
					
						
							|  |  |  |           reqs << req | 
					
						
							| 
									
										
										
										
											2013-01-30 17:55:04 -06:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:46 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-01-30 17:55:04 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:46 -05:00
										 |  |  |       reqs | 
					
						
							| 
									
										
										
										
											2013-01-30 17:55:04 -06:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |     def prune?(dependent, req, &_block) | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:46 -05:00
										 |  |  |       catch(:prune) do | 
					
						
							|  |  |  |         if block_given? | 
					
						
							|  |  |  |           yield dependent, req | 
					
						
							|  |  |  |         elsif req.optional? || req.recommended? | 
					
						
							| 
									
										
										
										
											2013-12-09 14:36:10 -06:00
										 |  |  |           prune unless dependent.build.with?(req) | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:46 -05:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-01-30 17:55:04 -06:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-03 15:08:46 -05:00
										 |  |  |     # Used to prune requirements when calling expand with a block. | 
					
						
							|  |  |  |     def prune | 
					
						
							|  |  |  |       throw(:prune, true) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2013-01-30 17:55:04 -06:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2013-01-26 20:05:39 -06:00
										 |  |  | end |