| 
									
										
										
										
											2016-04-16 00:20:04 +08:00
										 |  |  | #:  * `uses` [`--installed`] [`--recursive`] [`--include-build`] [`--include-optional`] [`--skip-recommended`] [`--devel`|`--HEAD`] <formulae>: | 
					
						
							| 
									
										
										
										
											2016-04-08 16:28:43 +02:00
										 |  |  | #:    Show the formulae that specify <formulae> as a dependency. When given | 
					
						
							|  |  |  | #:    multiple formula arguments, show the intersection of formulae that use | 
					
						
							|  |  |  | #:    <formulae>. | 
					
						
							|  |  |  | #: | 
					
						
							|  |  |  | #:    Use `--recursive` to resolve more than one level of dependencies. | 
					
						
							|  |  |  | #: | 
					
						
							|  |  |  | #:    If `--installed` is passed, only list installed formulae. | 
					
						
							|  |  |  | #: | 
					
						
							| 
									
										
										
										
											2016-04-16 00:20:04 +08:00
										 |  |  | #:    By default, `uses` shows all formulae that specify <formulae> as a required | 
					
						
							|  |  |  | #:    or recommended dependency. To include the `:build` type dependencies, pass | 
					
						
							|  |  |  | #:    `--include-build`. Similarly, pass `--include-optional` to include `:optional` | 
					
						
							|  |  |  | #:    dependencies. To skip `:recommended` type dependencies, pass `--skip-recommended`. | 
					
						
							| 
									
										
										
										
											2016-04-08 16:28:43 +02:00
										 |  |  | #: | 
					
						
							| 
									
										
										
										
											2017-02-25 17:37:57 -05:00
										 |  |  | #:    By default, `uses` shows usages of <formulae> by stable builds. To find | 
					
						
							|  |  |  | #:    cases where <formulae> is used by development or HEAD build, pass | 
					
						
							| 
									
										
										
										
											2016-04-08 16:28:43 +02:00
										 |  |  | #:    `--devel` or `--HEAD`. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  | require "formula" | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-26 11:03:08 -07:00
										 |  |  | # `brew uses foo bar` returns formulae that use both foo and bar | 
					
						
							|  |  |  | # If you want the union, run the command twice and concatenate the results. | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  | # The intersection is harder to achieve with shell tools. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 22:41:47 -05:00
										 |  |  | module Homebrew | 
					
						
							| 
									
										
										
										
											2016-09-26 01:44:51 +02:00
										 |  |  |   module_function | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |   def uses | 
					
						
							| 
									
										
										
										
											2012-02-04 00:01:29 -06:00
										 |  |  |     raise FormulaUnspecifiedError if ARGV.named.empty? | 
					
						
							| 
									
										
										
										
											2012-01-29 21:42:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-17 15:06:24 +01:00
										 |  |  |     used_formulae_missing = false | 
					
						
							|  |  |  |     used_formulae = begin | 
					
						
							|  |  |  |       ARGV.formulae | 
					
						
							|  |  |  |     rescue FormulaUnavailableError => e | 
					
						
							|  |  |  |       opoo e | 
					
						
							|  |  |  |       used_formulae_missing = true | 
					
						
							|  |  |  |       # If the formula doesn't exist: fake the needed formula object name. | 
					
						
							|  |  |  |       ARGV.named.map { |name| OpenStruct.new name: name, full_name: name } | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 10:24:57 +01:00
										 |  |  |     formulae = ARGV.include?("--installed") ? Formula.installed : Formula | 
					
						
							| 
									
										
										
										
											2014-04-07 13:07:04 -05:00
										 |  |  |     recursive = ARGV.flag? "--recursive" | 
					
						
							| 
									
										
										
										
											2016-04-16 00:20:04 +08:00
										 |  |  |     includes = [] | 
					
						
							| 
									
										
										
										
											2015-09-02 20:56:04 +08:00
										 |  |  |     ignores = [] | 
					
						
							| 
									
										
										
										
											2016-04-16 00:20:04 +08:00
										 |  |  |     if ARGV.include? "--include-build" | 
					
						
							|  |  |  |       includes << "build?" | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       ignores << "build?" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     if ARGV.include? "--include-optional" | 
					
						
							|  |  |  |       includes << "optional?" | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       ignores << "optional?" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     ignores << "recommended?" if ARGV.include? "--skip-recommended" | 
					
						
							| 
									
										
										
										
											2013-06-26 12:58:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-07 13:07:04 -05:00
										 |  |  |     uses = formulae.select do |f| | 
					
						
							| 
									
										
										
										
											2014-03-07 16:56:17 +01:00
										 |  |  |       used_formulae.all? do |ff| | 
					
						
							| 
									
										
										
										
											2017-01-07 13:15:18 +00:00
										 |  |  |         begin | 
					
						
							|  |  |  |           if recursive | 
					
						
							|  |  |  |             deps = f.recursive_dependencies do |dependent, dep| | 
					
						
							|  |  |  |               if dep.recommended? | 
					
						
							|  |  |  |                 Dependency.prune if ignores.include?("recommended?") || dependent.build.without?(dep) | 
					
						
							|  |  |  |               elsif dep.optional? | 
					
						
							|  |  |  |                 Dependency.prune if !includes.include?("optional?") && !dependent.build.with?(dep) | 
					
						
							|  |  |  |               elsif dep.build? | 
					
						
							|  |  |  |                 Dependency.prune unless includes.include?("build?") | 
					
						
							|  |  |  |               end | 
					
						
							| 
									
										
										
										
											2017-01-15 17:40:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-16 23:23:09 +00:00
										 |  |  |               # If a tap isn't installed, we can't find the dependencies of one | 
					
						
							|  |  |  |               # its formulae, and an exception will be thrown if we try. | 
					
						
							|  |  |  |               if dep.is_a?(TapDependency) && !dep.tap.installed? | 
					
						
							|  |  |  |                 Dependency.keep_but_prune_recursive_deps | 
					
						
							|  |  |  |               end | 
					
						
							| 
									
										
										
										
											2015-09-02 20:56:04 +08:00
										 |  |  |             end | 
					
						
							| 
									
										
										
										
											2017-01-16 22:41:37 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             dep_formulae = deps.map do |dep| | 
					
						
							|  |  |  |               begin | 
					
						
							|  |  |  |                 dep.to_formula | 
					
						
							|  |  |  |               rescue | 
					
						
							|  |  |  |               end | 
					
						
							|  |  |  |             end.compact | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             reqs_by_formula = ([f] + dep_formulae).flat_map do |formula| | 
					
						
							|  |  |  |               formula.requirements.map { |req| [formula, req] } | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             reqs_by_formula.reject! do |dependent, req| | 
					
						
							| 
									
										
										
										
											2017-01-07 13:15:18 +00:00
										 |  |  |               if req.recommended? | 
					
						
							| 
									
										
										
										
											2017-01-16 22:41:37 +00:00
										 |  |  |                 ignores.include?("recommended?") || dependent.build.without?(req) | 
					
						
							| 
									
										
										
										
											2017-01-07 13:15:18 +00:00
										 |  |  |               elsif req.optional? | 
					
						
							| 
									
										
										
										
											2017-01-16 22:41:37 +00:00
										 |  |  |                 !includes.include?("optional?") && !dependent.build.with?(req) | 
					
						
							| 
									
										
										
										
											2017-01-07 13:15:18 +00:00
										 |  |  |               elsif req.build? | 
					
						
							| 
									
										
										
										
											2017-01-16 22:41:37 +00:00
										 |  |  |                 !includes.include?("build?") | 
					
						
							| 
									
										
										
										
											2017-01-07 13:15:18 +00:00
										 |  |  |               end | 
					
						
							| 
									
										
										
										
											2015-05-19 21:40:04 +08:00
										 |  |  |             end | 
					
						
							| 
									
										
										
										
											2017-01-16 22:41:37 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             reqs = reqs_by_formula.map(&:last) | 
					
						
							| 
									
										
										
										
											2017-01-07 13:15:18 +00:00
										 |  |  |           else | 
					
						
							|  |  |  |             deps = f.deps.reject do |dep| | 
					
						
							| 
									
										
										
										
											2017-05-29 18:24:52 +01:00
										 |  |  |               ignores.any? { |ignore| dep.send(ignore) } && includes.none? { |include| dep.send(include) } | 
					
						
							| 
									
										
										
										
											2017-01-07 13:15:18 +00:00
										 |  |  |             end | 
					
						
							|  |  |  |             reqs = f.requirements.reject do |req| | 
					
						
							| 
									
										
										
										
											2017-05-29 18:24:52 +01:00
										 |  |  |               ignores.any? { |ignore| req.send(ignore) } && includes.none? { |include| req.send(include) } | 
					
						
							| 
									
										
										
										
											2016-09-10 10:24:57 +01:00
										 |  |  |             end | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2017-01-07 13:15:18 +00:00
										 |  |  |           next true if deps.any? do |dep| | 
					
						
							|  |  |  |             begin | 
					
						
							|  |  |  |               dep.to_formula.full_name == ff.full_name | 
					
						
							|  |  |  |             rescue | 
					
						
							|  |  |  |               dep.name == ff.name | 
					
						
							|  |  |  |             end | 
					
						
							| 
									
										
										
										
											2017-01-05 00:24:49 +00:00
										 |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-07 13:15:18 +00:00
										 |  |  |           reqs.any? do |req| | 
					
						
							|  |  |  |             req.name == ff.name || [ff.name, ff.full_name].include?(req.default_formula) | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         rescue FormulaUnavailableError | 
					
						
							|  |  |  |           # Silently ignore this case as we don't care about things used in | 
					
						
							|  |  |  |           # taps that aren't currently tapped. | 
					
						
							| 
									
										
										
										
											2012-03-16 22:00:59 -07:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2012-01-29 21:42:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-02 07:57:21 +02:00
										 |  |  |     return if uses.empty? | 
					
						
							| 
									
										
										
										
											2016-10-02 20:39:43 +02:00
										 |  |  |     puts Formatter.columns(uses.map(&:full_name)) | 
					
						
							| 
									
										
										
										
											2017-04-17 15:06:24 +01:00
										 |  |  |     odie "Missing formulae should not have dependents!" if used_formulae_missing | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |   end | 
					
						
							|  |  |  | end |