216 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			216 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| #:  * `deps` [`--1`] [`-n`] [`--union`] [`--full-name`] [`--installed`] [`--include-build`] [`--include-optional`] [`--skip-recommended`] [`--include-requirements`] <formulae>:
 | |
| #:    Show dependencies for <formulae>. When given multiple formula arguments,
 | |
| #:    show the intersection of dependencies for <formulae>.
 | |
| #:
 | |
| #:    If `--1` is passed, only show dependencies one level down, instead of
 | |
| #:    recursing.
 | |
| #:
 | |
| #:    If `-n` is passed, show dependencies in topological order.
 | |
| #:
 | |
| #:    If `--union` is passed, show the union of dependencies for <formulae>,
 | |
| #:    instead of the intersection.
 | |
| #:
 | |
| #:    If `--full-name` is passed, list dependencies by their full name.
 | |
| #:
 | |
| #:    If `--installed` is passed, only list those dependencies that are
 | |
| #:    currently installed.
 | |
| #:
 | |
| #:    By default, `deps` shows required and recommended dependencies for
 | |
| #:    <formulae>. To include the `:build` type dependencies, pass `--include-build`.
 | |
| #:    Similarly, pass `--include-optional` to include `:optional` dependencies or
 | |
| #:    `--include-test` to include (non-recursive) `:test` dependencies.
 | |
| #:    To skip `:recommended` type dependencies, pass `--skip-recommended`.
 | |
| #:    To include requirements in addition to dependencies, pass `--include-requirements`.
 | |
| #:
 | |
| #:  * `deps` `--tree` [`--1`] [<filters>] [`--annotate`] (<formulae>|`--installed`):
 | |
| #:    Show dependencies as a tree. When given multiple formula arguments, output
 | |
| #:    individual trees for every formula.
 | |
| #:
 | |
| #:    If `--1` is passed, only one level of children is displayed.
 | |
| #:
 | |
| #:    If `--installed` is passed, output a tree for every installed formula.
 | |
| #:
 | |
| #:    The <filters> placeholder is any combination of options `--include-build`,
 | |
| #:    `--include-optional`, `--include-test`, `--skip-recommended`, and
 | |
| #:    `--include-requirements` as documented above.
 | |
| #:
 | |
| #:    If `--annotate` is passed, the build, optional, and recommended dependencies
 | |
| #:    are marked as such in the output.
 | |
| #:
 | |
| #:  * `deps` [<filters>] (`--installed`|`--all`):
 | |
| #:    Show dependencies for installed or all available formulae. Every line of
 | |
| #:    output starts with the formula name, followed by a colon and all direct
 | |
| #:    dependencies of that formula.
 | |
| #:
 | |
| #:    The <filters> placeholder is any combination of options `--include-build`,
 | |
| #:    `--include-optional`, `--include-test`, and `--skip-recommended` as
 | |
| #:    documented above.
 | |
| 
 | |
| # The undocumented `--for-each` option will switch into the mode used by `deps --all`,
 | |
| # but only list dependencies for specified formula, one specified formula per line.
 | |
| # This is used for debugging the `--installed`/`--all` display mode.
 | |
| 
 | |
| # encoding: UTF-8
 | |
| 
 | |
| require "formula"
 | |
| require "ostruct"
 | |
| 
 | |
| module Homebrew
 | |
|   module_function
 | |
| 
 | |
|   def deps
 | |
|     mode = OpenStruct.new(
 | |
|       installed?: ARGV.include?("--installed"),
 | |
|       tree?: ARGV.include?("--tree"),
 | |
|       all?: ARGV.include?("--all"),
 | |
|       topo_order?: ARGV.include?("-n"),
 | |
|       union?: ARGV.include?("--union"),
 | |
|       for_each?: ARGV.include?("--for-each"),
 | |
|     )
 | |
| 
 | |
|     if mode.tree?
 | |
|       if mode.installed?
 | |
|         puts_deps_tree Formula.installed.sort, !ARGV.one?
 | |
|       else
 | |
|         raise FormulaUnspecifiedError if ARGV.named.empty?
 | |
|         puts_deps_tree ARGV.formulae, !ARGV.one?
 | |
|       end
 | |
|       return
 | |
|     elsif mode.all?
 | |
|       puts_deps Formula.sort
 | |
|       return
 | |
|     elsif !ARGV.named.empty? && mode.for_each?
 | |
|       puts_deps ARGV.formulae
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     @only_installed_arg = ARGV.include?("--installed") &&
 | |
|                           !ARGV.include?("--include-build") &&
 | |
|                           !ARGV.include?("--include-test") &&
 | |
|                           !ARGV.include?("--include-optional") &&
 | |
|                           !ARGV.include?("--skip-recommended")
 | |
| 
 | |
|     if ARGV.named.empty?
 | |
|       raise FormulaUnspecifiedError unless mode.installed?
 | |
|       puts_deps Formula.installed.sort
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     all_deps = deps_for_formulae(ARGV.formulae, !ARGV.one?, &(mode.union? ? :| : :&))
 | |
|     all_deps = condense_requirements(all_deps)
 | |
|     all_deps.select!(&:installed?) if mode.installed?
 | |
|     all_deps.map!(&method(:dep_display_name))
 | |
|     all_deps.uniq!
 | |
|     all_deps.sort! unless mode.topo_order?
 | |
|     puts all_deps
 | |
|   end
 | |
| 
 | |
|   def condense_requirements(deps)
 | |
|     return deps if ARGV.include?("--include-requirements")
 | |
|     deps.select { |dep| dep.is_a? Dependency }
 | |
|   end
 | |
| 
 | |
|   def dep_display_name(dep)
 | |
|     str = if dep.is_a? Requirement
 | |
|       if ARGV.include?("--include-requirements")
 | |
|         ":#{dep.display_s}"
 | |
|       else
 | |
|         # This shouldn't happen, but we'll put something here to help debugging
 | |
|         "::#{dep.name}"
 | |
|       end
 | |
|     elsif ARGV.include?("--full-name")
 | |
|       dep.to_formula.full_name
 | |
|     else
 | |
|       dep.name
 | |
|     end
 | |
| 
 | |
|     if ARGV.include?("--annotate")
 | |
|       str = "#{str}  [build]" if dep.build?
 | |
|       str = "#{str}  [test]" if dep.test?
 | |
|       str = "#{str}  [optional" if dep.optional?
 | |
|       str = "#{str}  [recommended]" if dep.recommended?
 | |
|     end
 | |
| 
 | |
|     str
 | |
|   end
 | |
| 
 | |
|   def deps_for_formula(f, recursive = false)
 | |
|     includes, ignores = argv_includes_ignores(ARGV)
 | |
| 
 | |
|     deps = f.runtime_dependencies if @only_installed_arg
 | |
| 
 | |
|     if recursive
 | |
|       deps ||= recursive_includes(Dependency,  f, includes, ignores)
 | |
|       reqs   = recursive_includes(Requirement, f, includes, ignores)
 | |
|     else
 | |
|       deps ||= reject_ignores(f.deps, ignores, includes)
 | |
|       reqs   = reject_ignores(f.requirements, ignores, includes)
 | |
|     end
 | |
| 
 | |
|     deps + reqs.to_a
 | |
|   end
 | |
| 
 | |
|   def deps_for_formulae(formulae, recursive = false, &block)
 | |
|     formulae.map { |f| deps_for_formula(f, recursive) }.inject(&block)
 | |
|   end
 | |
| 
 | |
|   def puts_deps(formulae)
 | |
|     formulae.each do |f|
 | |
|       deps = deps_for_formula(f)
 | |
|       deps = condense_requirements(deps)
 | |
|       deps.sort_by!(&:name)
 | |
|       deps.map!(&method(:dep_display_name))
 | |
|       puts "#{f.full_name}: #{deps.join(" ")}"
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def puts_deps_tree(formulae, recursive = false)
 | |
|     formulae.each do |f|
 | |
|       puts f.full_name
 | |
|       @dep_stack = []
 | |
|       recursive_deps_tree(f, "", recursive)
 | |
|       puts
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def recursive_deps_tree(f, prefix, recursive)
 | |
|     reqs = f.requirements
 | |
|     deps = f.deps
 | |
|     dependables = reqs + deps
 | |
|     dependables.reject!(&:optional?) unless ARGV.include?("--include-optional")
 | |
|     dependables.reject!(&:build?) unless ARGV.include?("--include-build")
 | |
|     dependables.reject!(&:test?) unless ARGV.include?("--include-test")
 | |
|     dependables.reject!(&:recommended?) if ARGV.include?("--skip-recommended")
 | |
|     max = dependables.length - 1
 | |
|     @dep_stack.push f.name
 | |
|     dependables.each_with_index do |dep, i|
 | |
|       next if !ARGV.include?("--include-requirements") && dep.is_a?(Requirement)
 | |
| 
 | |
|       tree_lines = if i == max
 | |
|         "└──"
 | |
|       else
 | |
|         "├──"
 | |
|       end
 | |
| 
 | |
|       display_s = "#{tree_lines} #{dep_display_name(dep)}"
 | |
|       is_circular = @dep_stack.include?(dep.name)
 | |
|       display_s = "#{display_s} (CIRCULAR DEPENDENCY)" if is_circular
 | |
|       puts "#{prefix}#{display_s}"
 | |
| 
 | |
|       next if !recursive || is_circular
 | |
| 
 | |
|       prefix_addition = if i == max
 | |
|         "    "
 | |
|       else
 | |
|         "│   "
 | |
|       end
 | |
| 
 | |
|       if dep.is_a? Dependency
 | |
|         recursive_deps_tree(Formulary.factory(dep.name), prefix + prefix_addition, true)
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     @dep_stack.pop
 | |
|   end
 | |
| end
 | 
