bump: bump synced formula together

This commit is contained in:
Bevan Kay 2025-04-09 22:55:56 +10:00
parent e021a6b7dd
commit 7ccdc340a3
No known key found for this signature in database
GPG Key ID: C55CB024B5314B57
6 changed files with 435 additions and 335 deletions

View File

@ -160,13 +160,17 @@ module Homebrew
run_cask_style(cask, old_contents)
pr_info = {
branch_name:,
commits: [{
commit_message:,
old_contents:,
pr_message: "Created with `brew bump-cask-pr`.",
sourcefile_path: cask.sourcefile_path,
}],
branch_name:,
pr_message: "Created with `brew bump-cask-pr`.",
tap: cask.tap,
pr_title: commit_message,
}
GitHub.create_bump_pr(pr_info, args:)
end

View File

@ -81,13 +81,14 @@ module Homebrew
description: "Include these additional Python packages when finding resources."
comma_array "--python-exclude-packages=",
description: "Exclude these Python packages when finding resources."
comma_array "--bump-synced=",
hidden: true
conflicts "--dry-run", "--write-only"
conflicts "--no-audit", "--strict"
conflicts "--no-audit", "--online"
conflicts "--url", "--tag"
named_args :formula, max: 1, without_api: true
named_args :named_formula, max: 1, without_api: true
end
sig { override.void }
@ -103,18 +104,17 @@ module Homebrew
# Use the user's browser, too.
ENV["BROWSER"] = Homebrew::EnvConfig.browser
formula = args.named.to_formulae.first
new_url = args.url
raise FormulaUnspecifiedError if formula.blank?
named_formula = args.named.to_formulae.first
raise FormulaUnspecifiedError if named_formula.blank?
odie "This formula is disabled!" if formula.disabled?
odie "This formula is deprecated and does not build!" if formula.deprecation_reason == :does_not_build
tap = formula.tap
odie "This formula is disabled!" if named_formula.disabled?
odie "This formula is deprecated and does not build!" if named_formula.deprecation_reason == :does_not_build
tap = named_formula.tap
odie "This formula is not in a tap!" if tap.blank?
odie "This formula's tap is not a Git repository!" unless tap.git?
odie <<~EOS unless tap.allow_bump?(formula.name)
Whoops, the #{formula.name} formula has its version update
odie <<~EOS unless tap.allow_bump?(named_formula.name)
Whoops, the #{named_formula.name} formula has its version update
pull requests automatically opened by BrewTestBot every ~3 hours!
We'd still love your contributions, though, so try another one
that's not in the autobump list:
@ -123,8 +123,8 @@ module Homebrew
odie "You have too many PRs open: close or merge some first!" if GitHub.too_many_open_prs?(tap)
formula_spec = formula.stable
odie "#{formula}: no stable specification found!" if formula_spec.blank?
named_formula_spec = named_formula.stable
odie "#{named_formula}: no stable specification found!" if named_formula_spec.blank?
# This will be run by `brew audit` later so run it first to not start
# spamming during normal output.
@ -135,8 +135,29 @@ module Homebrew
remote_branch = tap.git_repository.origin_branch_name
previous_branch = "-"
check_pull_requests(formula, tap_remote_repo, state: "open")
check_pull_requests(named_formula, tap_remote_repo, state: "open")
all_formulae = []
if args.bump_synced.present?
Array(args.bump_synced).each do |formula_name|
all_formulae << formula_name
end
else
all_formulae << args.named.first.to_s
end
return if all_formulae.empty?
commits = all_formulae.filter_map do |formula_name|
formula = Formula[formula_name]
raise FormulaUnspecifiedError if formula.blank?
formula_spec = formula.stable
odie "#{formula}: no stable specification found!" if formula_spec.blank?
formula_pr_message = ""
new_url = args.url
new_version = args.version
check_new_version(formula, tap_remote_repo, version: new_version) if new_version.present?
@ -192,7 +213,7 @@ module Homebrew
elsif new_url.blank? && new_version.blank?
raise UsageError, "#{formula}: no `--url` or `--version` argument specified!"
else
return unless new_version.present?
next unless new_version.present?
new_url ||= PyPI.update_pypi_url(old_url, new_version)
if new_url.blank?
@ -351,13 +372,10 @@ module Homebrew
version: new_formula_version.to_s
end
run_audit(formula, alias_rename, old_contents)
pr_message = "Created with `brew bump-formula-pr`."
if resources_checked.nil? && formula.resources.any? do |resource|
resource.livecheck.formula != :parent && !resource.name.start_with?("homebrew-")
end
pr_message += <<~EOS
formula_pr_message += <<~EOS
- [ ] `resource` blocks have been checked for updates.
@ -377,7 +395,7 @@ module Homebrew
if github_release_data.present?
pre = "pre" if github_release_data["prerelease"].present?
pr_message += <<~XML
formula_pr_message += <<~XML
<details>
<summary>#{pre}release notes</summary>
<pre>#{github_release_data["body"]}</pre>
@ -386,14 +404,59 @@ module Homebrew
end
end
pr_info = {
{
sourcefile_path: formula.path,
old_contents:,
commit_message: "#{formula.name} #{args.version}",
additional_files: alias_rename,
formula_pr_message:,
formula_name: formula.name,
new_version: new_formula_version,
}
end
commits.each do |commit|
commit_formula = Formula[commit[:formula_name]]
# For each formula, run `brew audit` to check for any issues.
audit_result = run_audit(commit_formula, commit[:additional_files],
skip_synced_versions: args.bump_synced.present?)
next unless audit_result
# If `brew audit` fails, revert the changes made to any formula.
commits.each do |revert|
revert_formula = Formula[revert[:formula_name]]
revert_formula.path.atomic_write(revert[:old_contents]) unless args.dry_run?
revert_alias_rename = revert[:additional_files]
if revert_alias_rename && (source = revert_alias_rename.first) && (destination = revert_alias_rename.last)
FileUtils.mv source, destination
end
end
odie "`brew audit` failed for #{commit[:formula_name]}!"
end
new_formula_version = T.must(commits.first)[:new_version]
pr_title = if args.bump_synced.nil?
"#{named_formula.name} #{new_formula_version}"
else
"#{Array(args.bump_synced).join(" ")} #{new_formula_version}"
end
pr_message = "Created by `brew bump-formula-pr`."
commits.each do |commit|
next if commit[:formula_pr_message].empty?
pr_message += "<h4>#{commit[:formula_name]}</h4>" if commits.length != 1
pr_message += "#{commit[:formula_pr_message]}<hr>"
end
pr_info = {
commits:,
remote:,
remote_branch:,
branch_name: "bump-#{formula.name}-#{new_formula_version}",
commit_message: "#{formula.name} #{new_formula_version}",
branch_name: "bump-#{named_formula.name}-#{new_formula_version}",
pr_title:,
previous_branch:,
tap: tap,
tap_remote_repo:,
@ -604,11 +667,15 @@ module Homebrew
[versioned_alias, "#{name}@#{new_alias_version}"]
end
sig { params(formula: Formula, alias_rename: T.nilable(T::Array[String]), old_contents: String).void }
def run_audit(formula, alias_rename, old_contents)
sig {
params(formula: Formula, alias_rename: T.nilable(T::Array[String]),
skip_synced_versions: T::Boolean).returns(T::Boolean)
}
def run_audit(formula, alias_rename, skip_synced_versions: false)
audit_args = ["--formula"]
audit_args << "--strict" if args.strict?
audit_args << "--online" if args.online?
audit_args << "--except=synced_versions_formulae" if skip_synced_versions
if args.dry_run?
if args.no_audit?
ohai "Skipping `brew audit`"
@ -617,7 +684,7 @@ module Homebrew
else
ohai "brew audit #{formula.path.basename}"
end
return
return true
end
if alias_rename && (source = alias_rename.first) && (destination = alias_rename.last)
FileUtils.mv source, destination
@ -632,13 +699,7 @@ module Homebrew
system HOMEBREW_BREW_FILE, "audit", formula.full_name
failed_audit = !$CHILD_STATUS.success?
end
return unless failed_audit
formula.path.atomic_write(old_contents)
if alias_rename && (source = alias_rename.first) && (destination = alias_rename.last)
FileUtils.mv source, destination
end
odie "`brew audit` failed!"
failed_audit
end
end
end

View File

@ -53,6 +53,8 @@ module Homebrew
description: "Open a pull request for the new version if none have been opened yet."
flag "--start-with=",
description: "Letter or word that the list of package results should alphabetically follow."
switch "--bump-synced",
description: "Bump additional formulae marked as synced with the given formulae."
conflicts "--cask", "--formula"
conflicts "--tap=", "--installed"
@ -470,11 +472,13 @@ module Homebrew
EOS
if formula_or_cask.is_a?(Formula) && formula_or_cask.synced_with_other_formulae?
outdated_synced_formulae = synced_with(formula_or_cask, new_version.general)
puts <<~EOS if outdated_synced_formulae.present?
if !args.bump_synced? && outdated_synced_formulae.present?
puts <<~EOS
Version syncing: #{title_name} version should be kept in sync with
#{outdated_synced_formulae.join(", ")}.
EOS
end
end
if !args.no_pull_requests? &&
(new_version.general != "unable to get versions") &&
!versions_equal
@ -519,7 +523,7 @@ module Homebrew
"--version=#{new_version.general}"
end
bump_cask_pr_args = [
bump_pr_args = [
"bump-#{version_info.type}-pr",
name,
*version_args,
@ -527,9 +531,13 @@ module Homebrew
"--message=Created by `brew bump`",
]
bump_cask_pr_args << "--no-fork" if args.no_fork?
bump_pr_args << "--no-fork" if args.no_fork?
system HOMEBREW_BREW_FILE, *bump_cask_pr_args
if args.bump_synced? && outdated_synced_formulae.present?
bump_pr_args << "--bump-synced=#{outdated_synced_formulae.join(",")}"
end
system HOMEBREW_BREW_FILE, *bump_pr_args
end
sig {

View File

@ -14,6 +14,9 @@ class Homebrew::DevCmd::Bump::Args < Homebrew::CLI::Args
sig { returns(T::Boolean) }
def auto?; end
sig { returns(T::Boolean) }
def bump_synced?; end
sig { returns(T::Boolean) }
def cask?; end

View File

@ -11,6 +11,9 @@ class Homebrew::DevCmd::BumpFormulaPr
end
class Homebrew::DevCmd::BumpFormulaPr::Args < Homebrew::CLI::Args
sig { returns(T.nilable(String)) }
def bump_synced; end
sig { returns(T::Boolean) }
def commit?; end

View File

@ -706,16 +706,49 @@ module GitHub
def self.create_bump_pr(info, args:)
tap = info[:tap]
sourcefile_path = info[:sourcefile_path]
old_contents = info[:old_contents]
additional_files = info[:additional_files] || []
remote = info[:remote] || "origin"
remote_branch = info[:remote_branch] || tap.git_repository.origin_branch_name
branch = info[:branch_name]
commit_message = info[:commit_message]
previous_branch = info[:previous_branch] || "-"
tap_remote_repo = info[:tap_remote_repo] || tap.full_name
pr_message = info[:pr_message]
pr_title = info[:pr_title]
commits = info[:commits]
username = tap.user
remote_url = Utils.popen_read("git", "remote", "get-url", "--push", "origin").chomp
tap.path.cd do
require "utils/popen"
git_dir = Utils.popen_read("git", "rev-parse", "--git-dir").chomp
shallow = !git_dir.empty? && File.exist?("#{git_dir}/shallow")
unless args.commit?
if args.no_fork?
remote_url = Utils.popen_read("git", "remote", "get-url", "--push", "origin").chomp
add_auth_token_to_url!(remote_url)
username = tap.user
else
begin
forked_repo_info!(tap_remote_repo, org: args.fork_org)
rescue *API::ERRORS => e
commits.each do |commit|
commit[:sourcefile_path].atomic_write(commit[:old_contents])
end
odie "Unable to fork: #{e.message}!"
end
end
safe_system "git", "fetch", "--unshallow", "origin" if shallow
end
safe_system "git", "checkout", "--no-track", "-b", branch, "#{remote}/#{remote_branch}" unless args.commit?
Utils::Git.set_name_email!
end
commits.each do |commit|
sourcefile_path = commit[:sourcefile_path]
commit_message = commit[:commit_message]
additional_files = commit[:additional_files] || []
sourcefile_path.parent.cd do
require "utils/popen"
@ -733,41 +766,27 @@ module GitHub
ohai fork_message
"FORK_URL"
end
ohai "git checkout --no-track -b #{branch} #{remote}/#{remote_branch}"
ohai "git fetch --unshallow origin" if shallow
ohai "git add #{changed_files.join(" ")}"
ohai "git checkout --no-track -b #{branch} #{remote}/#{remote_branch}"
ohai "git commit --no-edit --verbose --message='#{commit_message}' " \
"-- #{changed_files.join(" ")}"
ohai "git push --set-upstream #{remote_url} #{branch}:#{branch}"
ohai "git checkout --quiet #{previous_branch}"
ohai "create pull request with GitHub API (base branch: #{remote_branch})"
else
unless args.commit?
if args.no_fork?
remote_url = Utils.popen_read("git", "remote", "get-url", "--push", "origin").chomp
add_auth_token_to_url!(remote_url)
username = tap.user
else
begin
remote_url, username = forked_repo_info!(tap_remote_repo, org: args.fork_org)
rescue *API::ERRORS => e
sourcefile_path.atomic_write(old_contents)
odie "Unable to fork: #{e.message}!"
end
end
safe_system "git", "fetch", "--unshallow", "origin" if shallow
end
safe_system "git", "add", *changed_files
safe_system "git", "checkout", "--no-track", "-b", branch, "#{remote}/#{remote_branch}" unless args.commit?
Utils::Git.set_name_email!
safe_system "git", "commit", "--no-edit", "--verbose",
"--message=#{commit_message}",
"--", *changed_files
end
end
end
return if args.commit?
tap.path.cd do
system_command!("git", args: ["push", "--set-upstream", remote_url, "#{branch}:#{branch}"],
print_stdout: true)
safe_system "git", "checkout", "--quiet", previous_branch
@ -786,7 +805,7 @@ module GitHub
end
begin
url = create_pull_request(tap_remote_repo, commit_message,
url = create_pull_request(tap_remote_repo, pr_title,
"#{username}:#{branch}", remote_branch, pr_message)["html_url"]
if args.no_browse?
puts url
@ -794,8 +813,10 @@ module GitHub
exec_browser url
end
rescue *API::ERRORS => e
odie "Unable to open pull request: #{e.message}!"
commits.each do |commit|
commit[:sourcefile_path].atomic_write(commit[:old_contents])
end
odie "Unable to open pull request: #{e.message}!"
end
end
end