2024-06-02 15:14:25 +01:00
|
|
|
# typed: strict
|
2023-03-07 22:17:02 +09:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require "env_config"
|
|
|
|
require "cask/config"
|
|
|
|
|
|
|
|
module Cask
|
|
|
|
class Upgrade
|
|
|
|
sig {
|
|
|
|
params(
|
|
|
|
casks: Cask,
|
|
|
|
args: Homebrew::CLI::Args,
|
|
|
|
force: T.nilable(T::Boolean),
|
|
|
|
greedy: T.nilable(T::Boolean),
|
|
|
|
greedy_latest: T.nilable(T::Boolean),
|
|
|
|
greedy_auto_updates: T.nilable(T::Boolean),
|
|
|
|
dry_run: T.nilable(T::Boolean),
|
|
|
|
skip_cask_deps: T.nilable(T::Boolean),
|
|
|
|
verbose: T.nilable(T::Boolean),
|
2024-07-15 11:50:37 -05:00
|
|
|
quiet: T.nilable(T::Boolean),
|
2023-03-07 22:17:02 +09:00
|
|
|
binaries: T.nilable(T::Boolean),
|
|
|
|
quarantine: T.nilable(T::Boolean),
|
|
|
|
require_sha: T.nilable(T::Boolean),
|
|
|
|
).returns(T::Boolean)
|
|
|
|
}
|
|
|
|
def self.upgrade_casks(
|
|
|
|
*casks,
|
|
|
|
args:,
|
|
|
|
force: false,
|
|
|
|
greedy: false,
|
|
|
|
greedy_latest: false,
|
|
|
|
greedy_auto_updates: false,
|
|
|
|
dry_run: false,
|
|
|
|
skip_cask_deps: false,
|
|
|
|
verbose: false,
|
2024-07-15 11:50:37 -05:00
|
|
|
quiet: false,
|
2023-03-07 22:17:02 +09:00
|
|
|
binaries: nil,
|
|
|
|
quarantine: nil,
|
|
|
|
require_sha: nil
|
|
|
|
)
|
|
|
|
quarantine = true if quarantine.nil?
|
|
|
|
|
2023-04-05 09:44:32 -04:00
|
|
|
greedy = true if Homebrew::EnvConfig.upgrade_greedy?
|
2023-04-05 09:28:15 -04:00
|
|
|
|
2025-02-25 15:34:49 +00:00
|
|
|
greedy_casks = if (upgrade_greedy_casks = Homebrew::EnvConfig.upgrade_greedy_casks.presence)
|
|
|
|
upgrade_greedy_casks.split
|
|
|
|
else
|
|
|
|
[]
|
|
|
|
end
|
|
|
|
|
2023-03-07 22:17:02 +09:00
|
|
|
outdated_casks = if casks.empty?
|
|
|
|
Caskroom.casks(config: Config.from_args(args)).select do |cask|
|
2025-02-25 15:34:49 +00:00
|
|
|
cask_greedy = greedy || greedy_casks.include?(cask.token)
|
|
|
|
cask.outdated?(greedy: cask_greedy, greedy_latest:,
|
2024-03-07 16:20:20 +00:00
|
|
|
greedy_auto_updates:)
|
2023-03-07 22:17:02 +09:00
|
|
|
end
|
|
|
|
else
|
|
|
|
casks.select do |cask|
|
|
|
|
raise CaskNotInstalledError, cask if !cask.installed? && !force
|
|
|
|
|
|
|
|
if cask.outdated?(greedy: true)
|
|
|
|
true
|
|
|
|
elsif cask.version.latest?
|
2024-07-15 11:50:37 -05:00
|
|
|
opoo "Not upgrading #{cask.token}, the downloaded artifact has not changed" unless quiet
|
2023-03-07 22:17:02 +09:00
|
|
|
false
|
|
|
|
else
|
2024-07-15 11:50:37 -05:00
|
|
|
opoo "Not upgrading #{cask.token}, the latest version is already installed" unless quiet
|
2023-03-07 22:17:02 +09:00
|
|
|
false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
manual_installer_casks = outdated_casks.select do |cask|
|
2024-08-11 16:56:20 -07:00
|
|
|
cask.artifacts.any? do |artifact|
|
|
|
|
artifact.is_a?(Artifact::Installer) && artifact.manual_install
|
|
|
|
end
|
2023-03-07 22:17:02 +09:00
|
|
|
end
|
|
|
|
|
|
|
|
if manual_installer_casks.present?
|
|
|
|
count = manual_installer_casks.count
|
|
|
|
ofail "Not upgrading #{count} `installer manual` #{::Utils.pluralize("cask", count)}."
|
|
|
|
puts manual_installer_casks.map(&:to_s)
|
|
|
|
outdated_casks -= manual_installer_casks
|
|
|
|
end
|
|
|
|
|
|
|
|
return false if outdated_casks.empty?
|
|
|
|
|
2025-02-25 15:34:49 +00:00
|
|
|
if casks.empty? && !greedy && greedy_casks.empty?
|
2023-03-07 22:17:02 +09:00
|
|
|
if !greedy_auto_updates && !greedy_latest
|
|
|
|
ohai "Casks with 'auto_updates true' or 'version :latest' " \
|
|
|
|
"will not be upgraded; pass `--greedy` to upgrade them."
|
|
|
|
end
|
|
|
|
if greedy_auto_updates && !greedy_latest
|
|
|
|
ohai "Casks with 'version :latest' will not be upgraded; pass `--greedy-latest` to upgrade them."
|
|
|
|
end
|
|
|
|
if !greedy_auto_updates && greedy_latest
|
|
|
|
ohai "Casks with 'auto_updates true' will not be upgraded; pass `--greedy-auto-updates` to upgrade them."
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
verb = dry_run ? "Would upgrade" : "Upgrading"
|
|
|
|
oh1 "#{verb} #{outdated_casks.count} outdated #{::Utils.pluralize("package", outdated_casks.count)}:"
|
|
|
|
|
|
|
|
caught_exceptions = []
|
|
|
|
|
|
|
|
upgradable_casks = outdated_casks.map do |c|
|
2023-05-15 11:10:08 +02:00
|
|
|
unless c.installed?
|
2023-03-07 22:17:02 +09:00
|
|
|
odie <<~EOS
|
|
|
|
The cask '#{c.token}' was affected by a bug and cannot be upgraded as-is. To fix this, run:
|
|
|
|
brew reinstall --cask --force #{c.token}
|
|
|
|
EOS
|
|
|
|
end
|
|
|
|
|
|
|
|
[CaskLoader.load(c.installed_caskfile), c]
|
|
|
|
end
|
|
|
|
|
|
|
|
puts upgradable_casks
|
|
|
|
.map { |(old_cask, new_cask)| "#{new_cask.full_name} #{old_cask.version} -> #{new_cask.version}" }
|
|
|
|
.join("\n")
|
|
|
|
return true if dry_run
|
|
|
|
|
|
|
|
upgradable_casks.each do |(old_cask, new_cask)|
|
|
|
|
upgrade_cask(
|
|
|
|
old_cask, new_cask,
|
2024-03-07 16:20:20 +00:00
|
|
|
binaries:, force:, skip_cask_deps:, verbose:,
|
|
|
|
quarantine:, require_sha:
|
2023-03-07 22:17:02 +09:00
|
|
|
)
|
|
|
|
rescue => e
|
2023-05-10 14:46:51 +01:00
|
|
|
new_exception = e.exception("#{new_cask.full_name}: #{e}")
|
|
|
|
new_exception.set_backtrace(e.backtrace)
|
|
|
|
caught_exceptions << new_exception
|
2023-03-07 22:17:02 +09:00
|
|
|
next
|
|
|
|
end
|
|
|
|
|
|
|
|
return true if caught_exceptions.empty?
|
|
|
|
raise MultipleCaskErrors, caught_exceptions if caught_exceptions.count > 1
|
2023-03-18 16:03:10 -07:00
|
|
|
raise caught_exceptions.fetch(0) if caught_exceptions.count == 1
|
|
|
|
|
|
|
|
false
|
2023-03-07 22:17:02 +09:00
|
|
|
end
|
|
|
|
|
2023-05-12 14:47:56 -05:00
|
|
|
sig {
|
|
|
|
params(
|
|
|
|
old_cask: Cask,
|
|
|
|
new_cask: Cask,
|
|
|
|
binaries: T.nilable(T::Boolean),
|
|
|
|
force: T.nilable(T::Boolean),
|
|
|
|
quarantine: T.nilable(T::Boolean),
|
|
|
|
require_sha: T.nilable(T::Boolean),
|
|
|
|
skip_cask_deps: T.nilable(T::Boolean),
|
|
|
|
verbose: T.nilable(T::Boolean),
|
|
|
|
).void
|
|
|
|
}
|
2023-03-07 22:17:02 +09:00
|
|
|
def self.upgrade_cask(
|
|
|
|
old_cask, new_cask,
|
|
|
|
binaries:, force:, quarantine:, require_sha:, skip_cask_deps:, verbose:
|
|
|
|
)
|
|
|
|
require "cask/installer"
|
|
|
|
|
|
|
|
start_time = Time.now
|
|
|
|
odebug "Started upgrade process for Cask #{old_cask}"
|
|
|
|
old_config = old_cask.config
|
|
|
|
|
|
|
|
old_options = {
|
2024-03-07 16:20:20 +00:00
|
|
|
binaries:,
|
|
|
|
verbose:,
|
|
|
|
force:,
|
2023-03-07 22:17:02 +09:00
|
|
|
upgrade: true,
|
|
|
|
}.compact
|
|
|
|
|
|
|
|
old_cask_installer =
|
|
|
|
Installer.new(old_cask, **old_options)
|
|
|
|
|
|
|
|
new_cask.config = new_cask.default_config.merge(old_config)
|
|
|
|
|
|
|
|
new_options = {
|
2024-03-07 16:20:20 +00:00
|
|
|
binaries:,
|
|
|
|
verbose:,
|
|
|
|
force:,
|
|
|
|
skip_cask_deps:,
|
|
|
|
require_sha:,
|
2023-03-07 22:17:02 +09:00
|
|
|
upgrade: true,
|
2024-03-07 16:20:20 +00:00
|
|
|
quarantine:,
|
2023-03-07 22:17:02 +09:00
|
|
|
}.compact
|
|
|
|
|
|
|
|
new_cask_installer =
|
|
|
|
Installer.new(new_cask, **new_options)
|
|
|
|
|
|
|
|
started_upgrade = false
|
|
|
|
new_artifacts_installed = false
|
|
|
|
|
|
|
|
begin
|
|
|
|
oh1 "Upgrading #{Formatter.identifier(old_cask)}"
|
|
|
|
|
|
|
|
# Start new cask's installation steps
|
|
|
|
new_cask_installer.check_conflicts
|
|
|
|
|
|
|
|
if (caveats = new_cask_installer.caveats)
|
|
|
|
puts caveats
|
|
|
|
end
|
|
|
|
|
|
|
|
new_cask_installer.fetch
|
|
|
|
|
|
|
|
# Move the old cask's artifacts back to staging
|
2023-04-08 16:13:12 -05:00
|
|
|
old_cask_installer.start_upgrade(successor: new_cask)
|
2023-03-07 22:17:02 +09:00
|
|
|
# And flag it so in case of error
|
|
|
|
started_upgrade = true
|
|
|
|
|
|
|
|
# Install the new cask
|
|
|
|
new_cask_installer.stage
|
|
|
|
|
2023-04-08 16:13:12 -05:00
|
|
|
new_cask_installer.install_artifacts(predecessor: old_cask)
|
2023-03-07 22:17:02 +09:00
|
|
|
new_artifacts_installed = true
|
|
|
|
|
2023-04-08 14:10:58 +02:00
|
|
|
# If successful, wipe the old cask from staging.
|
2023-03-07 22:17:02 +09:00
|
|
|
old_cask_installer.finalize_upgrade
|
|
|
|
rescue => e
|
2023-04-08 16:13:12 -05:00
|
|
|
new_cask_installer.uninstall_artifacts(successor: old_cask) if new_artifacts_installed
|
2023-03-07 22:17:02 +09:00
|
|
|
new_cask_installer.purge_versioned_files
|
2023-04-08 16:13:12 -05:00
|
|
|
old_cask_installer.revert_upgrade(predecessor: new_cask) if started_upgrade
|
2023-03-07 22:17:02 +09:00
|
|
|
raise e
|
|
|
|
end
|
|
|
|
|
|
|
|
end_time = Time.now
|
|
|
|
Homebrew.messages.package_installed(new_cask.token, end_time - start_time)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|