| 
									
										
										
										
											2024-08-12 10:30:59 +01:00
										 |  |  | # typed: true # rubocop:todo Sorbet/StrictSigil | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  | require "abstract_command" | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  | require "formula" | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  | require "cask/caskroom" | 
					
						
							| 
									
										
										
										
											2020-07-31 11:35:45 -04:00
										 |  |  | require "dependencies_helpers" | 
					
						
							| 
									
										
										
										
											2010-09-25 12:49:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 22:41:47 -05:00
										 |  |  | module Homebrew | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |   module Cmd | 
					
						
							|  |  |  |     class Deps < AbstractCommand | 
					
						
							|  |  |  |       include DependenciesHelpers | 
					
						
							|  |  |  |       cmd_args do | 
					
						
							|  |  |  |         description <<~EOS | 
					
						
							|  |  |  |           Show dependencies for <formula>. When given multiple formula arguments, | 
					
						
							|  |  |  |           show the intersection of dependencies for each formula. By default, `deps` | 
					
						
							|  |  |  |           shows all required and recommended dependencies. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           If any version of each formula argument is installed and no other options | 
					
						
							|  |  |  |           are passed, this command displays their actual runtime dependencies (similar | 
					
						
							| 
									
										
										
										
											2025-06-10 15:55:39 +01:00
										 |  |  |           to `brew linkage`), which may differ from a formula's declared dependencies. | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |           *Note:* `--missing` and `--skip-recommended` have precedence over `--include-*`. | 
					
						
							|  |  |  |         EOS | 
					
						
							|  |  |  |         switch "-n", "--topological", | 
					
						
							|  |  |  |                description: "Sort dependencies in topological order." | 
					
						
							|  |  |  |         switch "-1", "--direct", "--declared", "--1", | 
					
						
							|  |  |  |                description: "Show only the direct dependencies declared in the formula." | 
					
						
							|  |  |  |         switch "--union", | 
					
						
							|  |  |  |                description: "Show the union of dependencies for multiple <formula>, instead of the intersection." | 
					
						
							|  |  |  |         switch "--full-name", | 
					
						
							|  |  |  |                description: "List dependencies by their full name." | 
					
						
							| 
									
										
										
										
											2024-12-24 12:50:31 -05:00
										 |  |  |         switch "--include-implicit", | 
					
						
							| 
									
										
										
										
											2025-02-03 17:40:17 +01:00
										 |  |  |                description: "Include implicit dependencies used to download and unpack source files." | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         switch "--include-build", | 
					
						
							|  |  |  |                description: "Include `:build` dependencies for <formula>." | 
					
						
							|  |  |  |         switch "--include-optional", | 
					
						
							|  |  |  |                description: "Include `:optional` dependencies for <formula>." | 
					
						
							|  |  |  |         switch "--include-test", | 
					
						
							|  |  |  |                description: "Include `:test` dependencies for <formula> (non-recursive)." | 
					
						
							|  |  |  |         switch "--skip-recommended", | 
					
						
							|  |  |  |                description: "Skip `:recommended` dependencies for <formula>." | 
					
						
							|  |  |  |         switch "--include-requirements", | 
					
						
							|  |  |  |                description: "Include requirements in addition to dependencies for <formula>." | 
					
						
							|  |  |  |         switch "--tree", | 
					
						
							|  |  |  |                description: "Show dependencies as a tree. When given multiple formula arguments, " \ | 
					
						
							|  |  |  |                             "show individual trees for each formula." | 
					
						
							|  |  |  |         switch "--graph", | 
					
						
							|  |  |  |                description: "Show dependencies as a directed graph." | 
					
						
							|  |  |  |         switch "--dot", | 
					
						
							|  |  |  |                depends_on:  "--graph", | 
					
						
							|  |  |  |                description: "Show text-based graph description in DOT format." | 
					
						
							|  |  |  |         switch "--annotate", | 
					
						
							|  |  |  |                description: "Mark any build, test, implicit, optional, or recommended dependencies as " \ | 
					
						
							|  |  |  |                             "such in the output." | 
					
						
							|  |  |  |         switch "--installed", | 
					
						
							|  |  |  |                description: "List dependencies for formulae that are currently installed. If <formula> is " \ | 
					
						
							|  |  |  |                             "specified, list only its dependencies that are currently installed." | 
					
						
							|  |  |  |         switch "--missing", | 
					
						
							|  |  |  |                description: "Show only missing dependencies." | 
					
						
							|  |  |  |         switch "--eval-all", | 
					
						
							|  |  |  |                description: "Evaluate all available formulae and casks, whether installed or not, to list " \ | 
					
						
							|  |  |  |                             "their dependencies." | 
					
						
							|  |  |  |         switch "--for-each", | 
					
						
							|  |  |  |                description: "Switch into the mode used by the `--eval-all` option, but only list dependencies " \ | 
					
						
							|  |  |  |                             "for each provided <formula>, one formula per line. This is used for " \ | 
					
						
							|  |  |  |                             "debugging the `--installed`/`--eval-all` display mode." | 
					
						
							|  |  |  |         switch "--HEAD", | 
					
						
							|  |  |  |                description: "Show dependencies for HEAD version instead of stable version." | 
					
						
							| 
									
										
										
										
											2024-04-21 13:16:33 -04:00
										 |  |  |         flag   "--os=", | 
					
						
							|  |  |  |                description: "Show dependencies for the given operating system." | 
					
						
							|  |  |  |         flag   "--arch=", | 
					
						
							|  |  |  |                description: "Show dependencies for the given CPU architecture." | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         switch "--formula", "--formulae", | 
					
						
							|  |  |  |                description: "Treat all named arguments as formulae." | 
					
						
							|  |  |  |         switch "--cask", "--casks", | 
					
						
							|  |  |  |                description: "Treat all named arguments as casks." | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         conflicts "--tree", "--graph" | 
					
						
							|  |  |  |         conflicts "--installed", "--missing" | 
					
						
							|  |  |  |         conflicts "--installed", "--eval-all" | 
					
						
							|  |  |  |         conflicts "--formula", "--cask" | 
					
						
							|  |  |  |         formula_options | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         named_args [:formula, :cask] | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       sig { override.void } | 
					
						
							|  |  |  |       def run | 
					
						
							| 
									
										
										
										
											2024-04-21 13:16:33 -04:00
										 |  |  |         raise UsageError, "`brew deps --os=all` is not supported" if args.os == "all" | 
					
						
							|  |  |  |         raise UsageError, "`brew deps --arch=all` is not supported" if args.arch == "all" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         os, arch = T.must(args.os_arch_combinations.first) | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         all = args.eval_all? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Formulary.enable_factory_cache! | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 13:16:33 -04:00
										 |  |  |         SimulateSystem.with(os:, arch:) do | 
					
						
							| 
									
										
										
										
											2025-06-10 15:55:39 +01:00
										 |  |  |           @use_runtime_dependencies = true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 13:16:33 -04:00
										 |  |  |           installed = args.installed? || dependents(args.named.to_formulae_and_casks).all?(&:any_version_installed?) | 
					
						
							| 
									
										
										
										
											2025-06-10 15:55:39 +01:00
										 |  |  |           unless installed | 
					
						
							|  |  |  |             not_using_runtime_dependencies_reason = if args.installed? | 
					
						
							|  |  |  |               "not all the named formulae were installed" | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |               "`--installed` was not passed" | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             @use_runtime_dependencies = false | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           %w[direct tree graph HEAD skip_recommended missing
 | 
					
						
							|  |  |  |              include_implicit include_build include_test include_optional].each do |arg| | 
					
						
							|  |  |  |             next unless args.public_send("#{arg}?") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             not_using_runtime_dependencies_reason = "--#{arg.tr("_", "-")} was passed" | 
					
						
							| 
									
										
										
										
											2024-04-21 13:16:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-10 15:55:39 +01:00
										 |  |  |             @use_runtime_dependencies = false | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           %w[os arch].each do |arg| | 
					
						
							|  |  |  |             next if args.public_send(arg).nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             not_using_runtime_dependencies_reason = "--#{arg.tr("_", "-")} was passed" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             @use_runtime_dependencies = false | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if !@use_runtime_dependencies && !Homebrew::EnvConfig.no_env_hints? | 
					
						
							|  |  |  |             opoo <<~EOS | 
					
						
							|  |  |  |               `brew deps` is not the actual runtime dependencies because #{not_using_runtime_dependencies_reason}! | 
					
						
							|  |  |  |               This means dependencies may differ from a formula's declared dependencies. | 
					
						
							|  |  |  |               Hide these hints with HOMEBREW_NO_ENV_HINTS (see `man brew`). | 
					
						
							|  |  |  |             EOS | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           recursive = !args.direct? | 
					
						
							| 
									
										
										
										
											2024-04-21 13:16:33 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |           if args.tree? || args.graph? | 
					
						
							|  |  |  |             dependents = if args.named.present? | 
					
						
							|  |  |  |               sorted_dependents(args.named.to_formulae_and_casks) | 
					
						
							|  |  |  |             elsif args.installed? | 
					
						
							|  |  |  |               case args.only_formula_or_cask | 
					
						
							|  |  |  |               when :formula | 
					
						
							|  |  |  |                 sorted_dependents(Formula.installed) | 
					
						
							|  |  |  |               when :cask | 
					
						
							|  |  |  |                 sorted_dependents(Cask::Caskroom.casks) | 
					
						
							|  |  |  |               else | 
					
						
							|  |  |  |                 sorted_dependents(Formula.installed + Cask::Caskroom.casks) | 
					
						
							|  |  |  |               end | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |               raise FormulaUnspecifiedError | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if args.graph? | 
					
						
							|  |  |  |               dot_code = dot_code(dependents, recursive:) | 
					
						
							|  |  |  |               if args.dot? | 
					
						
							|  |  |  |                 puts dot_code | 
					
						
							|  |  |  |               else | 
					
						
							|  |  |  |                 exec_browser "https://dreampuf.github.io/GraphvizOnline/##{ERB::Util.url_encode(dot_code)}" | 
					
						
							|  |  |  |               end | 
					
						
							|  |  |  |               return | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             puts_deps_tree(dependents, recursive:) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |           elsif all | 
					
						
							|  |  |  |             puts_deps(sorted_dependents( | 
					
						
							|  |  |  |                         Formula.all(eval_all: args.eval_all?) + Cask::Cask.all(eval_all: args.eval_all?), | 
					
						
							|  |  |  |                       ), recursive:) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |           elsif !args.no_named? && args.for_each? | 
					
						
							|  |  |  |             puts_deps(sorted_dependents(args.named.to_formulae_and_casks), recursive:) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if args.no_named? | 
					
						
							|  |  |  |             raise FormulaUnspecifiedError unless args.installed? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             sorted_dependents_formulae_and_casks = case args.only_formula_or_cask | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |             when :formula | 
					
						
							|  |  |  |               sorted_dependents(Formula.installed) | 
					
						
							|  |  |  |             when :cask | 
					
						
							|  |  |  |               sorted_dependents(Cask::Caskroom.casks) | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |               sorted_dependents(Formula.installed + Cask::Caskroom.casks) | 
					
						
							|  |  |  |             end | 
					
						
							| 
									
										
										
										
											2024-04-21 13:16:33 -04:00
										 |  |  |             puts_deps(sorted_dependents_formulae_and_casks, recursive:) | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |             return | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 13:16:33 -04:00
										 |  |  |           dependents = dependents(args.named.to_formulae_and_casks) | 
					
						
							|  |  |  |           check_head_spec(dependents) if args.HEAD? | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 13:16:33 -04:00
										 |  |  |           all_deps = deps_for_dependents(dependents, recursive:, &(args.union? ? :| : :&)) | 
					
						
							|  |  |  |           condense_requirements(all_deps) | 
					
						
							|  |  |  |           all_deps.map! { |d| dep_display_name(d) } | 
					
						
							|  |  |  |           all_deps.uniq! | 
					
						
							|  |  |  |           all_deps.sort! unless args.topological? | 
					
						
							|  |  |  |           puts all_deps | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-12-04 10:11:17 +09:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-30 16:31:13 -07:00
										 |  |  |       private | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def sorted_dependents(formulae_or_casks) | 
					
						
							|  |  |  |         dependents(formulae_or_casks).sort_by(&:name) | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def condense_requirements(deps) | 
					
						
							|  |  |  |         deps.select! { |dep| dep.is_a?(Dependency) } unless args.include_requirements? | 
					
						
							|  |  |  |         deps.select! { |dep| dep.is_a?(Requirement) || dep.installed? } if args.installed? | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def dep_display_name(dep) | 
					
						
							|  |  |  |         str = if dep.is_a? Requirement | 
					
						
							|  |  |  |           if args.include_requirements? | 
					
						
							|  |  |  |             ":#{dep.display_s}" | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             # This shouldn't happen, but we'll put something here to help debugging | 
					
						
							|  |  |  |             "::#{dep.name}" | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         elsif args.full_name? | 
					
						
							|  |  |  |           dep.to_formula.full_name | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           dep.name | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-12-20 03:39:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         if args.annotate? | 
					
						
							|  |  |  |           str = "#{str} " if args.tree? | 
					
						
							|  |  |  |           str = "#{str} [build]" if dep.build? | 
					
						
							|  |  |  |           str = "#{str} [test]" if dep.test? | 
					
						
							|  |  |  |           str = "#{str} [optional]" if dep.optional? | 
					
						
							|  |  |  |           str = "#{str} [recommended]" if dep.recommended? | 
					
						
							|  |  |  |           str = "#{str} [implicit]" if dep.implicit? | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2015-03-20 21:31:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         str | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def deps_for_dependent(dependency, recursive: false) | 
					
						
							|  |  |  |         includes, ignores = args_includes_ignores(args) | 
					
						
							| 
									
										
										
										
											2014-02-27 12:56:42 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         deps = dependency.runtime_dependencies if @use_runtime_dependencies | 
					
						
							| 
									
										
										
										
											2013-10-29 17:27:21 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         if recursive | 
					
						
							| 
									
										
										
										
											2025-02-15 23:04:04 -08:00
										 |  |  |           deps ||= recursive_dep_includes(dependency, includes, ignores) | 
					
						
							|  |  |  |           reqs   = recursive_req_includes(dependency, includes, ignores) | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         else | 
					
						
							|  |  |  |           deps ||= select_includes(dependency.deps, ignores, includes) | 
					
						
							|  |  |  |           reqs   = select_includes(dependency.requirements, ignores, includes) | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2013-06-22 12:54:45 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         deps + reqs.to_a | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2023-10-12 07:58:25 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def deps_for_dependents(dependents, recursive: false, &block) | 
					
						
							|  |  |  |         dependents.map { |d| deps_for_dependent(d, recursive:) }.reduce(&block) | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-06-22 12:54:45 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def check_head_spec(dependents) | 
					
						
							|  |  |  |         headless = dependents.select { |d| d.is_a?(Formula) && d.active_spec_sym != :head } | 
					
						
							|  |  |  |                              .to_sentence two_words_connector: " or ", last_word_connector: " or " | 
					
						
							|  |  |  |         opoo "No head spec for #{headless}, using stable spec instead" unless headless.empty? | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def puts_deps(dependents, recursive: false) | 
					
						
							|  |  |  |         check_head_spec(dependents) if args.HEAD? | 
					
						
							|  |  |  |         dependents.each do |dependent| | 
					
						
							|  |  |  |           deps = deps_for_dependent(dependent, recursive:) | 
					
						
							|  |  |  |           condense_requirements(deps) | 
					
						
							|  |  |  |           deps.sort_by!(&:name) | 
					
						
							|  |  |  |           deps.map! { |d| dep_display_name(d) } | 
					
						
							|  |  |  |           puts "#{dependent.full_name}: #{deps.join(" ")}" | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def dot_code(dependents, recursive:) | 
					
						
							|  |  |  |         dep_graph = {} | 
					
						
							|  |  |  |         dependents.each do |d| | 
					
						
							|  |  |  |           graph_deps(d, dep_graph:, recursive:) | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         dot_code = dep_graph.map do |d, deps| | 
					
						
							|  |  |  |           deps.map do |dep| | 
					
						
							|  |  |  |             attributes = [] | 
					
						
							|  |  |  |             attributes << "style = dotted" if dep.build? | 
					
						
							|  |  |  |             attributes << "arrowhead = empty" if dep.test? | 
					
						
							|  |  |  |             if dep.optional? | 
					
						
							|  |  |  |               attributes << "color = red" | 
					
						
							|  |  |  |             elsif dep.recommended? | 
					
						
							|  |  |  |               attributes << "color = green" | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |             comment = " # #{dep.tags.map(&:inspect).join(", ")}" if dep.tags.any? | 
					
						
							|  |  |  |             "  \"#{d.name}\" -> \"#{dep}\"#{" [#{attributes.join(", ")}]" if attributes.any?}#{comment}" | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end.flatten.join("\n") | 
					
						
							|  |  |  |         "digraph {\n#{dot_code}\n}" | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def graph_deps(formula, dep_graph:, recursive:) | 
					
						
							|  |  |  |         return if dep_graph.key?(formula) | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         dependables = dependables(formula) | 
					
						
							|  |  |  |         dep_graph[formula] = dependables | 
					
						
							|  |  |  |         return unless recursive | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         dependables.each do |dep| | 
					
						
							|  |  |  |           next unless dep.is_a? Dependency | 
					
						
							| 
									
										
										
										
											2013-06-22 12:54:45 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |           graph_deps(Formulary.factory(dep.name), | 
					
						
							|  |  |  |                      dep_graph:, | 
					
						
							|  |  |  |                      recursive: true) | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-12-20 03:59:15 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def puts_deps_tree(dependents, recursive: false) | 
					
						
							|  |  |  |         check_head_spec(dependents) if args.HEAD? | 
					
						
							|  |  |  |         dependents.each do |d| | 
					
						
							|  |  |  |           puts d.full_name | 
					
						
							|  |  |  |           recursive_deps_tree(d, dep_stack: [], prefix: "", recursive:) | 
					
						
							|  |  |  |           puts | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2022-08-30 12:21:02 +00:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def dependables(formula) | 
					
						
							|  |  |  |         includes, ignores = args_includes_ignores(args) | 
					
						
							|  |  |  |         deps = @use_runtime_dependencies ? formula.runtime_dependencies : formula.deps | 
					
						
							|  |  |  |         deps = select_includes(deps, ignores, includes) | 
					
						
							|  |  |  |         reqs = select_includes(formula.requirements, ignores, includes) if args.include_requirements? | 
					
						
							|  |  |  |         reqs ||= [] | 
					
						
							|  |  |  |         reqs + deps | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |       def recursive_deps_tree(formula, dep_stack:, prefix:, recursive:) | 
					
						
							|  |  |  |         dependables = dependables(formula) | 
					
						
							|  |  |  |         max = dependables.length - 1
 | 
					
						
							|  |  |  |         dep_stack.push formula.name | 
					
						
							|  |  |  |         dependables.each_with_index do |dep, i| | 
					
						
							|  |  |  |           tree_lines = if i == max | 
					
						
							|  |  |  |             "└──" | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             "├──" | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           display_s = "#{tree_lines} #{dep_display_name(dep)}" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           # Detect circular dependencies and consider them a failure if present. | 
					
						
							|  |  |  |           is_circular = dep_stack.include?(dep.name) | 
					
						
							|  |  |  |           if is_circular | 
					
						
							|  |  |  |             display_s = "#{display_s} (CIRCULAR DEPENDENCY)" | 
					
						
							|  |  |  |             Homebrew.failed = true | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           puts "#{prefix}#{display_s}" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           next if !recursive || is_circular | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           prefix_addition = if i == max | 
					
						
							|  |  |  |             "    " | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             "│   " | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           next unless dep.is_a? Dependency | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           recursive_deps_tree(Formulary.factory(dep.name), | 
					
						
							|  |  |  |                               dep_stack:, | 
					
						
							|  |  |  |                               prefix:    prefix + prefix_addition, | 
					
						
							|  |  |  |                               recursive: true) | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-11-12 13:50:53 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-29 18:09:51 -07:00
										 |  |  |         dep_stack.pop | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-06-09 12:59:42 -05:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  | end |