mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
383 lines
12 KiB
Ruby
383 lines
12 KiB
Ruby
# typed: true
|
|
# frozen_string_literal: true
|
|
|
|
require "missing_formula"
|
|
require "caveats"
|
|
require "cli/parser"
|
|
require "options"
|
|
require "formula"
|
|
require "keg"
|
|
require "tab"
|
|
require "json"
|
|
require "utils/spdx"
|
|
require "deprecate_disable"
|
|
require "api"
|
|
|
|
module Homebrew
|
|
extend T::Sig
|
|
|
|
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
|
|
|
|
sig { returns(CLI::Parser) }
|
|
def info_args
|
|
Homebrew::CLI::Parser.new do
|
|
description <<~EOS
|
|
Display brief statistics for your Homebrew installation.
|
|
|
|
If a <formula> or <cask> is provided, show summary of information about it.
|
|
EOS
|
|
switch "--analytics",
|
|
description: "List global Homebrew analytics data or, if specified, installation and " \
|
|
"build error data for <formula> (provided neither `HOMEBREW_NO_ANALYTICS` " \
|
|
"nor `HOMEBREW_NO_GITHUB_API` are set)."
|
|
flag "--days=",
|
|
depends_on: "--analytics",
|
|
description: "How many days of analytics data to retrieve. " \
|
|
"The value for <days> must be `30`, `90` or `365`. The default is `30`."
|
|
flag "--category=",
|
|
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`."
|
|
switch "--github",
|
|
description: "Open the GitHub source page for <formula> and <cask> in a browser. " \
|
|
"To view the history locally: `brew log -p` <formula> or <cask>"
|
|
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 "--installed",
|
|
depends_on: "--json",
|
|
description: "Print JSON of formulae that are currently installed."
|
|
switch "--eval-all",
|
|
depends_on: "--json",
|
|
description: "Evaluate all available formulae and casks, whether installed or not, to print their " \
|
|
"JSON. Implied if `HOMEBREW_EVAL_ALL` is set."
|
|
switch "--all",
|
|
hidden: true,
|
|
depends_on: "--json"
|
|
switch "--variations",
|
|
depends_on: "--json",
|
|
description: "Include the variations hash in each formula's JSON output."
|
|
switch "-v", "--verbose",
|
|
description: "Show more verbose analytics data for <formula>."
|
|
switch "--formula", "--formulae",
|
|
description: "Treat all named arguments as formulae."
|
|
switch "--cask", "--casks",
|
|
description: "Treat all named arguments as casks."
|
|
|
|
conflicts "--installed", "--eval-all"
|
|
conflicts "--installed", "--all"
|
|
conflicts "--formula", "--cask"
|
|
|
|
named_args [:formula, :cask]
|
|
end
|
|
end
|
|
|
|
sig { void }
|
|
def info
|
|
args = info_args.parse
|
|
|
|
if args.analytics?
|
|
if args.days.present? && VALID_DAYS.exclude?(args.days)
|
|
raise UsageError, "`--days` must be one of #{VALID_DAYS.join(", ")}."
|
|
end
|
|
|
|
if args.category.present?
|
|
if args.named.present? && VALID_FORMULA_CATEGORIES.exclude?(args.category)
|
|
raise UsageError,
|
|
"`--category` must be one of #{VALID_FORMULA_CATEGORIES.join(", ")} when querying formulae."
|
|
end
|
|
|
|
unless VALID_CATEGORIES.include?(args.category)
|
|
raise UsageError, "`--category` must be one of #{VALID_CATEGORIES.join(", ")}."
|
|
end
|
|
end
|
|
|
|
print_analytics(args: args)
|
|
elsif args.json
|
|
all = args.eval_all?
|
|
if !all && args.all? && !Homebrew::EnvConfig.eval_all?
|
|
odisabled "brew info --all", "brew info --eval-all or HOMEBREW_EVAL_ALL"
|
|
all = true
|
|
end
|
|
|
|
print_json(all, args: args)
|
|
elsif args.github?
|
|
raise FormulaOrCaskUnspecifiedError if args.no_named?
|
|
|
|
exec_browser(*args.named.to_formulae_and_casks.map { |f| github_info(f) })
|
|
elsif args.no_named?
|
|
print_statistics
|
|
else
|
|
print_info(args: args)
|
|
end
|
|
end
|
|
|
|
sig { void }
|
|
def print_statistics
|
|
return unless HOMEBREW_CELLAR.exist?
|
|
|
|
count = Formula.racks.length
|
|
puts "#{Utils.pluralize("keg", count, include_count: true)}, #{HOMEBREW_CELLAR.dup.abv}"
|
|
end
|
|
|
|
sig { params(args: CLI::Args).void }
|
|
def print_analytics(args:)
|
|
if args.no_named?
|
|
Utils::Analytics.output(args: args)
|
|
return
|
|
end
|
|
|
|
args.named.to_formulae_and_casks_and_unavailable.each_with_index do |obj, i|
|
|
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
|
|
|
|
sig { params(args: CLI::Args).void }
|
|
def print_info(args:)
|
|
args.named.to_formulae_and_casks_and_unavailable.each_with_index do |obj, i|
|
|
puts unless i.zero?
|
|
|
|
case obj
|
|
when Formula
|
|
info_formula(obj, args: args)
|
|
when Cask::Cask
|
|
info_cask(obj, args: args)
|
|
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
|
|
when FormulaOrCaskUnavailableError
|
|
# The formula/cask could not be found
|
|
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
|
|
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
|
|
|
|
sig { params(all: T::Boolean, args: T.untyped).void }
|
|
def print_json(all, args:)
|
|
raise FormulaOrCaskUnspecifiedError if !(all || args.installed?) && args.no_named?
|
|
|
|
json = case json_version(args.json)
|
|
when :v1, :default
|
|
raise UsageError, "Cannot specify `--cask` when using `--json=v1`!" if args.cask?
|
|
|
|
formulae = if all
|
|
Formula.all.sort
|
|
elsif args.installed?
|
|
Formula.installed.sort
|
|
else
|
|
args.named.to_formulae
|
|
end
|
|
|
|
if args.variations?
|
|
formulae.map(&:to_hash_with_variations)
|
|
else
|
|
formulae.map(&:to_hash)
|
|
end
|
|
when :v2
|
|
formulae, casks = if all
|
|
[Formula.all.sort, Cask::Cask.all.sort_by(&:full_name)]
|
|
elsif args.installed?
|
|
[Formula.installed.sort, Cask::Caskroom.casks.sort_by(&:full_name)]
|
|
else
|
|
args.named.to_formulae_to_casks
|
|
end
|
|
|
|
if args.variations?
|
|
{
|
|
"formulae" => formulae.map(&:to_hash_with_variations),
|
|
"casks" => casks.map(&:to_hash_with_variations),
|
|
}
|
|
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(formula)
|
|
return formula.path if formula.tap.blank? || formula.tap.remote.blank?
|
|
|
|
path = case formula
|
|
when Formula
|
|
formula.path.relative_path_from(formula.tap.path)
|
|
when Cask::Cask
|
|
return "#{formula.tap.default_remote}/blob/HEAD/Casks/#{formula.token}.rb" if formula.sourcefile_path.blank?
|
|
|
|
formula.sourcefile_path.relative_path_from(formula.tap.path)
|
|
end
|
|
github_remote_path(formula.tap.remote, path)
|
|
end
|
|
|
|
def info_formula(formula, args:)
|
|
specs = []
|
|
|
|
if (stable = formula.stable)
|
|
string = "stable #{stable.version}"
|
|
string += " (bottled)" if stable.bottled? && formula.pour_bottle?
|
|
specs << string
|
|
end
|
|
|
|
specs << "HEAD" if formula.head
|
|
|
|
attrs = []
|
|
attrs << "pinned at #{formula.pinned_version}" if formula.pinned?
|
|
attrs << "keg-only" if formula.keg_only?
|
|
|
|
puts "#{oh1_title(formula.full_name)}: #{specs * ", "}#{" [#{attrs * ", "}]" unless attrs.empty?}"
|
|
puts formula.desc if formula.desc
|
|
puts Formatter.url(formula.homepage) if formula.homepage
|
|
|
|
deprecate_disable_type, deprecate_disable_reason = DeprecateDisable.deprecate_disable_info formula
|
|
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
|
|
|
|
conflicts = formula.conflicts.map do |conflict|
|
|
reason = " (because #{conflict.reason})" if conflict.reason
|
|
"#{conflict.name}#{reason}"
|
|
end.sort!
|
|
unless conflicts.empty?
|
|
puts <<~EOS
|
|
Conflicts with:
|
|
#{conflicts.join("\n ")}
|
|
EOS
|
|
end
|
|
|
|
kegs = formula.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
|
|
|
|
puts "From: #{Formatter.url(github_info(formula))}"
|
|
|
|
puts "License: #{SPDX.license_expression_to_string formula.license}" if formula.license.present?
|
|
|
|
unless formula.deps.empty?
|
|
ohai "Dependencies"
|
|
%w[build required recommended optional].map do |type|
|
|
deps = formula.deps.send(type).uniq
|
|
puts "#{type.capitalize}: #{decorate_dependencies deps}" unless deps.empty?
|
|
end
|
|
end
|
|
|
|
unless formula.requirements.to_a.empty?
|
|
ohai "Requirements"
|
|
%w[build required recommended optional].map do |type|
|
|
reqs = formula.requirements.select(&:"#{type}?")
|
|
next if reqs.to_a.empty?
|
|
|
|
puts "#{type.capitalize}: #{decorate_requirements(reqs)}"
|
|
end
|
|
end
|
|
|
|
if !formula.options.empty? || formula.head
|
|
ohai "Options"
|
|
Options.dump_for_formula formula
|
|
end
|
|
|
|
caveats = Caveats.new(formula)
|
|
ohai "Caveats", caveats.to_s unless caveats.empty?
|
|
|
|
Utils::Analytics.formula_output(formula, 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
|
|
|
|
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)
|
|
end
|
|
req_status.join(", ")
|
|
end
|
|
|
|
def dep_display_s(dep)
|
|
return dep.name if dep.option_tags.empty?
|
|
|
|
"#{dep.name} #{dep.option_tags.map { |o| "--#{o}" }.join(" ")}"
|
|
end
|
|
|
|
def info_cask(cask, args:)
|
|
require "cask/info"
|
|
|
|
Cask::Info.info(cask)
|
|
end
|
|
end
|