| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  | # typed: true | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 17:02:08 +02:00
										 |  |  | require "missing_formula" | 
					
						
							| 
									
										
										
										
											2014-06-19 17:52:42 -05:00
										 |  |  | require "caveats" | 
					
						
							| 
									
										
										
										
											2019-04-17 18:25:08 +09:00
										 |  |  | require "cli/parser" | 
					
						
							| 
									
										
										
										
											2015-12-27 19:12:27 +01:00
										 |  |  | require "options" | 
					
						
							| 
									
										
										
										
											2014-06-19 17:52:42 -05:00
										 |  |  | require "formula" | 
					
						
							|  |  |  | require "keg" | 
					
						
							|  |  |  | require "tab" | 
					
						
							| 
									
										
										
										
											2016-11-20 13:00:01 -05:00
										 |  |  | require "json" | 
					
						
							| 
									
										
										
										
											2020-08-18 10:58:32 -04:00
										 |  |  | require "utils/spdx" | 
					
						
							| 
									
										
										
										
											2020-09-07 13:00:02 -04:00
										 |  |  | require "deprecate_disable" | 
					
						
							| 
									
										
										
										
											2021-08-06 02:30:44 -04:00
										 |  |  | require "api" | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 22:41:47 -05:00
										 |  |  | module Homebrew | 
					
						
							| 
									
										
										
										
											2020-10-20 12:03:48 +02:00
										 |  |  |   extend T::Sig | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-26 01:44:51 +02:00
										 |  |  |   module_function | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-20 17:20:09 -05:00
										 |  |  |   VALID_DAYS = %w[30 90 365].freeze | 
					
						
							|  |  |  |   VALID_FORMULA_CATEGORIES = %w[install install-on-request build-error].freeze | 
					
						
							|  |  |  |   VALID_CATEGORIES = (VALID_FORMULA_CATEGORIES + %w[cask-install os-version]).freeze | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-20 12:03:48 +02:00
										 |  |  |   sig { returns(CLI::Parser) } | 
					
						
							| 
									
										
										
										
											2018-11-07 19:31:20 +05:30
										 |  |  |   def info_args | 
					
						
							|  |  |  |     Homebrew::CLI::Parser.new do | 
					
						
							| 
									
										
										
										
											2021-01-15 15:04:02 -05:00
										 |  |  |       description <<~EOS | 
					
						
							| 
									
										
										
										
											2018-11-07 19:31:20 +05:30
										 |  |  |         Display brief statistics for your Homebrew installation. | 
					
						
							| 
									
										
										
										
											2019-03-11 12:59:39 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-20 10:26:10 +01:00
										 |  |  |         If a <formula> or <cask> is provided, show summary of information about it. | 
					
						
							| 
									
										
										
										
											2018-11-07 19:31:20 +05:30
										 |  |  |       EOS | 
					
						
							|  |  |  |       switch "--analytics", | 
					
						
							| 
									
										
										
										
											2022-06-28 10:09:59 +01:00
										 |  |  |              description: "List global Homebrew analytics data or, if specified, installation and " \ | 
					
						
							|  |  |  |                           "build error data for <formula> (provided neither `HOMEBREW_NO_ANALYTICS` " \ | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |                           "nor `HOMEBREW_NO_GITHUB_API` are set)." | 
					
						
							| 
									
										
										
										
											2020-06-20 01:38:37 +05:30
										 |  |  |       flag   "--days=", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              depends_on:  "--analytics", | 
					
						
							| 
									
										
										
										
											2022-06-28 10:09:59 +01:00
										 |  |  |              description: "How many days of analytics data to retrieve. " \ | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |                           "The value for <days> must be `30`, `90` or `365`. The default is `30`." | 
					
						
							| 
									
										
										
										
											2020-06-20 01:38:37 +05:30
										 |  |  |       flag   "--category=", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              depends_on:  "--analytics", | 
					
						
							| 
									
										
										
										
											2022-06-28 10:09:59 +01:00
										 |  |  |              description: "Which type of analytics data to retrieve. " \ | 
					
						
							|  |  |  |                           "The value for <category> must be `install`, `install-on-request` or `build-error`; " \ | 
					
						
							|  |  |  |                           "`cask-install` or `os-version` may be specified if <formula> is not. " \ | 
					
						
							| 
									
										
										
										
											2019-11-20 17:20:09 -05:00
										 |  |  |                           "The default is `install`." | 
					
						
							| 
									
										
										
										
											2018-11-07 19:31:20 +05:30
										 |  |  |       switch "--github", | 
					
						
							| 
									
										
										
										
											2022-06-28 10:09:59 +01:00
										 |  |  |              description: "Open the GitHub source page for <formula> and <cask> in a browser. " \ | 
					
						
							| 
									
										
										
										
											2021-10-01 09:38:31 +09:00
										 |  |  |                           "To view the history locally: `brew log -p` <formula> or <cask>" | 
					
						
							| 
									
										
										
										
											2019-08-06 13:23:19 -04:00
										 |  |  |       flag   "--json", | 
					
						
							| 
									
										
										
										
											2022-06-28 10:09:59 +01:00
										 |  |  |              description: "Print a JSON representation. Currently the default value for <version> is `v1` for " \ | 
					
						
							|  |  |  |                           "<formula>. For <formula> and <cask> use `v2`. See the docs for examples of using the " \ | 
					
						
							| 
									
										
										
										
											2020-11-25 16:32:10 +00:00
										 |  |  |                           "JSON output: <https://docs.brew.sh/Querying-Brew>" | 
					
						
							| 
									
										
										
										
											2018-11-07 19:31:20 +05:30
										 |  |  |       switch "--installed", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              depends_on:  "--json", | 
					
						
							|  |  |  |              description: "Print JSON of formulae that are currently installed." | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |       switch "--eval-all", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              depends_on:  "--json", | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |              description: "Evaluate all available formulae and casks, whether installed or not, to print their " \ | 
					
						
							| 
									
										
										
										
											2023-02-10 23:15:40 -05:00
										 |  |  |                           "JSON. Implied if `HOMEBREW_EVAL_ALL` is set." | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |       switch "--all", | 
					
						
							|  |  |  |              hidden:     true, | 
					
						
							|  |  |  |              depends_on: "--json" | 
					
						
							| 
									
										
										
										
											2022-07-21 16:41:15 +02:00
										 |  |  |       switch "--variations", | 
					
						
							|  |  |  |              depends_on:  "--json", | 
					
						
							|  |  |  |              description: "Include the variations hash in each formula's JSON output." | 
					
						
							| 
									
										
										
										
											2020-07-30 18:40:10 +02:00
										 |  |  |       switch "-v", "--verbose", | 
					
						
							| 
									
										
										
										
											2019-04-30 08:44:35 +01:00
										 |  |  |              description: "Show more verbose analytics data for <formula>." | 
					
						
							| 
									
										
										
										
											2020-11-20 10:26:10 +01:00
										 |  |  |       switch "--formula", "--formulae", | 
					
						
							|  |  |  |              description: "Treat all named arguments as formulae." | 
					
						
							|  |  |  |       switch "--cask", "--casks", | 
					
						
							|  |  |  |              description: "Treat all named arguments as casks." | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |       conflicts "--installed", "--eval-all" | 
					
						
							| 
									
										
										
										
											2019-03-11 12:59:39 -04:00
										 |  |  |       conflicts "--installed", "--all" | 
					
						
							| 
									
										
										
										
											2020-11-27 11:41:08 -05:00
										 |  |  |       conflicts "--formula", "--cask" | 
					
						
							| 
									
										
										
										
											2021-01-10 14:26:40 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |       named_args [:formula, :cask] | 
					
						
							| 
									
										
										
										
											2018-11-07 19:31:20 +05:30
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-20 10:26:10 +01:00
										 |  |  |   sig { void } | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |   def info | 
					
						
							| 
									
										
										
										
											2020-07-30 18:40:10 +02:00
										 |  |  |     args = info_args.parse | 
					
						
							| 
									
										
										
										
											2019-12-13 15:39:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |     if args.analytics? | 
					
						
							| 
									
										
										
										
											2020-12-01 17:04:59 +00:00
										 |  |  |       if args.days.present? && VALID_DAYS.exclude?(args.days) | 
					
						
							| 
									
										
										
										
											2023-02-10 23:15:40 -05:00
										 |  |  |         raise UsageError, "`--days` must be one of #{VALID_DAYS.join(", ")}." | 
					
						
							| 
									
										
										
										
											2019-11-20 17:20:09 -05:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |       if args.category.present? | 
					
						
							| 
									
										
										
										
											2020-12-01 17:04:59 +00:00
										 |  |  |         if args.named.present? && VALID_FORMULA_CATEGORIES.exclude?(args.category) | 
					
						
							| 
									
										
										
										
											2023-02-10 23:15:40 -05:00
										 |  |  |           raise UsageError, | 
					
						
							|  |  |  |                 "`--category` must be one of #{VALID_FORMULA_CATEGORIES.join(", ")} when querying formulae." | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2019-11-20 17:20:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |         unless VALID_CATEGORIES.include?(args.category) | 
					
						
							| 
									
										
										
										
											2023-02-10 23:15:40 -05:00
										 |  |  |           raise UsageError, "`--category` must be one of #{VALID_CATEGORIES.join(", ")}." | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2019-11-20 17:20:09 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2019-02-19 13:12:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-17 21:14:18 +09:00
										 |  |  |       print_analytics(args: args) | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |     elsif args.json | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |       all = args.eval_all? | 
					
						
							|  |  |  |       if !all && args.all? && !Homebrew::EnvConfig.eval_all? | 
					
						
							| 
									
										
										
										
											2023-02-07 19:25:51 +01:00
										 |  |  |         odisabled "brew info --all", "brew info --eval-all or HOMEBREW_EVAL_ALL" | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |         all = true | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       print_json(all, args: args) | 
					
						
							| 
									
										
										
										
											2018-11-07 19:31:20 +05:30
										 |  |  |     elsif args.github? | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |       raise FormulaOrCaskUnspecifiedError if args.no_named? | 
					
						
							| 
									
										
										
										
											2019-12-13 15:44:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-17 21:14:18 +09:00
										 |  |  |       exec_browser(*args.named.to_formulae_and_casks.map { |f| github_info(f) }) | 
					
						
							| 
									
										
										
										
											2020-11-09 12:29:33 -05:00
										 |  |  |     elsif args.no_named? | 
					
						
							|  |  |  |       print_statistics | 
					
						
							| 
									
										
										
										
											2012-08-15 22:08:40 -05:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2020-12-17 21:14:18 +09:00
										 |  |  |       print_info(args: args) | 
					
						
							| 
									
										
										
										
											2012-08-15 22:08:40 -05:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-20 10:26:10 +01:00
										 |  |  |   sig { void } | 
					
						
							| 
									
										
										
										
											2020-11-09 12:29:33 -05:00
										 |  |  |   def print_statistics | 
					
						
							|  |  |  |     return unless HOMEBREW_CELLAR.exist? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     count = Formula.racks.length | 
					
						
							| 
									
										
										
										
											2023-02-27 20:16:34 -08:00
										 |  |  |     puts "#{count} #{Utils.pluralize("keg", count)}, #{HOMEBREW_CELLAR.dup.abv}" | 
					
						
							| 
									
										
										
										
											2020-11-09 12:29:33 -05:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-17 21:14:18 +09:00
										 |  |  |   sig { params(args: CLI::Args).void } | 
					
						
							|  |  |  |   def print_analytics(args:) | 
					
						
							| 
									
										
										
										
											2020-03-04 17:28:15 +00:00
										 |  |  |     if args.no_named? | 
					
						
							| 
									
										
										
										
											2020-11-09 12:29:33 -05:00
										 |  |  |       Utils::Analytics.output(args: args) | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |       return | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-17 21:14:18 +09:00
										 |  |  |     args.named.to_formulae_and_casks_and_unavailable.each_with_index do |obj, i| | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |       puts unless i.zero? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case obj | 
					
						
							|  |  |  |       when Formula | 
					
						
							|  |  |  |         Utils::Analytics.formula_output(obj, args: args) | 
					
						
							|  |  |  |       when Cask::Cask | 
					
						
							|  |  |  |         Utils::Analytics.cask_output(obj, args: args) | 
					
						
							|  |  |  |       when FormulaOrCaskUnavailableError | 
					
						
							|  |  |  |         Utils::Analytics.output(filter: obj.name, args: args) | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         raise | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-17 21:14:18 +09:00
										 |  |  |   sig { params(args: CLI::Args).void } | 
					
						
							|  |  |  |   def print_info(args:) | 
					
						
							|  |  |  |     args.named.to_formulae_and_casks_and_unavailable.each_with_index do |obj, i| | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |       puts unless i.zero? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case obj | 
					
						
							|  |  |  |       when Formula | 
					
						
							|  |  |  |         info_formula(obj, args: args) | 
					
						
							|  |  |  |       when Cask::Cask | 
					
						
							|  |  |  |         info_cask(obj, args: args) | 
					
						
							| 
									
										
										
										
											2021-09-17 00:08:19 +01:00
										 |  |  |       when FormulaUnreadableError, FormulaClassUnavailableError, | 
					
						
							|  |  |  |          TapFormulaUnreadableError, TapFormulaClassUnavailableError, | 
					
						
							|  |  |  |          Cask::CaskUnreadableError | 
					
						
							|  |  |  |         # We found the formula/cask, but failed to read it | 
					
						
							|  |  |  |         $stderr.puts obj.backtrace if Homebrew::EnvConfig.developer? | 
					
						
							|  |  |  |         ofail obj.message | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |       when FormulaOrCaskUnavailableError | 
					
						
							| 
									
										
										
										
											2021-09-17 00:08:19 +01:00
										 |  |  |         # The formula/cask could not be found | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |         ofail obj.message | 
					
						
							|  |  |  |         # No formula with this name, try a missing formula lookup | 
					
						
							|  |  |  |         if (reason = MissingFormula.reason(obj.name, show_info: true)) | 
					
						
							|  |  |  |           $stderr.puts reason | 
					
						
							| 
									
										
										
										
											2013-04-29 10:12:40 -07:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |       else | 
					
						
							|  |  |  |         raise | 
					
						
							| 
									
										
										
										
											2013-04-29 10:12:40 -07:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-09 21:09:07 -04:00
										 |  |  |   def json_version(version) | 
					
						
							|  |  |  |     version_hash = { | 
					
						
							|  |  |  |       true => :default, | 
					
						
							|  |  |  |       "v1" => :v1, | 
					
						
							|  |  |  |       "v2" => :v2, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     raise UsageError, "invalid JSON version: #{version}" unless version_hash.include?(version) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     version_hash[version] | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 09:49:53 -08:00
										 |  |  |   sig { params(all: T::Boolean, args: T.untyped).void } | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |   def print_json(all, args:) | 
					
						
							|  |  |  |     raise FormulaOrCaskUnspecifiedError if !(all || args.installed?) && args.no_named? | 
					
						
							| 
									
										
										
										
											2020-10-09 21:09:07 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     json = case json_version(args.json) | 
					
						
							|  |  |  |     when :v1, :default | 
					
						
							| 
									
										
										
										
											2023-02-10 23:15:40 -05:00
										 |  |  |       raise UsageError, "Cannot specify `--cask` when using `--json=v1`!" if args.cask? | 
					
						
							| 
									
										
										
										
											2020-11-25 16:32:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |       formulae = if all | 
					
						
							| 
									
										
										
										
											2022-05-04 16:37:22 +08:00
										 |  |  |         Formula.all.sort | 
					
						
							| 
									
										
										
										
											2020-10-09 21:09:07 -04:00
										 |  |  |       elsif args.installed? | 
					
						
							|  |  |  |         Formula.installed.sort | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         args.named.to_formulae | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 12:07:36 -05:00
										 |  |  |       if args.variations? | 
					
						
							| 
									
										
										
										
											2022-07-21 16:41:15 +02:00
										 |  |  |         formulae.map(&:to_hash_with_variations) | 
					
						
							| 
									
										
										
										
											2021-06-05 12:27:24 -04:00
										 |  |  |       else | 
					
						
							|  |  |  |         formulae.map(&:to_hash) | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-10-09 21:09:07 -04:00
										 |  |  |     when :v2 | 
					
						
							| 
									
										
										
										
											2022-09-05 13:57:22 +01:00
										 |  |  |       formulae, casks = if all | 
					
						
							| 
									
										
										
										
											2022-05-04 16:37:22 +08:00
										 |  |  |         [Formula.all.sort, Cask::Cask.all.sort_by(&:full_name)] | 
					
						
							| 
									
										
										
										
											2020-10-09 21:09:07 -04:00
										 |  |  |       elsif args.installed? | 
					
						
							|  |  |  |         [Formula.installed.sort, Cask::Caskroom.casks.sort_by(&:full_name)] | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2020-12-17 21:14:18 +09:00
										 |  |  |         args.named.to_formulae_to_casks | 
					
						
							| 
									
										
										
										
											2020-10-09 21:09:07 -04:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 12:07:36 -05:00
										 |  |  |       if args.variations? | 
					
						
							| 
									
										
										
										
											2022-07-21 16:41:15 +02:00
										 |  |  |         { | 
					
						
							|  |  |  |           "formulae" => formulae.map(&:to_hash_with_variations), | 
					
						
							|  |  |  |           "casks"    => casks.map(&:to_hash_with_variations), | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-05 12:27:24 -04:00
										 |  |  |       else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           "formulae" => formulae.map(&:to_hash), | 
					
						
							|  |  |  |           "casks"    => casks.map(&:to_h), | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2020-10-09 21:09:07 -04:00
										 |  |  |       raise | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2020-10-09 21:09:07 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 14:06:21 +01:00
										 |  |  |     puts JSON.pretty_generate(json) | 
					
						
							| 
									
										
										
										
											2012-08-15 22:08:40 -05:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-08 15:57:48 +08:00
										 |  |  |   def github_remote_path(remote, path) | 
					
						
							|  |  |  |     if remote =~ %r{^(?:https?://|git(?:@|://))github\.com[:/](.+)/(.+?)(?:\.git)?$} | 
					
						
							| 
									
										
										
										
											2020-06-25 11:20:57 +01:00
										 |  |  |       "https://github.com/#{Regexp.last_match(1)}/#{Regexp.last_match(2)}/blob/HEAD/#{path}" | 
					
						
							| 
									
										
										
										
											2015-09-08 15:57:48 +08:00
										 |  |  |     else | 
					
						
							|  |  |  |       "#{remote}/#{path}" | 
					
						
							| 
									
										
										
										
											2012-03-06 17:35:10 +00:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |   def github_info(f) | 
					
						
							| 
									
										
										
										
											2021-01-08 00:14:33 +09:00
										 |  |  |     return f.path if f.tap.blank? || f.tap.remote.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-08 23:54:49 +08:00
										 |  |  |     path = case f | 
					
						
							|  |  |  |     when Formula | 
					
						
							| 
									
										
										
										
											2021-01-08 00:14:33 +09:00
										 |  |  |       f.path.relative_path_from(f.tap.path) | 
					
						
							| 
									
										
										
										
											2021-10-08 23:54:49 +08:00
										 |  |  |     when Cask::Cask | 
					
						
							| 
									
										
										
										
											2023-02-24 15:44:53 +00:00
										 |  |  |       return "#{f.tap.default_remote}/blob/HEAD/Casks/#{f.token}.rb" if f.sourcefile_path.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-08 00:14:33 +09:00
										 |  |  |       f.sourcefile_path.relative_path_from(f.tap.path) | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2021-01-08 00:14:33 +09:00
										 |  |  |     github_remote_path(f.tap.remote, path) | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 23:16:30 +02:00
										 |  |  |   def info_formula(f, args:) | 
					
						
							| 
									
										
										
										
											2012-04-05 21:11:49 -05:00
										 |  |  |     specs = [] | 
					
						
							| 
									
										
										
										
											2014-03-10 14:56:02 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-07 00:13:56 +00:00
										 |  |  |     if (stable = f.stable) | 
					
						
							| 
									
										
										
										
											2021-07-14 18:20:23 -04:00
										 |  |  |       s = "stable #{stable.version}" | 
					
						
							| 
									
										
										
										
											2020-12-04 17:29:31 -06:00
										 |  |  |       s += " (bottled)" if stable.bottled? && f.pour_bottle? | 
					
						
							| 
									
										
										
										
											2014-03-10 14:56:02 -05:00
										 |  |  |       specs << s | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-05 21:11:49 -05:00
										 |  |  |     specs << "HEAD" if f.head | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-16 20:10:25 +08:00
										 |  |  |     attrs = [] | 
					
						
							|  |  |  |     attrs << "pinned at #{f.pinned_version}" if f.pinned? | 
					
						
							|  |  |  |     attrs << "keg-only" if f.keg_only? | 
					
						
							| 
									
										
										
										
											2012-04-05 21:11:49 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-12 23:44:45 -07:00
										 |  |  |     puts "#{oh1_title(f.full_name)}: #{specs * ", "}#{" [#{attrs * ", "}]" unless attrs.empty?}" | 
					
						
							| 
									
										
										
										
											2015-05-19 13:05:42 -04:00
										 |  |  |     puts f.desc if f.desc | 
					
						
							| 
									
										
										
										
											2016-08-30 21:38:13 +02:00
										 |  |  |     puts Formatter.url(f.homepage) if f.homepage | 
					
						
							| 
									
										
										
										
											2011-06-21 13:57:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-07 13:00:02 -04:00
										 |  |  |     deprecate_disable_type, deprecate_disable_reason = DeprecateDisable.deprecate_disable_info f | 
					
						
							|  |  |  |     if deprecate_disable_type.present? | 
					
						
							|  |  |  |       if deprecate_disable_reason.present? | 
					
						
							|  |  |  |         puts "#{deprecate_disable_type.capitalize} because it #{deprecate_disable_reason}!" | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         puts "#{deprecate_disable_type.capitalize}!" | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-09 17:14:09 -04:00
										 |  |  |     conflicts = f.conflicts.map do |c| | 
					
						
							| 
									
										
										
										
											2017-05-14 15:09:01 -04:00
										 |  |  |       reason = " (because #{c.reason})" if c.reason | 
					
						
							|  |  |  |       "#{c.name}#{reason}" | 
					
						
							| 
									
										
										
										
											2017-04-09 17:14:09 -04:00
										 |  |  |     end.sort! | 
					
						
							| 
									
										
										
										
											2017-05-15 10:40:07 +01:00
										 |  |  |     unless conflicts.empty? | 
					
						
							| 
									
										
										
										
											2017-10-15 02:28:32 +02:00
										 |  |  |       puts <<~EOS | 
					
						
							| 
									
										
										
										
											2017-05-15 10:40:07 +01:00
										 |  |  |         Conflicts with: | 
					
						
							| 
									
										
										
										
											2017-05-16 10:04:15 +01:00
										 |  |  |           #{conflicts.join("\n  ")} | 
					
						
							| 
									
										
										
										
											2017-05-15 10:40:07 +01:00
										 |  |  |       EOS | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 09:21:38 +00:00
										 |  |  |     kegs = f.installed_kegs | 
					
						
							|  |  |  |     heads, versioned = kegs.partition { |k| k.version.head? } | 
					
						
							|  |  |  |     kegs = [ | 
					
						
							|  |  |  |       *heads.sort_by { |k| -Tab.for_keg(k).time.to_i }, | 
					
						
							|  |  |  |       *versioned.sort_by(&:version), | 
					
						
							|  |  |  |     ] | 
					
						
							| 
									
										
										
										
											2016-08-05 22:01:32 +08:00
										 |  |  |     if kegs.empty? | 
					
						
							|  |  |  |       puts "Not installed" | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |       kegs.each do |keg| | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |         puts "#{keg} (#{keg.abv})#{" *" if keg.linked?}" | 
					
						
							| 
									
										
										
										
											2013-03-27 23:07:33 -05:00
										 |  |  |         tab = Tab.for_keg(keg).to_s | 
					
						
							|  |  |  |         puts "  #{tab}" unless tab.empty? | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:38:13 +02:00
										 |  |  |     puts "From: #{Formatter.url(github_info(f))}" | 
					
						
							| 
									
										
										
										
											2012-03-06 17:35:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-18 10:58:32 -04:00
										 |  |  |     puts "License: #{SPDX.license_expression_to_string f.license}" if f.license.present? | 
					
						
							| 
									
										
										
										
											2020-06-10 12:29:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-10 23:45:06 -05:00
										 |  |  |     unless f.deps.empty? | 
					
						
							|  |  |  |       ohai "Dependencies" | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |       %w[build required recommended optional].map do |type| | 
					
						
							| 
									
										
										
											
												cmd/info: prevent duplicate dependency display.
Before:
    $ brew info llvm
    ==> Dependencies
    Build: xz ✔, xz ✔, xz ✔, xz ✔, xz ✔, xz ✔
    $ brew info --json=v1 llvm
    ... "dependencies":["xz","xz","xz","xz","xz","xz"], ...
After
    $ brew info llvm
    ==> Dependencies
    Build: xz ✔
    $ brew info --json=v1 llvm
    ... "dependencies":["xz"], ...
Closes Homebrew/homebrew#36653.
Signed-off-by: Mike McQuaid <mike@mikemcquaid.com>
											
										 
											2015-02-08 23:20:45 +08:00
										 |  |  |         deps = f.deps.send(type).uniq | 
					
						
							| 
									
										
										
										
											2013-11-11 14:15:46 +00:00
										 |  |  |         puts "#{type.capitalize}: #{decorate_dependencies deps}" unless deps.empty? | 
					
						
							| 
									
										
										
										
											2013-05-10 23:45:06 -05:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 00:12:49 -04:00
										 |  |  |     unless f.requirements.to_a.empty? | 
					
						
							|  |  |  |       ohai "Requirements" | 
					
						
							|  |  |  |       %w[build required recommended optional].map do |type| | 
					
						
							|  |  |  |         reqs = f.requirements.select(&:"#{type}?") | 
					
						
							| 
									
										
										
										
											2016-09-18 00:37:02 -04:00
										 |  |  |         next if reqs.to_a.empty? | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 00:37:02 -04:00
										 |  |  |         puts "#{type.capitalize}: #{decorate_requirements(reqs)}" | 
					
						
							| 
									
										
										
										
											2016-09-18 00:12:49 -04:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 10:34:22 +01:00
										 |  |  |     if !f.options.empty? || f.head | 
					
						
							| 
									
										
										
										
											2012-08-04 15:40:36 -04:00
										 |  |  |       ohai "Options" | 
					
						
							| 
									
										
										
										
											2020-08-17 19:53:37 +02:00
										 |  |  |       Options.dump_for_formula f | 
					
						
							| 
									
										
										
										
											2012-08-04 15:40:36 -04:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-26 07:30:28 +02:00
										 |  |  |     caveats = Caveats.new(f) | 
					
						
							|  |  |  |     ohai "Caveats", caveats.to_s unless caveats.empty? | 
					
						
							| 
									
										
											  
											
												cmd/info: display analytics data.
When users don't have `HOMEBREW_NO_ANALYTICS` or
`HOMEBREW_NO_GITHUB_API` set let's display some analytics data in
`brew info`. This should be useful for both maintainers and for users of
Homebrew.
Note this by default combines all installs across options for a single
number; for formulae with lots of options it's a bit overwhelming to
print the installs per-option. However, for `HOMEBREW_DEVELOPER`s print
the full output.
Sample non-developer output:
```console
$ brew info wget
wget: stable 1.19.5 (bottled), HEAD
Internet file retriever
https://www.gnu.org/software/wget/
/usr/local/Cellar/wget/1.19.5 (49 files, 3.7MB) *
  Built from source on 2018-09-03 at 20:46:32
From: https://github.com/Homebrew/homebrew-core/blob/master/Formula/wget.rb
==> Dependencies
Build: pkg-config ✔
Required: libidn2 ✔, openssl ✔
Optional: pcre ✔, libmetalink ✘, gpgme ✘
==> Options
--with-debug
	Build with debug support
--with-gpgme
	Build with gpgme support
--with-libmetalink
	Build with libmetalink support
--with-pcre
	Build with pcre support
--HEAD
	Install HEAD version
==> Analytics
install: 84638 (30d), 353800 (90d), 1372775 (365d)
install_on_request: 77926 (30d), 291305 (90d), 1044898 (365d)
build_error: 11 (30d)
```
Sample developer output:
```console
$ brew info wget
wget: stable 1.19.5 (bottled), HEAD
Internet file retriever
https://www.gnu.org/software/wget/
/usr/local/Cellar/wget/1.19.5 (49 files, 3.7MB) *
  Built from source on 2018-09-03 at 20:46:32
From: https://github.com/Homebrew/homebrew-core/blob/master/Formula/wget.rb
==> Dependencies
Build: pkg-config ✔
Required: libidn2 ✔, openssl ✔
Optional: pcre ✔, libmetalink ✘, gpgme ✘
==> Options
--with-debug
	Build with debug support
--with-gpgme
	Build with gpgme support
--with-libmetalink
	Build with libmetalink support
--with-pcre
	Build with pcre support
--HEAD
	Install HEAD version
==> Analytics
==> install (30d)
wget: 84516
wget --with-debug: 51
wget --with-libressl: 16
wget --with-pcre: 14
wget --with-pcre --with-libmetalink --with-gpgme: 12
wget --with-debug --with-pcre --with-libmetalink --with-gpgme: 8
wget --HEAD: 3
wget --HEAD --with-debug --with-libmetalink --with-gpgme: 3
wget --with-gpgme: 3
wget --with-libmetalink: 3
wget --with-pcre --with-libmetalink: 3
wget --with-debug --with-pcre: 2
wget --with-libmetalink --with-gpgme: 2
wget --with-pcre --with-gpgme: 2
==> install (90d)
wget: 353131
wget --with-debug: 188
wget --with-pcre: 138
wget --with-pcre --with-libmetalink --with-gpgme: 118
wget --with-libressl: 81
wget --with-debug --with-pcre --with-libmetalink --with-gpgme: 47
wget --with-pcre --with-libmetalink: 31
wget --HEAD: 13
wget --with-pcre --with-gpgme: 12
wget --with-gpgme: 11
wget --with-debug --with-pcre: 10
wget --with-libmetalink: 8
wget --HEAD --with-pcre --with-libmetalink --with-gpgme: 4
wget --with-debug --with-pcre --with-libmetalink: 4
wget --with-libmetalink --with-gpgme: 4
==> install (365d)
wget: 1369530
wget --with-pcre: 810
wget --with-debug: 649
wget --with-pcre --with-libmetalink --with-gpgme: 554
wget --with-libressl: 479
wget --with-debug --with-pcre --with-libmetalink --with-gpgme: 235
wget --with-pcre --with-libmetalink: 184
wget --with-gpgme: 67
wget --with-pcre --with-gpgme: 67
wget --with-debug --with-pcre: 65
wget --HEAD: 54
wget --with-libmetalink: 30
wget --with-libmetalink --with-gpgme: 27
wget --with-debug --with-pcre --with-libmetalink: 24
==> install_on_request (30d)
wget: 77827
wget --with-debug: 48
wget --with-pcre: 12
wget --with-pcre --with-libmetalink --with-gpgme: 11
wget --with-debug --with-pcre --with-libmetalink --with-gpgme: 8
wget --HEAD: 3
wget --HEAD --with-debug --with-libmetalink --with-gpgme: 3
wget --with-gpgme: 3
wget --with-libmetalink: 3
wget --with-debug --with-pcre: 2
wget --with-libmetalink --with-gpgme: 2
wget --with-pcre --with-gpgme: 2
wget --with-pcre --with-libmetalink: 2
==> install_on_request (90d)
wget: 290818
wget --with-debug: 157
wget --with-pcre --with-libmetalink --with-gpgme: 101
wget --with-pcre: 100
wget --with-debug --with-pcre --with-libmetalink --with-gpgme: 42
wget --with-pcre --with-libmetalink: 30
wget --HEAD: 13
wget --with-pcre --with-gpgme: 11
wget --with-gpgme: 10
wget --with-debug --with-pcre: 8
wget --with-libmetalink: 7
wget --HEAD --with-pcre --with-libmetalink --with-gpgme: 4
wget --with-debug --with-pcre --with-libmetalink: 4
==> install_on_request (365d)
wget: 1042845
wget --with-pcre: 504
wget --with-debug: 458
wget --with-pcre --with-libmetalink --with-gpgme: 432
wget --with-debug --with-pcre --with-libmetalink --with-gpgme: 201
wget --with-pcre --with-libmetalink: 158
wget --with-gpgme: 61
wget --HEAD: 54
wget --with-pcre --with-gpgme: 49
wget --with-debug --with-pcre: 47
wget --with-debug --with-pcre --with-libmetalink: 24
wget --with-libressl: 23
wget --with-libmetalink: 22
wget --with-libmetalink --with-gpgme: 20
==> build_error (30d)
wget: 9
wget --HEAD: 1
wget --with-debug: 1
```
											
										 
											2018-09-06 14:18:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 23:16:30 +02:00
										 |  |  |     Utils::Analytics.formula_output(f, args: args) | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |   def decorate_dependencies(dependencies) | 
					
						
							| 
									
										
										
										
											2018-09-02 20:14:54 +01:00
										 |  |  |     deps_status = dependencies.map do |dep| | 
					
						
							| 
									
										
										
										
											2016-10-17 04:13:48 -04:00
										 |  |  |       if dep.satisfied?([]) | 
					
						
							|  |  |  |         pretty_installed(dep_display_s(dep)) | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         pretty_uninstalled(dep_display_s(dep)) | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-11-11 14:15:46 +00:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-10-17 04:13:48 -04:00
										 |  |  |     deps_status.join(", ") | 
					
						
							| 
									
										
										
										
											2013-11-11 14:15:46 +00:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2016-09-18 00:12:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   def decorate_requirements(requirements) | 
					
						
							| 
									
										
										
										
											2018-09-02 20:14:54 +01:00
										 |  |  |     req_status = requirements.map do |req| | 
					
						
							| 
									
										
										
										
											2016-09-18 00:37:02 -04:00
										 |  |  |       req_s = req.display_s | 
					
						
							| 
									
										
										
										
											2020-07-28 14:08:40 +02:00
										 |  |  |       req.satisfied? ? pretty_installed(req_s) : pretty_uninstalled(req_s) | 
					
						
							| 
									
										
										
										
											2016-09-18 00:12:49 -04:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-18 00:37:02 -04:00
										 |  |  |     req_status.join(", ") | 
					
						
							| 
									
										
										
										
											2016-09-18 00:12:49 -04:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2016-10-17 04:13:48 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   def dep_display_s(dep) | 
					
						
							|  |  |  |     return dep.name if dep.option_tags.empty? | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-17 04:13:48 -04:00
										 |  |  |     "#{dep.name} #{dep.option_tags.map { |o| "--#{o}" }.join(" ")}" | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   def info_cask(cask, args:) | 
					
						
							| 
									
										
										
										
											2023-03-01 00:00:54 +09:00
										 |  |  |     require "cask/info" | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-01 00:00:54 +09:00
										 |  |  |     Cask::Info.info(cask) | 
					
						
							| 
									
										
										
										
											2020-10-08 19:55:24 -04:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2010-09-11 20:22:54 +01:00
										 |  |  | end |