mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
334 lines
9.5 KiB
Ruby
334 lines
9.5 KiB
Ruby
# typed: false
|
|
# 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"
|
|
|
|
module Homebrew
|
|
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
|
|
|
|
def info_args
|
|
Homebrew::CLI::Parser.new do
|
|
usage_banner <<~EOS
|
|
`info` [<options>] [<formula>]
|
|
|
|
Display brief statistics for your Homebrew installation.
|
|
|
|
If <formula> is provided, show summary of information about <formula>.
|
|
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> in a browser. "\
|
|
"To view formula history locally: `brew log -p` <formula>"
|
|
flag "--json",
|
|
description: "Print a JSON representation of <formula>. Currently the default and only accepted "\
|
|
"value for <version> is `v1`. 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 "--all",
|
|
depends_on: "--json",
|
|
description: "Print JSON of all available formulae."
|
|
switch "-v", "--verbose",
|
|
description: "Show more verbose analytics data for <formula>."
|
|
|
|
conflicts "--installed", "--all"
|
|
end
|
|
end
|
|
|
|
def info
|
|
args = info_args.parse
|
|
|
|
if args.analytics?
|
|
if args.days.present? && !VALID_DAYS.include?(args.days)
|
|
raise UsageError, "--days must be one of #{VALID_DAYS.join(", ")}"
|
|
end
|
|
|
|
if args.category.present?
|
|
if args.named.present? && !VALID_FORMULA_CATEGORIES.include?(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
|
|
print_json(args: args)
|
|
elsif args.github?
|
|
raise FormulaOrCaskUnspecifiedError if args.no_named?
|
|
|
|
exec_browser(*args.named.to_formulae_and_casks.map { |f| github_info(f) })
|
|
else
|
|
print_info(args: args)
|
|
end
|
|
end
|
|
|
|
def print_analytics(args:)
|
|
if args.no_named?
|
|
if args.analytics?
|
|
Utils::Analytics.output(args: args)
|
|
elsif HOMEBREW_CELLAR.exist?
|
|
count = Formula.racks.length
|
|
puts "#{count} #{"keg".pluralize(count)}, #{HOMEBREW_CELLAR.dup.abv}"
|
|
end
|
|
|
|
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
|
|
|
|
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 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
|
|
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
|
|
|
|
def print_json(args:)
|
|
raise FormulaOrCaskUnspecifiedError if !(args.all? || args.installed?) && args.no_named?
|
|
|
|
json = case json_version(args.json)
|
|
when :v1, :default
|
|
formulae = if args.all?
|
|
Formula.sort
|
|
elsif args.installed?
|
|
Formula.installed.sort
|
|
else
|
|
args.named.to_formulae
|
|
end
|
|
|
|
formulae.map(&:to_hash)
|
|
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
|
|
args.named.to_formulae_to_casks
|
|
end
|
|
|
|
{
|
|
"formulae" => formulae.map(&:to_hash),
|
|
"casks" => casks.map(&:to_h),
|
|
}
|
|
else
|
|
raise
|
|
end
|
|
|
|
puts JSON.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)
|
|
if f.tap
|
|
if remote = f.tap.remote
|
|
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
|
|
github_remote_path(remote, path)
|
|
else
|
|
f.path
|
|
end
|
|
else
|
|
f.path
|
|
end
|
|
end
|
|
|
|
def info_formula(f, args:)
|
|
specs = []
|
|
|
|
if stable = f.stable
|
|
s = "stable #{stable.version}"
|
|
s += " (bottled)" if stable.bottled?
|
|
specs << s
|
|
end
|
|
|
|
specs << "HEAD" if f.head
|
|
|
|
attrs = []
|
|
attrs << "pinned at #{f.pinned_version}" if f.pinned?
|
|
attrs << "keg-only" if f.keg_only?
|
|
|
|
puts "#{f.full_name}: #{specs * ", "}#{" [#{attrs * ", "}]" unless attrs.empty?}"
|
|
puts f.desc if f.desc
|
|
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
|
|
|
|
conflicts = f.conflicts.map do |c|
|
|
reason = " (because #{c.reason})" if c.reason
|
|
"#{c.name}#{reason}"
|
|
end.sort!
|
|
unless conflicts.empty?
|
|
puts <<~EOS
|
|
Conflicts with:
|
|
#{conflicts.join("\n ")}
|
|
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
|
|
|
|
puts "From: #{Formatter.url(github_info(f))}"
|
|
|
|
puts "License: #{SPDX.license_expression_to_string f.license}" if f.license.present?
|
|
|
|
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?
|
|
end
|
|
end
|
|
|
|
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?
|
|
|
|
puts "#{type.capitalize}: #{decorate_requirements(reqs)}"
|
|
end
|
|
end
|
|
|
|
if !f.options.empty? || f.head
|
|
ohai "Options"
|
|
Options.dump_for_formula f
|
|
end
|
|
|
|
caveats = Caveats.new(f)
|
|
ohai "Caveats", caveats.to_s unless caveats.empty?
|
|
|
|
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
|
|
|
|
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/cmd/info"
|
|
|
|
Cask::Cmd::Info.info(cask)
|
|
end
|
|
end
|