373 lines
11 KiB
Ruby
Raw Normal View History

2020-10-10 14:16:11 +02:00
# typed: false
# frozen_string_literal: true
require "missing_formula"
2014-06-19 17:52:42 -05:00
require "caveats"
2019-04-17 18:25:08 +09:00
require "cli/parser"
require "options"
2014-06-19 17:52:42 -05:00
require "formula"
require "keg"
require "tab"
require "json"
2020-08-18 10:58:32 -04:00
require "utils/spdx"
require "deprecate_disable"
2021-08-06 02:30:44 -04:00
require "api"
module Homebrew
2020-10-20 12:03:48 +02:00
extend T::Sig
2016-09-26 01:44:51 +02:00
module_function
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
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",
description: "List global Homebrew analytics data or, if specified, installation and "\
2019-04-30 08:44:35 +01:00
"build error data for <formula> (provided neither `HOMEBREW_NO_ANALYTICS` "\
"nor `HOMEBREW_NO_GITHUB_API` are set)."
flag "--days=",
2019-04-30 08:44:35 +01:00
depends_on: "--analytics",
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`."
flag "--category=",
2019-04-30 08:44:35 +01:00
depends_on: "--analytics",
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. "\
"The default is `install`."
2018-11-07 19:31:20 +05:30
switch "--github",
description: "Open the GitHub source page for <formula> in a browser. "\
2019-04-30 08:44:35 +01:00
"To view formula history locally: `brew log -p` <formula>"
2019-08-06 13:23:19 -04:00
flag "--json",
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 "\
"JSON output: <https://docs.brew.sh/Querying-Brew>"
switch "--bottle",
depends_on: "--json",
2021-06-08 22:02:32 -04:00
description: "Output information about the bottles for <formula> and its dependencies.",
hidden: true
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."
2019-03-11 12:59:39 -04:00
switch "--all",
2019-04-30 08:44:35 +01:00
depends_on: "--json",
description: "Print JSON of all available formulae."
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."
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
%w[--cask --analytics --github].each do |conflict|
conflicts "--bottle", conflict
end
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 }
def info
2020-07-30 18:40:10 +02:00
args = info_args.parse
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)
2020-10-08 19:55:24 -04:00
raise UsageError, "--days must be one of #{VALID_DAYS.join(", ")}"
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)
2020-10-08 19:55:24 -04:00
raise UsageError, "--category must be one of #{VALID_FORMULA_CATEGORIES.join(", ")} when querying formulae"
end
2020-10-08 19:55:24 -04:00
unless VALID_CATEGORIES.include?(args.category)
raise UsageError, "--category must be one of #{VALID_CATEGORIES.join(", ")}"
end
end
2020-12-17 21:14:18 +09:00
print_analytics(args: args)
2020-10-08 19:55:24 -04:00
elsif args.json
2020-12-17 21:14:18 +09:00
print_json(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
else
2020-12-17 21:14:18 +09:00
print_info(args: args)
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
puts "#{count} #{"keg".pluralize(count)}, #{HOMEBREW_CELLAR.dup.abv}"
end
2020-12-17 21:14:18 +09:00
sig { params(args: CLI::Args).void }
def print_analytics(args:)
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)
when FormulaOrCaskUnavailableError
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
end
2020-10-08 19:55:24 -04:00
else
raise
end
end
end
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
2020-12-17 21:14:18 +09:00
sig { params(args: CLI::Args).void }
def print_json(args:)
raise FormulaOrCaskUnspecifiedError if !(args.all? || args.installed?) && args.no_named?
json = case json_version(args.json)
when :v1, :default
raise UsageError, "cannot specify --cask with --json=v1!" if args.cask?
formulae = if args.all?
Formula.sort
elsif args.installed?
Formula.installed.sort
else
args.named.to_formulae
end
if args.bottle?
formulae.map(&:to_recursive_bottle_hash)
else
formulae.map(&:to_hash)
end
when :v2
formulae, casks = if args.all?
[Formula.sort, Cask::Cask.to_a.sort_by(&:full_name)]
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
end
if args.bottle?
{ "formulae" => formulae.map(&:to_recursive_bottle_hash) }
else
{
"formulae" => formulae.map(&:to_hash),
"casks" => casks.map(&:to_h),
}
end
else
raise
end
puts JSON.pretty_generate(json)
end
def github_remote_path(remote, path)
if remote =~ %r{^(?:https?://|git(?:@|://))github\.com[:/](.+)/(.+?)(?:\.git)?$}
"https://github.com/#{Regexp.last_match(1)}/#{Regexp.last_match(2)}/blob/HEAD/#{path}"
else
"#{remote}/#{path}"
end
end
def github_info(f)
2021-01-08 00:14:33 +09:00
return f.path if f.tap.blank? || f.tap.remote.blank?
path = if f.class.superclass == Formula
f.path.relative_path_from(f.tap.path)
elsif f.is_a?(Cask::Cask)
f.sourcefile_path.relative_path_from(f.tap.path)
end
2021-01-08 00:14:33 +09:00
github_remote_path(f.tap.remote, path)
end
def info_formula(f, args:)
2012-04-05 21:11:49 -05:00
specs = []
if ENV["HOMEBREW_INSTALL_FROM_API"].present? && Homebrew::API::Bottle.available?(f.name)
2021-08-06 02:30:44 -04:00
info = Homebrew::API::Bottle.fetch(f.name)
2021-07-14 18:16:23 -04:00
latest_version = info["pkg_version"].split("_").first
bottle_exists = info["bottles"].key?(Utils::Bottles.tag.to_s) || info["bottles"].key?("all")
s = "stable #{latest_version}"
s += " (bottled)" if bottle_exists
specs << s
elsif (stable = f.stable)
2021-07-14 18:20:23 -04:00
s = "stable #{stable.version}"
s += " (bottled)" if stable.bottled? && f.pour_bottle?
specs << s
end
2012-04-05 21:11:49 -05:00
specs << "HEAD" if f.head
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
puts "#{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
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
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),
]
if kegs.empty?
puts "Not installed"
else
kegs.each do |keg|
puts "#{keg} (#{keg.abv})#{" *" if keg.linked?}"
tab = Tab.for_keg(keg).to_s
puts " #{tab}" unless tab.empty?
end
end
2016-08-30 21:38:13 +02:00
puts "From: #{Formatter.url(github_info(f))}"
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"
%w[build required recommended optional].map do |type|
deps = f.deps.send(type).uniq
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}?")
next if reqs.to_a.empty?
2018-09-17 02:45:00 +02:00
puts "#{type.capitalize}: #{decorate_requirements(reqs)}"
2016-09-18 00:12:49 -04:00
end
end
if !f.options.empty? || f.head
ohai "Options"
2020-08-17 19:53:37 +02:00
Options.dump_for_formula f
end
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
Utils::Analytics.formula_output(f, args: args)
end
def decorate_dependencies(dependencies)
deps_status = dependencies.map do |dep|
if dep.satisfied?([])
pretty_installed(dep_display_s(dep))
else
pretty_uninstalled(dep_display_s(dep))
end
end
deps_status.join(", ")
end
2016-09-18 00:12:49 -04:00
def decorate_requirements(requirements)
req_status = requirements.map do |req|
req_s = req.display_s
req.satisfied? ? pretty_installed(req_s) : pretty_uninstalled(req_s)
2016-09-18 00:12:49 -04:00
end
req_status.join(", ")
2016-09-18 00:12:49 -04:00
end
def dep_display_s(dep)
return dep.name if dep.option_tags.empty?
2018-09-17 02:45:00 +02:00
"#{dep.name} #{dep.option_tags.map { |o| "--#{o}" }.join(" ")}"
end
2020-10-08 19:55:24 -04:00
def info_cask(cask, args:)
require "cask/cmd"
2020-10-08 19:55:24 -04:00
require "cask/cmd/info"
Cask::Cmd::Info.info(cask)
end
end