mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
287 lines
9.4 KiB
Ruby
287 lines
9.4 KiB
Ruby
# typed: true
|
|
# frozen_string_literal: true
|
|
|
|
require "cli/parser"
|
|
require "livecheck/livecheck"
|
|
|
|
module Homebrew
|
|
extend T::Sig
|
|
|
|
module_function
|
|
|
|
sig { returns(CLI::Parser) }
|
|
def bump_args
|
|
Homebrew::CLI::Parser.new do
|
|
description <<~EOS
|
|
Display out-of-date brew formulae and the latest version available. If the
|
|
returned current and livecheck versions differ or when querying specific
|
|
formulae, also displays whether a pull request has been opened with the URL.
|
|
EOS
|
|
switch "--full-name",
|
|
description: "Print formulae/casks with fully-qualified names."
|
|
switch "--no-pull-requests",
|
|
description: "Do not retrieve pull requests from GitHub."
|
|
switch "--formula", "--formulae",
|
|
description: "Check only formulae."
|
|
switch "--cask", "--casks",
|
|
description: "Check only casks."
|
|
switch "--open-pr",
|
|
description: "Open a pull request for the new version if there are none already open."
|
|
flag "--limit=",
|
|
description: "Limit number of package results returned."
|
|
flag "--start-with=",
|
|
description: "Letter or word that the list of package results should alphabetically follow."
|
|
|
|
conflicts "--cask", "--formula"
|
|
conflicts "--no-pull-requests", "--open-pr"
|
|
|
|
named_args [:formula, :cask]
|
|
end
|
|
end
|
|
|
|
def bump
|
|
args = bump_args.parse
|
|
|
|
if args.limit.present? && !args.formula? && !args.cask?
|
|
raise UsageError, "`--limit` must be used with either `--formula` or `--cask`."
|
|
end
|
|
|
|
formulae_and_casks = if args.formula?
|
|
args.named.to_formulae
|
|
elsif args.cask?
|
|
args.named.to_casks
|
|
else
|
|
args.named.to_formulae_and_casks
|
|
end
|
|
formulae_and_casks = formulae_and_casks&.sort_by do |formula_or_cask|
|
|
formula_or_cask.respond_to?(:token) ? formula_or_cask.token : formula_or_cask.name
|
|
end
|
|
|
|
limit = args.limit.to_i if args.limit.present?
|
|
|
|
unless Utils::Curl.curl_supports_tls13?
|
|
begin
|
|
ensure_formula_installed!("curl", reason: "Repology queries") unless HOMEBREW_BREWED_CURL_PATH.exist?
|
|
rescue FormulaUnavailableError
|
|
opoo "A newer `curl` is required for Repology queries."
|
|
end
|
|
end
|
|
|
|
if formulae_and_casks.present?
|
|
Livecheck.load_other_tap_strategies(formulae_and_casks)
|
|
|
|
ambiguous_casks = []
|
|
if !args.formula? && !args.cask?
|
|
ambiguous_casks = formulae_and_casks.group_by { |item| Livecheck.formula_or_cask_name(item, full_name: true) }
|
|
.values
|
|
.select { |items| items.length > 1 }
|
|
.flatten
|
|
.select { |item| item.is_a?(Cask::Cask) }
|
|
end
|
|
|
|
ambiguous_names = []
|
|
unless args.full_name?
|
|
ambiguous_names =
|
|
(formulae_and_casks - ambiguous_casks).group_by { |item| Livecheck.formula_or_cask_name(item) }
|
|
.values
|
|
.select { |items| items.length > 1 }
|
|
.flatten
|
|
end
|
|
|
|
formulae_and_casks.each_with_index do |formula_or_cask, i|
|
|
puts if i.positive?
|
|
|
|
use_full_name = args.full_name? || ambiguous_names.include?(formula_or_cask)
|
|
name = Livecheck.formula_or_cask_name(formula_or_cask, full_name: use_full_name)
|
|
repository = if formula_or_cask.is_a?(Formula)
|
|
if formula_or_cask.head_only?
|
|
ohai name
|
|
puts "Formula is HEAD-only."
|
|
next
|
|
end
|
|
|
|
Repology::HOMEBREW_CORE
|
|
else
|
|
Repology::HOMEBREW_CASK
|
|
end
|
|
|
|
package_data = if formula_or_cask.is_a?(Formula) && formula_or_cask.versioned_formula?
|
|
nil
|
|
else
|
|
Repology.single_package_query(name, repository: repository)
|
|
end
|
|
|
|
retrieve_and_display_info_and_open_pr(
|
|
formula_or_cask,
|
|
name,
|
|
package_data&.values&.first,
|
|
args: args,
|
|
ambiguous_cask: ambiguous_casks.include?(formula_or_cask),
|
|
)
|
|
end
|
|
else
|
|
api_response = {}
|
|
unless args.cask?
|
|
api_response[:formulae] =
|
|
Repology.parse_api_response(limit, args.start_with, repository: Repology::HOMEBREW_CORE)
|
|
end
|
|
unless args.formula?
|
|
api_response[:casks] =
|
|
Repology.parse_api_response(limit, args.start_with, repository: Repology::HOMEBREW_CASK)
|
|
end
|
|
|
|
api_response.each_with_index do |(package_type, outdated_packages), idx|
|
|
repository = if package_type == :formulae
|
|
Repology::HOMEBREW_CORE
|
|
else
|
|
Repology::HOMEBREW_CASK
|
|
end
|
|
puts if idx.positive?
|
|
oh1 package_type.capitalize if api_response.size > 1
|
|
|
|
outdated_packages.each_with_index do |(_name, repositories), i|
|
|
break if limit && i >= limit
|
|
|
|
homebrew_repo = repositories.find do |repo|
|
|
repo["repo"] == repository
|
|
end
|
|
|
|
next if homebrew_repo.blank?
|
|
|
|
formula_or_cask = begin
|
|
if repository == Repology::HOMEBREW_CORE
|
|
Formula[homebrew_repo["srcname"]]
|
|
else
|
|
Cask::CaskLoader.load(homebrew_repo["srcname"])
|
|
end
|
|
rescue
|
|
next
|
|
end
|
|
name = Livecheck.formula_or_cask_name(formula_or_cask)
|
|
ambiguous_cask = begin
|
|
formula_or_cask.is_a?(Cask::Cask) && !args.cask? && Formula[name]
|
|
rescue FormulaUnavailableError
|
|
false
|
|
end
|
|
|
|
puts if i.positive?
|
|
retrieve_and_display_info_and_open_pr(
|
|
formula_or_cask,
|
|
name,
|
|
repositories,
|
|
args: args,
|
|
ambiguous_cask: ambiguous_cask,
|
|
)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
def livecheck_result(formula_or_cask)
|
|
name = Livecheck.formula_or_cask_name(formula_or_cask)
|
|
|
|
referenced_formula_or_cask, =
|
|
Livecheck.resolve_livecheck_reference(formula_or_cask, full_name: false, debug: false)
|
|
|
|
# Check skip conditions for a referenced formula/cask
|
|
if referenced_formula_or_cask
|
|
skip_info = Livecheck::SkipConditions.referenced_skip_information(
|
|
referenced_formula_or_cask,
|
|
name,
|
|
full_name: false,
|
|
verbose: false,
|
|
)
|
|
end
|
|
|
|
skip_info ||= Livecheck::SkipConditions.skip_information(formula_or_cask, full_name: false, verbose: false)
|
|
if skip_info.present?
|
|
return "#{skip_info[:status]}#{" - #{skip_info[:messages].join(", ")}" if skip_info[:messages].present?}"
|
|
end
|
|
|
|
version_info = Livecheck.latest_version(
|
|
formula_or_cask,
|
|
referenced_formula_or_cask: referenced_formula_or_cask,
|
|
json: true, full_name: false, verbose: true, debug: false
|
|
)
|
|
return "unable to get versions" if version_info.blank?
|
|
|
|
latest = version_info[:latest]
|
|
strategy = version_info[:meta][:strategy]
|
|
|
|
[Version.new(latest), strategy]
|
|
rescue => e
|
|
"error: #{e}"
|
|
end
|
|
|
|
def retrieve_pull_requests(formula_or_cask, name)
|
|
tap_remote_repo = formula_or_cask.tap&.remote_repo || formula_or_cask.tap&.full_name
|
|
pull_requests = GitHub.fetch_pull_requests(name, tap_remote_repo, state: "open")
|
|
if pull_requests.try(:any?)
|
|
pull_requests = pull_requests.map { |pr| "#{pr["title"]} (#{Formatter.url(pr["html_url"])})" }.join(", ")
|
|
end
|
|
|
|
pull_requests
|
|
end
|
|
|
|
def retrieve_and_display_info_and_open_pr(formula_or_cask, name, repositories, args:, ambiguous_cask: false)
|
|
if formula_or_cask.is_a?(Formula)
|
|
current_version = formula_or_cask.stable.version
|
|
type = :formula
|
|
version_name = "formula version"
|
|
else
|
|
current_version = Version.new(formula_or_cask.version)
|
|
type = :cask
|
|
version_name = "cask version "
|
|
end
|
|
|
|
livecheck_latest, livecheck_strategy = livecheck_result(formula_or_cask)
|
|
|
|
repology_latest = if repositories.present?
|
|
Repology.latest_version(repositories)
|
|
else
|
|
"not found"
|
|
end
|
|
|
|
new_version = if livecheck_latest.is_a?(Version) && livecheck_latest > current_version
|
|
livecheck_latest
|
|
elsif repology_latest.is_a?(Version) && repology_latest > current_version && livecheck_strategy != "GithubLatest"
|
|
repology_latest
|
|
end.presence
|
|
|
|
pull_requests = if !args.no_pull_requests? && (args.named.present? || new_version)
|
|
retrieve_pull_requests(formula_or_cask, name)
|
|
end.presence
|
|
|
|
title_name = ambiguous_cask ? "#{name} (cask)" : name
|
|
title = if current_version == repology_latest &&
|
|
current_version == livecheck_latest
|
|
"#{title_name} #{Tty.green}is up to date!#{Tty.reset}"
|
|
else
|
|
title_name
|
|
end
|
|
|
|
ohai title
|
|
puts <<~EOS
|
|
Current #{version_name}: #{current_version}
|
|
Latest livecheck version: #{livecheck_latest}
|
|
Latest Repology version: #{repology_latest}
|
|
Open pull requests: #{pull_requests || "none"}
|
|
EOS
|
|
|
|
return unless args.open_pr?
|
|
|
|
if repology_latest > current_version &&
|
|
repology_latest > livecheck_latest &&
|
|
livecheck_strategy == "GithubLatest"
|
|
puts "#{title_name} was not bumped to the Repology version because that " \
|
|
"version is not the latest release on GitHub."
|
|
end
|
|
|
|
return unless new_version
|
|
return if pull_requests
|
|
|
|
system HOMEBREW_BREW_FILE, "bump-#{type}-pr", "--no-browse",
|
|
"--message=Created by `brew bump`", "--version=#{new_version}", name
|
|
end
|
|
end
|