brew/Library/Homebrew/dev-cmd/bump-unversioned-casks.rb

163 lines
4.5 KiB
Ruby
Raw Normal View History

2020-12-04 00:07:02 +01:00
# typed: false
# frozen_string_literal: true
require "cask/download"
require "cask/installer"
require "cask/cask_loader"
require "cli/parser"
require "tap"
require "unversioned_cask_checker"
2020-12-04 00:07:02 +01:00
module Homebrew
extend T::Sig
extend SystemCommand::Mixin
sig { returns(CLI::Parser) }
def self.bump_unversioned_casks_args
Homebrew::CLI::Parser.new do
usage_banner <<~EOS
`bump-unversioned-casks` [<options>] [<tap>]
Check all casks with unversioned URLs in a given <tap> for updates.
EOS
switch "-n", "--dry-run",
description: "Do everything except caching state and opening pull requests."
2020-12-04 00:07:02 +01:00
flag "--limit=",
2020-12-05 21:30:02 +01:00
description: "Maximum runtime in minutes."
2020-12-04 00:07:02 +01:00
flag "--state-file=",
description: "File for caching state."
2020-12-04 00:07:02 +01:00
named 1
end
end
sig { void }
def self.bump_unversioned_casks
args = bump_unversioned_casks_args.parse
state_file = if args.state_file.present?
Pathname(args.state_file).expand_path
else
HOMEBREW_CACHE/"bump_unversioned_casks.json"
end
state_file.dirname.mkpath
tap = Tap.fetch(args.named.first)
2020-12-05 21:30:02 +01:00
state = state_file.exist? ? JSON.parse(state_file.read) : {}
2020-12-04 00:07:02 +01:00
cask_files = tap.cask_files
unversioned_cask_files = cask_files.select do |cask_file|
url = cask_file.each_line do |line|
url = line[/\s*url\s+"([^"]+)"\s*/, 1]
break url if url
end
url.present? && url.exclude?('#{')
end.sort
unversioned_casks = unversioned_cask_files.map { |path| Cask::CaskLoader.load(path) }
ohai "Unversioned Casks: #{unversioned_casks.count}"
2020-12-04 00:07:02 +01:00
2020-12-05 21:30:02 +01:00
checked, unchecked = unversioned_casks.partition { |c| state.key?(c.full_name) }
queue = Queue.new
# Start with random casks which have not been checked.
2020-12-05 21:30:02 +01:00
unchecked.shuffle.each do |c|
queue.enq c
end
# Continue with previously checked casks, ordered by when they were last checked.
2020-12-05 21:30:02 +01:00
checked.sort_by { |c| state.dig(c.full_name, "check_time") }.each do |c|
queue.enq c
end
limit = args.limit.presence&.to_i
end_time = Time.now + limit.minutes if limit
until queue.empty? || (end_time && end_time < Time.now)
cask = queue.deq
2020-12-04 00:07:02 +01:00
ohai "Checking #{cask.full_name}"
unversioned_cask_checker = UnversionedCaskChecker.new(cask)
unless unversioned_cask_checker.single_app_cask? || unversioned_cask_checker.single_pkg_cask?
opoo "Skipping, not a single-app or PKG cask."
2020-12-06 19:14:09 +01:00
next
end
2020-12-05 21:30:02 +01:00
last_state = state.fetch(cask.full_name, {})
2020-12-04 00:07:02 +01:00
last_check_time = last_state["check_time"]&.yield_self { |t| Time.parse(t) }
check_time = Time.now
if last_check_time && check_time < (last_check_time + 1.day)
opoo "Skipping, already checked within the last 24 hours."
next
end
last_sha256 = last_state["sha256"]
last_time = last_state["time"]&.yield_self { |t| Time.parse(t) }
last_file_size = last_state["file_size"]
download = Cask::Download.new(cask)
time, file_size = begin
download.time_file_size
rescue
[nil, nil]
end
2020-12-04 00:07:02 +01:00
if last_time != time || last_file_size != file_size
begin
cached_download = unversioned_cask_checker.installer.download
rescue => e
onoe e
next
end
2020-12-04 00:07:02 +01:00
sha256 = cached_download.sha256
2020-12-04 00:07:02 +01:00
if last_sha256 != sha256 && (version = unversioned_cask_checker.guess_cask_version)
if cask.version == version
oh1 "Cask #{cask} is up-to-date at #{version}"
else
bump_cask_pr_args = [
"bump-cask-pr",
"--version", version.to_s,
"--sha256", ":no_check",
"--message", "Automatic update via `brew bump-unversioned-casks`.",
cask.sourcefile_path
]
if args.dry_run?
bump_cask_pr_args << "--dry-run"
oh1 "Would bump #{cask} from #{cask.version} to #{version}"
else
oh1 "Bumping #{cask} from #{cask.version} to #{version}"
end
begin
system_command! HOMEBREW_BREW_FILE, args: bump_cask_pr_args
rescue ErrorDuringExecution => e
onoe e
2020-12-06 03:24:32 +01:00
next
end
2020-12-04 00:07:02 +01:00
end
end
end
2020-12-05 21:30:02 +01:00
state[cask.full_name] = {
"sha256" => sha256,
"check_time" => check_time.iso8601,
"time" => time&.iso8601,
"file_size" => file_size,
}
2020-12-04 00:07:02 +01:00
state_file.atomic_write JSON.generate(state) unless args.dry_run?
end
2020-12-04 00:07:02 +01:00
end
end