brew/Library/Homebrew/cmd/outdated.rb

222 lines
7.8 KiB
Ruby
Raw Normal View History

# typed: strict
# frozen_string_literal: true
2024-04-01 09:15:53 -07:00
require "abstract_command"
require "formula"
require "keg"
2020-07-06 17:24:20 -04:00
require "cask/caskroom"
2021-08-06 02:30:44 -04:00
require "api"
module Homebrew
2024-04-01 09:15:53 -07:00
module Cmd
class Outdated < AbstractCommand
cmd_args do
description <<~EOS
List installed casks and formulae that have an updated version available. By default, version
2024-04-30 11:10:23 +02:00
information is displayed in interactive shells and suppressed otherwise.
2024-04-01 09:15:53 -07:00
EOS
switch "-q", "--quiet",
description: "List only the names of outdated kegs (takes precedence over `--verbose`)."
switch "-v", "--verbose",
description: "Include detailed version information."
switch "--formula", "--formulae",
description: "List only outdated formulae."
switch "--cask", "--casks",
description: "List only outdated casks."
flag "--json",
description: "Print output in JSON format. There are two versions: `v1` and `v2`. " \
"`v1` is deprecated and is currently the default if no version is specified. " \
"`v2` prints outdated formulae and casks."
switch "--fetch-HEAD",
description: "Fetch the upstream repository to detect if the HEAD installation of the " \
"formula is outdated. Otherwise, the repository's HEAD will only be checked for " \
"updates when a new stable or development version has been released."
switch "-g", "--greedy",
description: "Also include outdated casks with `auto_updates true` or `version :latest`.",
env: :upgrade_greedy
2024-04-01 09:15:53 -07:00
switch "--greedy-latest",
description: "Also include outdated casks including those with `version :latest`."
switch "--greedy-auto-updates",
description: "Also include outdated casks including those with `auto_updates true`."
conflicts "--quiet", "--verbose", "--json"
conflicts "--formula", "--cask"
named_args [:formula, :cask]
2020-07-17 08:53:19 -04:00
end
2024-04-01 09:15:53 -07:00
sig { override.void }
def run
case json_version(args.json)
when :v1
odie "`brew outdated --json=v1` is no longer supported. Use brew outdated --json=v2 instead."
when :v2, :default
formulae, casks = if args.formula?
[outdated_formulae, []]
elsif args.cask?
[[], outdated_casks]
else
2024-04-01 09:15:53 -07:00
outdated_formulae_casks
end
2024-04-01 09:15:53 -07:00
json = {
2024-11-07 09:11:00 -08:00
formulae: json_info(formulae),
casks: json_info(casks),
2024-04-01 09:15:53 -07:00
}
2024-11-07 11:16:47 -08:00
# json v2.8.1 is inconsistent it how it renders empty arrays,
# so we use `[]` for consistency:
puts JSON.pretty_generate(json).gsub(/\[\n\n\s*\]/, "[]")
2024-04-01 09:15:53 -07:00
outdated = formulae + casks
else
2024-04-01 09:15:53 -07:00
outdated = if args.formula?
outdated_formulae
elsif args.cask?
outdated_casks
else
outdated_formulae_casks.flatten
end
print_outdated(outdated)
end
2020-07-17 08:53:19 -04:00
2024-04-01 09:15:53 -07:00
Homebrew.failed = args.named.present? && outdated.present?
end
2024-04-01 09:15:53 -07:00
private
sig { params(formulae_or_casks: T::Array[T.any(Formula, Cask::Cask)]).void }
2024-04-01 09:15:53 -07:00
def print_outdated(formulae_or_casks)
formulae_or_casks.each do |formula_or_cask|
if formula_or_cask.is_a?(Formula)
f = formula_or_cask
if verbose?
outdated_kegs = f.outdated_kegs(fetch_head: args.fetch_HEAD?)
current_version = if f.alias_changed? && !f.latest_formula.latest_version_installed?
latest = f.latest_formula
"#{latest.name} (#{latest.pkg_version})"
elsif f.head? && outdated_kegs.any? { |k| k.version.to_s == f.pkg_version.to_s }
# There is a newer HEAD but the version number has not changed.
"latest HEAD"
else
f.pkg_version.to_s
end
outdated_versions = outdated_kegs.group_by { |keg| Formulary.from_keg(keg).full_name }
.sort_by { |full_name, _kegs| full_name }
.map do |full_name, kegs|
"#{full_name} (#{kegs.map(&:version).join(", ")})"
end.join(", ")
pinned_version = " [pinned at #{f.pinned_version}]" if f.pinned?
puts "#{outdated_versions} < #{current_version}#{pinned_version}"
else
puts f.full_installed_specified_name
end
else
c = formula_or_cask
2020-07-17 08:53:19 -04:00
2024-07-10 22:52:14 -07:00
puts c.outdated_info(args.greedy?, verbose?, false, args.greedy_latest?, args.greedy_auto_updates?)
2024-04-01 09:15:53 -07:00
end
end
2024-04-01 09:15:53 -07:00
end
sig {
params(
formulae_or_casks: T::Array[T.any(Formula, Cask::Cask)],
).returns(
T::Array[T.any(T::Hash[String, T.untyped], T::Hash[String, T.untyped])],
)
}
2024-04-01 09:15:53 -07:00
def json_info(formulae_or_casks)
formulae_or_casks.map do |formula_or_cask|
if formula_or_cask.is_a?(Formula)
f = formula_or_cask
outdated_versions = f.outdated_kegs(fetch_head: args.fetch_HEAD?).map(&:version)
current_version = if f.head? && outdated_versions.any? { |v| v.to_s == f.pkg_version.to_s }
"HEAD"
else
f.pkg_version.to_s
end
{ name: f.full_name,
installed_versions: outdated_versions.map(&:to_s),
current_version:,
pinned: f.pinned?,
pinned_version: f.pinned_version }
else
c = formula_or_cask
2020-07-17 08:53:19 -04:00
2024-07-10 22:52:14 -07:00
c.outdated_info(args.greedy?, verbose?, true, args.greedy_latest?, args.greedy_auto_updates?)
2024-04-01 09:15:53 -07:00
end
end
end
sig { returns(T::Boolean) }
2024-04-01 09:15:53 -07:00
def verbose?
($stdout.tty? || Context.current.verbose?) && !Context.current.quiet?
end
2020-07-06 17:24:20 -04:00
sig { params(version: T.nilable(T.any(TrueClass, String))).returns(T.nilable(Symbol)) }
2024-04-01 09:15:53 -07:00
def json_version(version)
version_hash = {
nil => nil,
true => :default,
"v1" => :v1,
"v2" => :v2,
}
2024-11-07 09:11:00 -08:00
version_hash.fetch(version) { raise UsageError, "invalid JSON version: #{version}" }
2024-04-01 09:15:53 -07:00
end
2020-07-06 17:24:20 -04:00
sig { returns(T::Array[Formula]) }
2024-04-01 09:15:53 -07:00
def outdated_formulae
T.cast(
2025-05-05 14:35:08 -07:00
select_outdated(args.named.to_resolved_formulae.presence || Formula.installed).sort,
T::Array[Formula],
)
2024-04-01 09:15:53 -07:00
end
2020-07-17 08:53:19 -04:00
sig { returns(T::Array[Cask::Cask]) }
2024-04-01 09:15:53 -07:00
def outdated_casks
outdated = if args.named.present?
2024-04-01 09:15:53 -07:00
select_outdated(args.named.to_casks)
else
select_outdated(Cask::Caskroom.casks)
end
T.cast(outdated, T::Array[Cask::Cask])
2024-04-01 09:15:53 -07:00
end
2020-07-17 08:53:19 -04:00
sig { returns([T::Array[T.any(Formula, Cask::Cask)], T::Array[T.any(Formula, Cask::Cask)]]) }
2024-04-01 09:15:53 -07:00
def outdated_formulae_casks
formulae, casks = args.named.to_resolved_formulae_to_casks
2020-07-17 08:53:19 -04:00
2024-04-01 09:15:53 -07:00
if formulae.blank? && casks.blank?
formulae = Formula.installed
casks = Cask::Caskroom.casks
end
2020-07-17 08:53:19 -04:00
2024-04-01 09:15:53 -07:00
[select_outdated(formulae).sort, select_outdated(casks)]
end
2020-07-17 08:53:19 -04:00
sig {
params(formulae_or_casks: T::Array[T.any(Formula, Cask::Cask)]).returns(T::Array[T.any(Formula, Cask::Cask)])
}
2024-04-01 09:15:53 -07:00
def select_outdated(formulae_or_casks)
formulae_or_casks.select do |formula_or_cask|
if formula_or_cask.is_a?(Formula)
formula_or_cask.outdated?(fetch_head: args.fetch_HEAD?)
else
2024-07-10 22:52:14 -07:00
formula_or_cask.outdated?(greedy: args.greedy?, greedy_latest: args.greedy_latest?,
2024-04-01 09:15:53 -07:00
greedy_auto_updates: args.greedy_auto_updates?)
end
end
end
2020-07-17 08:53:19 -04:00
end
2020-07-06 17:24:20 -04:00
end
end