| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  | # typed: true | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  | require "formula" | 
					
						
							| 
									
										
										
										
											2019-04-17 18:25:08 +09:00
										 |  |  | require "cli/parser" | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2020-07-23 02:00:44 +02:00
										 |  |  |   extend DependenciesHelpers | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-20 12:03:48 +02:00
										 |  |  |   sig { returns(CLI::Parser) } | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   def self.deps_args | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |     Homebrew::CLI::Parser.new do | 
					
						
							| 
									
										
										
										
											2021-01-15 15:04:02 -05:00
										 |  |  |       description <<~EOS | 
					
						
							| 
									
										
										
										
											2023-08-30 08:46:59 +01:00
										 |  |  |         Show dependencies for <formula>. When given multiple formula arguments, | 
					
						
							| 
									
										
										
										
											2023-08-28 22:14:26 -07:00
										 |  |  |         show the intersection of dependencies for each formula. By default, `deps` | 
					
						
							|  |  |  |         shows all required and recommended dependencies. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-09 11:06:07 -04:00
										 |  |  |         If any version of each formula argument is installed and no other options | 
					
						
							|  |  |  |         are passed, this command displays their actual runtime dependencies (similar | 
					
						
							|  |  |  |         to `brew linkage`), which may differ from the current versons' stated | 
					
						
							|  |  |  |         dependencies if the installed versions are outdated. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-09 12:09:38 -04:00
										 |  |  |         *Note:* `--missing` and `--skip-recommended` have precedence over `--include-*`. | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       EOS | 
					
						
							| 
									
										
										
										
											2022-08-31 17:21:44 +01:00
										 |  |  |       switch "-n", "--topological", | 
					
						
							| 
									
										
										
										
											2019-08-20 00:04:14 -04:00
										 |  |  |              description: "Sort dependencies in topological order." | 
					
						
							| 
									
										
										
										
											2022-08-31 17:21:44 +01:00
										 |  |  |       switch "-1", "--direct", "--declared", "--1", | 
					
						
							|  |  |  |              description: "Show only the direct dependencies declared in the formula." | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       switch "--union", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              description: "Show the union of dependencies for multiple <formula>, instead of the intersection." | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       switch "--full-name", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              description: "List dependencies by their full name." | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       switch "--include-build", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              description: "Include `:build` dependencies for <formula>." | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       switch "--include-optional", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              description: "Include `:optional` dependencies for <formula>." | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       switch "--include-test", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              description: "Include `:test` dependencies for <formula> (non-recursive)." | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       switch "--skip-recommended", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              description: "Skip `:recommended` dependencies for <formula>." | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       switch "--include-requirements", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              description: "Include requirements in addition to dependencies for <formula>." | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       switch "--tree", | 
					
						
							| 
									
										
										
										
											2022-06-28 10:09:59 +01:00
										 |  |  |              description: "Show dependencies as a tree. When given multiple formula arguments, " \ | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |                           "show individual trees for each formula." | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |       switch "--graph", | 
					
						
							|  |  |  |              description: "Show dependencies as a directed graph." | 
					
						
							|  |  |  |       switch "--dot", | 
					
						
							|  |  |  |              depends_on:  "--graph", | 
					
						
							|  |  |  |              description: "Show text-based graph description in DOT format." | 
					
						
							| 
									
										
										
										
											2019-03-06 23:44:46 -05:00
										 |  |  |       switch "--annotate", | 
					
						
							| 
									
										
										
										
											2023-10-12 07:58:25 -04:00
										 |  |  |              description: "Mark any build, test, implicit, optional, or recommended dependencies as " \ | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |                           "such in the output." | 
					
						
							| 
									
										
										
										
											2019-03-14 16:44:17 -04:00
										 |  |  |       switch "--installed", | 
					
						
							| 
									
										
										
										
											2022-06-28 10:09:59 +01:00
										 |  |  |              description: "List dependencies for formulae that are currently installed. If <formula> is " \ | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |                           "specified, list only its dependencies that are currently installed." | 
					
						
							| 
									
										
										
										
											2023-05-17 14:53:19 -05:00
										 |  |  |       switch "--missing", | 
					
						
							|  |  |  |              description: "Show only missing dependencies." | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |       switch "--eval-all", | 
					
						
							|  |  |  |              description: "Evaluate all available formulae and casks, whether installed or not, to list " \ | 
					
						
							|  |  |  |                           "their dependencies." | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       switch "--for-each", | 
					
						
							| 
									
										
										
										
											2023-10-12 07:58:25 -04:00
										 |  |  |              description: "Switch into the mode used by the `--eval-all` option, but only list dependencies " \ | 
					
						
							| 
									
										
										
										
											2022-06-28 10:09:59 +01:00
										 |  |  |                           "for each provided <formula>, one formula per line. This is used for " \ | 
					
						
							| 
									
										
										
										
											2023-10-12 07:58:25 -04:00
										 |  |  |                           "debugging the `--installed`/`--eval-all` display mode." | 
					
						
							|  |  |  |       switch "--HEAD", | 
					
						
							|  |  |  |              description: "Show dependencies for HEAD version instead of stable version." | 
					
						
							| 
									
										
										
										
											2020-12-04 10:11:17 +09:00
										 |  |  |       switch "--formula", "--formulae", | 
					
						
							|  |  |  |              description: "Treat all named arguments as formulae." | 
					
						
							|  |  |  |       switch "--cask", "--casks", | 
					
						
							|  |  |  |              description: "Treat all named arguments as casks." | 
					
						
							| 
									
										
										
										
											2020-11-27 11:41:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |       conflicts "--tree", "--graph" | 
					
						
							| 
									
										
										
										
											2023-05-17 14:53:19 -05:00
										 |  |  |       conflicts "--installed", "--missing" | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |       conflicts "--installed", "--eval-all" | 
					
						
							| 
									
										
										
										
											2020-12-04 10:11:17 +09:00
										 |  |  |       conflicts "--formula", "--cask" | 
					
						
							| 
									
										
										
										
											2019-01-30 13:18:02 +00:00
										 |  |  |       formula_options | 
					
						
							| 
									
										
										
										
											2021-01-10 14:26:40 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |       named_args [:formula, :cask] | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   def self.deps | 
					
						
							| 
									
										
										
										
											2020-07-30 18:40:10 +02:00
										 |  |  |     args = deps_args.parse | 
					
						
							| 
									
										
										
										
											2019-11-06 10:20:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |     all = args.eval_all? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-06 10:20:33 +00:00
										 |  |  |     Formulary.enable_factory_cache! | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-31 17:21:44 +01:00
										 |  |  |     recursive = !args.direct? | 
					
						
							| 
									
										
										
										
											2020-08-19 10:34:48 -04:00
										 |  |  |     installed = args.installed? || dependents(args.named.to_formulae_and_casks).all?(&:any_version_installed?) | 
					
						
							| 
									
										
										
										
											2020-02-12 22:30:44 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @use_runtime_dependencies = installed && recursive && | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  |                                 !args.tree? && | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |                                 !args.graph? && | 
					
						
							| 
									
										
										
										
											2023-10-12 07:58:25 -04:00
										 |  |  |                                 !args.HEAD? && | 
					
						
							| 
									
										
										
										
											2020-02-12 22:30:44 +01:00
										 |  |  |                                 !args.include_build? && | 
					
						
							|  |  |  |                                 !args.include_test? && | 
					
						
							|  |  |  |                                 !args.include_optional? && | 
					
						
							| 
									
										
										
										
											2023-08-24 21:18:24 -07:00
										 |  |  |                                 !args.skip_recommended? && | 
					
						
							|  |  |  |                                 !args.missing? | 
					
						
							| 
									
										
										
										
											2019-11-06 10:20:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |     if args.tree? || args.graph? | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  |       dependents = if args.named.present? | 
					
						
							| 
									
										
										
										
											2020-08-19 10:34:48 -04:00
										 |  |  |         sorted_dependents(args.named.to_formulae_and_casks) | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  |       elsif args.installed? | 
					
						
							| 
									
										
										
										
											2020-12-29 21:06:32 +09:00
										 |  |  |         case args.only_formula_or_cask | 
					
						
							|  |  |  |         when :formula | 
					
						
							| 
									
										
										
										
											2020-12-04 10:11:17 +09:00
										 |  |  |           sorted_dependents(Formula.installed) | 
					
						
							| 
									
										
										
										
											2020-12-29 21:06:32 +09:00
										 |  |  |         when :cask | 
					
						
							| 
									
										
										
										
											2021-03-22 22:26:06 +00:00
										 |  |  |           sorted_dependents(Cask::Caskroom.casks) | 
					
						
							| 
									
										
										
										
											2020-12-04 10:11:17 +09:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2021-03-22 22:26:06 +00:00
										 |  |  |           sorted_dependents(Formula.installed + Cask::Caskroom.casks) | 
					
						
							| 
									
										
										
										
											2020-12-04 10:11:17 +09:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  |         raise FormulaUnspecifiedError | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |       if args.graph? | 
					
						
							|  |  |  |         dot_code = dot_code(dependents, recursive: recursive, args: args) | 
					
						
							|  |  |  |         if args.dot? | 
					
						
							|  |  |  |           puts dot_code | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           exec_browser "https://dreampuf.github.io/GraphvizOnline/##{ERB::Util.url_encode(dot_code)}" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 17:12:32 +01:00
										 |  |  |       puts_deps_tree dependents, recursive: recursive, args: args | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  |       return | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |     elsif all | 
					
						
							| 
									
										
										
										
											2023-09-02 19:35:22 +09:00
										 |  |  |       puts_deps sorted_dependents(Formula.all(eval_all: args.eval_all?) + Cask::Cask.all), recursive: recursive, | 
					
						
							|  |  |  |                                                                                            args:      args | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  |       return | 
					
						
							| 
									
										
										
										
											2020-03-04 17:28:15 +00:00
										 |  |  |     elsif !args.no_named? && args.for_each? | 
					
						
							| 
									
										
										
										
											2020-08-19 10:34:48 -04:00
										 |  |  |       puts_deps sorted_dependents(args.named.to_formulae_and_casks), recursive: recursive, args: args | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  |       return | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-04 17:28:15 +00:00
										 |  |  |     if args.no_named? | 
					
						
							| 
									
										
										
										
											2019-11-06 10:20:20 +00:00
										 |  |  |       raise FormulaUnspecifiedError unless args.installed? | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 21:06:32 +09:00
										 |  |  |       sorted_dependents_formulae_and_casks = case args.only_formula_or_cask | 
					
						
							|  |  |  |       when :formula | 
					
						
							| 
									
										
										
										
											2020-12-04 10:11:17 +09:00
										 |  |  |         sorted_dependents(Formula.installed) | 
					
						
							| 
									
										
										
										
											2020-12-29 21:06:32 +09:00
										 |  |  |       when :cask | 
					
						
							| 
									
										
										
										
											2021-03-22 22:26:06 +00:00
										 |  |  |         sorted_dependents(Cask::Caskroom.casks) | 
					
						
							| 
									
										
										
										
											2020-12-04 10:11:17 +09:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2021-03-22 22:26:06 +00:00
										 |  |  |         sorted_dependents(Formula.installed + Cask::Caskroom.casks) | 
					
						
							| 
									
										
										
										
											2020-12-04 10:11:17 +09:00
										 |  |  |       end | 
					
						
							|  |  |  |       puts_deps sorted_dependents_formulae_and_casks, recursive: recursive, args: args | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  |       return | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 10:34:48 -04:00
										 |  |  |     dependents = dependents(args.named.to_formulae_and_casks) | 
					
						
							| 
									
										
										
										
											2023-10-12 07:58:25 -04:00
										 |  |  |     check_head_spec(dependents) if args.HEAD? | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 17:12:32 +01:00
										 |  |  |     all_deps = deps_for_dependents(dependents, recursive: recursive, args: args, &(args.union? ? :| : :&)) | 
					
						
							| 
									
										
										
										
											2020-08-01 12:52:37 -04:00
										 |  |  |     condense_requirements(all_deps, args: args) | 
					
						
							| 
									
										
										
										
											2020-07-31 19:09:53 +02:00
										 |  |  |     all_deps.map! { |d| dep_display_name(d, args: args) } | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  |     all_deps.uniq! | 
					
						
							| 
									
										
										
										
											2022-08-31 17:21:44 +01:00
										 |  |  |     all_deps.sort! unless args.topological? | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  |     puts all_deps | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2013-06-09 12:59:42 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   def self.sorted_dependents(formulae_or_casks) | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  |     dependents(formulae_or_casks).sort_by(&:name) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   def self.condense_requirements(deps, args:) | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  |     deps.select! { |dep| dep.is_a?(Dependency) } unless args.include_requirements? | 
					
						
							|  |  |  |     deps.select! { |dep| dep.is_a?(Requirement) || dep.installed? } if args.installed? | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   def self.dep_display_name(dep, args:) | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |     str = if dep.is_a? Requirement | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |       if args.include_requirements? | 
					
						
							| 
									
										
										
										
											2018-01-14 13:27:43 +00:00
										 |  |  |         ":#{dep.display_s}" | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |       else | 
					
						
							|  |  |  |         # This shouldn't happen, but we'll put something here to help debugging | 
					
						
							|  |  |  |         "::#{dep.name}" | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |     elsif args.full_name? | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  |       dep.to_formula.full_name | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  |       dep.name | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-23 08:34:24 +05:30
										 |  |  |     if args.annotate? | 
					
						
							| 
									
										
										
										
											2019-03-06 23:44:46 -05:00
										 |  |  |       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? | 
					
						
							| 
									
										
										
										
											2023-06-19 06:06:15 +01:00
										 |  |  |       str = "#{str} [implicit]" if dep.implicit? | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |     str | 
					
						
							| 
									
										
										
										
											2016-12-20 03:39:30 -05:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-08 00:27:19 +00:00
										 |  |  |   def self.deps_for_dependent(dependency, args:, recursive: false) | 
					
						
							| 
									
										
										
										
											2020-07-31 19:09:53 +02:00
										 |  |  |     includes, ignores = args_includes_ignores(args) | 
					
						
							| 
									
										
										
										
											2015-03-20 21:31:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-08 00:27:19 +00:00
										 |  |  |     deps = dependency.runtime_dependencies if @use_runtime_dependencies | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-29 17:27:21 -04:00
										 |  |  |     if recursive | 
					
						
							| 
									
										
										
										
											2023-03-08 00:27:19 +00:00
										 |  |  |       deps ||= recursive_includes(Dependency, dependency, includes, ignores) | 
					
						
							|  |  |  |       reqs   = recursive_includes(Requirement, dependency, includes, ignores) | 
					
						
							| 
									
										
										
										
											2013-10-29 17:27:21 -04:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2023-08-24 21:18:24 -07:00
										 |  |  |       deps ||= select_includes(dependency.deps, ignores, includes) | 
					
						
							|  |  |  |       reqs   = select_includes(dependency.requirements, ignores, includes) | 
					
						
							| 
									
										
										
										
											2013-10-29 17:27:21 -04:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2014-02-27 12:56:42 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |     deps + reqs.to_a | 
					
						
							| 
									
										
										
										
											2013-10-29 17:27:21 -04:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   def self.deps_for_dependents(dependents, args:, recursive: false, &block) | 
					
						
							| 
									
										
										
										
											2020-08-19 17:12:32 +01:00
										 |  |  |     dependents.map { |d| deps_for_dependent(d, recursive: recursive, args: args) }.reduce(&block) | 
					
						
							| 
									
										
										
										
											2013-06-22 12:54:45 -05:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-12 07:58:25 -04:00
										 |  |  |   def self.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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   def self.puts_deps(dependents, args:, recursive: false) | 
					
						
							| 
									
										
										
										
											2023-10-12 07:58:25 -04:00
										 |  |  |     check_head_spec(dependents) if args.HEAD? | 
					
						
							| 
									
										
										
										
											2020-07-31 19:09:53 +02:00
										 |  |  |     dependents.each do |dependent| | 
					
						
							| 
									
										
										
										
											2020-08-19 17:12:32 +01:00
										 |  |  |       deps = deps_for_dependent(dependent, recursive: recursive, args: args) | 
					
						
							| 
									
										
										
										
											2020-08-01 12:52:37 -04:00
										 |  |  |       condense_requirements(deps, args: args) | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  |       deps.sort_by!(&:name) | 
					
						
							| 
									
										
										
										
											2020-07-31 19:09:53 +02:00
										 |  |  |       deps.map! { |d| dep_display_name(d, args: args) } | 
					
						
							|  |  |  |       puts "#{dependent.full_name}: #{deps.join(" ")}" | 
					
						
							| 
									
										
										
										
											2016-12-20 03:39:30 -05:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2013-06-22 12:54:45 -05:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   def self.dot_code(dependents, recursive:, args:) | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |     dep_graph = {} | 
					
						
							|  |  |  |     dependents.each do |d| | 
					
						
							|  |  |  |       graph_deps(d, dep_graph: dep_graph, recursive: recursive, args: args) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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? | 
					
						
							| 
									
										
										
										
											2022-03-19 16:21:01 +09:00
										 |  |  |         "  \"#{d.name}\" -> \"#{dep}\"#{" [#{attributes.join(", ")}]" if attributes.any?}#{comment}" | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |       end | 
					
						
							|  |  |  |     end.flatten.join("\n") | 
					
						
							|  |  |  |     "digraph {\n#{dot_code}\n}" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |   def self.graph_deps(formula, dep_graph:, recursive:, args:) | 
					
						
							|  |  |  |     return if dep_graph.key?(formula) | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |     dependables = dependables(formula, args: args) | 
					
						
							|  |  |  |     dep_graph[formula] = dependables | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |     return unless recursive | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     dependables.each do |dep| | 
					
						
							|  |  |  |       next unless dep.is_a? Dependency | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       graph_deps(Formulary.factory(dep.name), | 
					
						
							|  |  |  |                  dep_graph: dep_graph, | 
					
						
							|  |  |  |                  recursive: true, | 
					
						
							|  |  |  |                  args:      args) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   def self.puts_deps_tree(dependents, args:, recursive: false) | 
					
						
							| 
									
										
										
										
											2023-10-12 07:58:25 -04:00
										 |  |  |     check_head_spec(dependents) if args.HEAD? | 
					
						
							| 
									
										
										
										
											2020-07-27 19:12:16 -04:00
										 |  |  |     dependents.each do |d| | 
					
						
							|  |  |  |       puts d.full_name | 
					
						
							| 
									
										
										
										
											2021-11-12 13:50:53 -08:00
										 |  |  |       recursive_deps_tree(d, dep_stack: [], prefix: "", recursive: recursive, args: args) | 
					
						
							| 
									
										
										
										
											2013-06-22 12:54:45 -05:00
										 |  |  |       puts | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |   def self.dependables(formula, args:) | 
					
						
							| 
									
										
										
										
											2020-07-31 19:09:53 +02:00
										 |  |  |     includes, ignores = args_includes_ignores(args) | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |     deps = @use_runtime_dependencies ? formula.runtime_dependencies : formula.deps | 
					
						
							| 
									
										
										
										
											2023-08-24 21:18:24 -07:00
										 |  |  |     deps = select_includes(deps, ignores, includes) | 
					
						
							|  |  |  |     reqs = select_includes(formula.requirements, ignores, includes) if args.include_requirements? | 
					
						
							| 
									
										
										
										
											2021-11-12 13:52:20 -08:00
										 |  |  |     reqs ||= [] | 
					
						
							|  |  |  |     reqs + deps | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2019-12-27 17:35:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |   def self.recursive_deps_tree(formula, dep_stack:, prefix:, recursive:, args:) | 
					
						
							|  |  |  |     dependables = dependables(formula, args: args) | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |     max = dependables.length - 1
 | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |     dep_stack.push formula.name | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |     dependables.each_with_index do |dep, i| | 
					
						
							|  |  |  |       tree_lines = if i == max | 
					
						
							| 
									
										
										
										
											2016-12-20 03:59:15 -05:00
										 |  |  |         "└──" | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         "├──" | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 19:09:53 +02:00
										 |  |  |       display_s = "#{tree_lines} #{dep_display_name(dep, args: args)}" | 
					
						
							| 
									
										
										
										
											2022-08-30 12:21:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       # Detect circular dependencies and consider them a failure if present. | 
					
						
							| 
									
										
										
										
											2021-11-12 13:50:53 -08:00
										 |  |  |       is_circular = dep_stack.include?(dep.name) | 
					
						
							| 
									
										
										
										
											2022-08-30 12:21:02 +00:00
										 |  |  |       if is_circular | 
					
						
							|  |  |  |         display_s = "#{display_s} (CIRCULAR DEPENDENCY)" | 
					
						
							|  |  |  |         Homebrew.failed = true | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |       puts "#{prefix}#{display_s}" | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |       next if !recursive || is_circular | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |       prefix_addition = if i == max | 
					
						
							|  |  |  |         "    " | 
					
						
							| 
									
										
										
										
											2016-12-20 03:59:15 -05:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2017-08-03 00:41:51 -04:00
										 |  |  |         "│   " | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-12 13:50:53 -08:00
										 |  |  |       next unless dep.is_a? Dependency | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       recursive_deps_tree(Formulary.factory(dep.name), | 
					
						
							|  |  |  |                           dep_stack: dep_stack, | 
					
						
							|  |  |  |                           prefix:    prefix + prefix_addition, | 
					
						
							|  |  |  |                           recursive: true, | 
					
						
							|  |  |  |                           args:      args) | 
					
						
							| 
									
										
										
										
											2013-06-09 12:59:42 -05:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-03-26 10:55:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-12 13:50:53 -08:00
										 |  |  |     dep_stack.pop | 
					
						
							| 
									
										
										
										
											2013-06-09 12:59:42 -05:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  | end |