2020-10-10 14:16:11 +02:00
|
|
|
# typed: false
|
2019-04-19 15:38:03 +09:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2018-09-03 19:39:07 +01:00
|
|
|
require "cask/cask_loader"
|
|
|
|
require "cask/config"
|
|
|
|
require "cask/dsl"
|
|
|
|
require "cask/metadata"
|
2018-06-07 14:42:58 +02:00
|
|
|
require "searchable"
|
2022-06-23 17:19:27 -04:00
|
|
|
require "utils/bottles"
|
2016-08-18 22:11:42 +03:00
|
|
|
|
2018-09-06 08:29:14 +02:00
|
|
|
module Cask
|
2020-08-24 22:50:21 +02:00
|
|
|
# An instance of a cask.
|
|
|
|
#
|
|
|
|
# @api private
|
2016-09-24 13:52:43 +02:00
|
|
|
class Cask
|
2020-10-20 12:03:48 +02:00
|
|
|
extend T::Sig
|
|
|
|
|
2016-09-24 13:52:43 +02:00
|
|
|
extend Forwardable
|
2018-06-07 14:42:58 +02:00
|
|
|
extend Searchable
|
2017-04-20 10:48:32 +02:00
|
|
|
include Metadata
|
2016-08-18 22:11:42 +03:00
|
|
|
|
2021-08-14 16:18:13 -04:00
|
|
|
attr_reader :token, :sourcefile_path, :source, :config, :default_config
|
2017-07-29 16:27:54 +02:00
|
|
|
|
2022-06-12 15:27:31 -04:00
|
|
|
attr_accessor :download, :allow_reassignment
|
2022-05-07 20:19:54 -07:00
|
|
|
|
2022-01-03 14:59:10 +00:00
|
|
|
def self.all
|
|
|
|
Tap.flat_map(&:cask_files).map do |f|
|
|
|
|
CaskLoader::FromTapPathLoader.new(f).load(config: nil)
|
2019-10-13 10:03:26 +01:00
|
|
|
rescue CaskUnreadableError => e
|
|
|
|
opoo e.message
|
2022-01-03 14:59:10 +00:00
|
|
|
|
|
|
|
nil
|
|
|
|
end.compact
|
2018-04-14 10:28:28 +02:00
|
|
|
end
|
|
|
|
|
2017-07-29 16:27:54 +02:00
|
|
|
def tap
|
|
|
|
return super if block_given? # Object#tap
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2017-07-29 16:27:54 +02:00
|
|
|
@tap
|
|
|
|
end
|
|
|
|
|
2022-06-12 15:27:31 -04:00
|
|
|
def initialize(token, sourcefile_path: nil, source: nil, tap: nil, config: nil, allow_reassignment: false, &block)
|
2016-09-24 13:52:43 +02:00
|
|
|
@token = token
|
|
|
|
@sourcefile_path = sourcefile_path
|
2021-08-14 16:18:13 -04:00
|
|
|
@source = source
|
2017-07-29 16:27:54 +02:00
|
|
|
@tap = tap
|
2022-06-12 15:27:31 -04:00
|
|
|
@allow_reassignment = allow_reassignment
|
2019-02-03 13:03:16 +01:00
|
|
|
@block = block
|
2020-09-29 23:46:30 +02:00
|
|
|
|
|
|
|
@default_config = config || Config.new
|
|
|
|
|
|
|
|
self.config = if config_path.exist?
|
2021-01-12 16:05:06 +01:00
|
|
|
Config.from_json(File.read(config_path), ignore_invalid_keys: true)
|
2020-09-29 23:46:30 +02:00
|
|
|
else
|
|
|
|
@default_config
|
|
|
|
end
|
2019-02-03 13:03:16 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
def config=(config)
|
2017-12-03 21:21:31 +01:00
|
|
|
@config = config
|
2019-02-03 13:03:16 +01:00
|
|
|
|
2022-06-23 17:19:27 -04:00
|
|
|
refresh
|
|
|
|
end
|
|
|
|
|
|
|
|
def refresh
|
2017-04-06 00:33:31 +02:00
|
|
|
@dsl = DSL.new(self)
|
2019-02-03 13:03:16 +01:00
|
|
|
return unless @block
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2019-02-03 13:03:16 +01:00
|
|
|
@dsl.instance_eval(&@block)
|
2016-10-23 14:26:17 +02:00
|
|
|
@dsl.language_eval
|
2016-09-24 13:52:43 +02:00
|
|
|
end
|
2016-08-18 22:11:42 +03:00
|
|
|
|
2016-09-24 13:52:43 +02:00
|
|
|
DSL::DSL_METHODS.each do |method_name|
|
2017-11-24 17:44:01 +01:00
|
|
|
define_method(method_name) { |&block| @dsl.send(method_name, &block) }
|
2016-09-24 13:52:43 +02:00
|
|
|
end
|
2016-08-18 22:11:42 +03:00
|
|
|
|
2020-10-20 12:03:48 +02:00
|
|
|
sig { returns(T::Array[[String, String]]) }
|
2016-09-24 13:52:43 +02:00
|
|
|
def timestamped_versions
|
2017-04-20 10:48:32 +02:00
|
|
|
Pathname.glob(metadata_timestamped_path(version: "*", timestamp: "*"))
|
|
|
|
.map { |p| p.relative_path_from(p.parent.parent) }
|
2016-09-24 13:52:43 +02:00
|
|
|
.sort_by(&:basename) # sort by timestamp
|
2017-03-06 22:50:22 +01:00
|
|
|
.map { |p| p.split.map(&:to_s) }
|
2016-09-24 13:52:43 +02:00
|
|
|
end
|
2016-08-18 22:11:42 +03:00
|
|
|
|
2016-09-24 13:52:43 +02:00
|
|
|
def versions
|
|
|
|
timestamped_versions.map(&:first)
|
|
|
|
.reverse
|
|
|
|
.uniq
|
|
|
|
.reverse
|
|
|
|
end
|
2016-08-18 22:11:42 +03:00
|
|
|
|
2021-08-24 11:35:19 -04:00
|
|
|
def os_versions
|
|
|
|
@os_versions ||= begin
|
|
|
|
version_os_hash = {}
|
|
|
|
actual_version = MacOS.full_version.to_s
|
|
|
|
|
|
|
|
MacOS::Version::SYMBOLS.each do |os_name, os_version|
|
|
|
|
MacOS.full_version = os_version
|
2021-08-25 10:30:38 -04:00
|
|
|
cask = CaskLoader.load(token)
|
|
|
|
version_os_hash[os_name] = cask.version if cask.version != version
|
2021-08-24 11:35:19 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
version_os_hash
|
2021-08-25 10:30:38 -04:00
|
|
|
ensure
|
|
|
|
MacOS.full_version = actual_version
|
2021-08-24 11:35:19 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-07-08 19:33:44 -07:00
|
|
|
def full_name
|
2018-04-14 11:32:29 +02:00
|
|
|
return token if tap.nil?
|
2018-06-30 06:03:51 +02:00
|
|
|
return token if tap.user == "Homebrew"
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2018-04-14 11:32:29 +02:00
|
|
|
"#{tap.name}/#{token}"
|
2017-07-08 19:33:44 -07:00
|
|
|
end
|
|
|
|
|
2016-09-24 13:52:43 +02:00
|
|
|
def installed?
|
|
|
|
!versions.empty?
|
|
|
|
end
|
2016-08-18 22:11:42 +03:00
|
|
|
|
2020-10-20 12:03:48 +02:00
|
|
|
sig { returns(T.nilable(Time)) }
|
2019-05-31 20:38:28 +02:00
|
|
|
def install_time
|
|
|
|
_, time = timestamped_versions.last
|
|
|
|
return unless time
|
|
|
|
|
|
|
|
Time.strptime(time, Metadata::TIMESTAMP_FORMAT)
|
|
|
|
end
|
|
|
|
|
2017-03-12 21:59:13 +01:00
|
|
|
def installed_caskfile
|
|
|
|
installed_version = timestamped_versions.last
|
2021-07-06 23:48:26 +05:30
|
|
|
metadata_main_container_path.join(*installed_version, "Casks", "#{token}.rb")
|
2017-03-12 21:59:13 +01:00
|
|
|
end
|
|
|
|
|
2019-02-02 17:11:37 +01:00
|
|
|
def config_path
|
2021-07-06 23:48:26 +05:30
|
|
|
metadata_main_container_path/"config.json"
|
2019-02-02 17:11:37 +01:00
|
|
|
end
|
|
|
|
|
2022-05-31 10:44:34 -07:00
|
|
|
def checksumable?
|
2022-06-12 11:24:50 -04:00
|
|
|
DownloadStrategyDetector.detect(url.to_s, url.using) <= AbstractFileDownloadStrategy
|
2022-05-31 10:44:34 -07:00
|
|
|
end
|
|
|
|
|
2022-05-07 20:19:54 -07:00
|
|
|
def download_sha_path
|
|
|
|
metadata_main_container_path/"LATEST_DOWNLOAD_SHA256"
|
|
|
|
end
|
|
|
|
|
|
|
|
def new_download_sha
|
|
|
|
require "cask/installer"
|
|
|
|
|
2022-05-31 10:44:34 -07:00
|
|
|
# Call checksumable? before hashing
|
2022-05-07 20:19:54 -07:00
|
|
|
@new_download_sha ||= Installer.new(self, verify_download_integrity: false)
|
|
|
|
.download(quiet: true)
|
|
|
|
.instance_eval { |x| Digest::SHA256.file(x).hexdigest }
|
|
|
|
end
|
|
|
|
|
|
|
|
def outdated_download_sha?
|
2022-05-31 10:44:34 -07:00
|
|
|
return true unless checksumable?
|
|
|
|
|
2022-05-07 20:19:54 -07:00
|
|
|
current_download_sha = download_sha_path.read if download_sha_path.exist?
|
|
|
|
current_download_sha.blank? || current_download_sha != new_download_sha
|
|
|
|
end
|
|
|
|
|
2019-02-03 13:03:16 +01:00
|
|
|
def caskroom_path
|
|
|
|
@caskroom_path ||= Caskroom.path.join(token)
|
|
|
|
end
|
|
|
|
|
2021-07-01 23:01:22 +10:00
|
|
|
def outdated?(greedy: false, greedy_latest: false, greedy_auto_updates: false)
|
2021-07-01 23:02:15 +10:00
|
|
|
!outdated_versions(greedy: greedy, greedy_latest: greedy_latest,
|
|
|
|
greedy_auto_updates: greedy_auto_updates).empty?
|
2017-02-27 22:33:34 +02:00
|
|
|
end
|
|
|
|
|
2021-07-01 23:01:22 +10:00
|
|
|
def outdated_versions(greedy: false, greedy_latest: false, greedy_auto_updates: false)
|
2017-02-27 22:33:34 +02:00
|
|
|
# special case: tap version is not available
|
|
|
|
return [] if version.nil?
|
|
|
|
|
2022-06-14 17:19:29 -04:00
|
|
|
if version.latest?
|
2022-05-24 22:38:30 +10:00
|
|
|
return versions if (greedy || greedy_latest) && outdated_download_sha?
|
2022-05-07 20:19:54 -07:00
|
|
|
|
2022-05-24 22:38:30 +10:00
|
|
|
return []
|
2022-05-28 10:14:51 +10:00
|
|
|
elsif auto_updates && !greedy && !greedy_auto_updates
|
2022-05-07 20:19:54 -07:00
|
|
|
return []
|
|
|
|
end
|
|
|
|
|
2017-02-27 22:33:34 +02:00
|
|
|
installed = versions
|
|
|
|
current = installed.last
|
|
|
|
|
|
|
|
# not outdated unless there is a different version on tap
|
2022-06-14 17:19:29 -04:00
|
|
|
return [] if current == version
|
2017-02-27 22:33:34 +02:00
|
|
|
|
|
|
|
# collect all installed versions that are different than tap version and return them
|
2022-06-14 17:19:29 -04:00
|
|
|
installed.reject { |v| v == version }
|
2017-02-27 22:33:34 +02:00
|
|
|
end
|
|
|
|
|
2021-07-01 23:01:22 +10:00
|
|
|
def outdated_info(greedy, verbose, json, greedy_latest, greedy_auto_updates)
|
2020-04-29 18:16:39 +08:00
|
|
|
return token if !verbose && !json
|
2020-04-28 12:21:51 +08:00
|
|
|
|
2021-07-01 23:02:15 +10:00
|
|
|
installed_versions = outdated_versions(greedy: greedy, greedy_latest: greedy_latest,
|
|
|
|
greedy_auto_updates: greedy_auto_updates).join(", ")
|
2020-04-28 12:21:51 +08:00
|
|
|
|
2020-04-26 21:07:42 +08:00
|
|
|
if json
|
|
|
|
{
|
2020-04-26 21:31:21 +08:00
|
|
|
name: token,
|
2020-04-28 12:21:51 +08:00
|
|
|
installed_versions: installed_versions,
|
2020-04-26 21:31:21 +08:00
|
|
|
current_version: version,
|
2020-04-26 21:07:42 +08:00
|
|
|
}
|
|
|
|
else
|
2020-04-28 12:21:51 +08:00
|
|
|
"#{token} (#{installed_versions}) != #{version}"
|
2020-04-26 21:07:42 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-09-24 13:52:43 +02:00
|
|
|
def to_s
|
|
|
|
@token
|
|
|
|
end
|
2016-08-18 22:11:42 +03:00
|
|
|
|
2017-06-28 17:53:59 +02:00
|
|
|
def hash
|
|
|
|
token.hash
|
|
|
|
end
|
|
|
|
|
|
|
|
def eql?(other)
|
2021-02-17 01:18:25 +05:30
|
|
|
instance_of?(other.class) && token == other.token
|
2017-06-28 17:53:59 +02:00
|
|
|
end
|
|
|
|
alias == eql?
|
|
|
|
|
2022-06-23 17:19:27 -04:00
|
|
|
def to_hash
|
2018-07-17 10:04:17 +01:00
|
|
|
{
|
2019-07-09 20:43:45 +02:00
|
|
|
"token" => token,
|
2021-02-07 19:00:07 +01:00
|
|
|
"full_token" => full_name,
|
|
|
|
"tap" => tap&.name,
|
2018-11-02 17:18:07 +00:00
|
|
|
"name" => name,
|
2020-07-29 15:40:31 +01:00
|
|
|
"desc" => desc,
|
2018-11-02 17:18:07 +00:00
|
|
|
"homepage" => homepage,
|
|
|
|
"url" => url,
|
|
|
|
"appcast" => appcast,
|
|
|
|
"version" => version,
|
2021-08-24 11:35:19 -04:00
|
|
|
"versions" => os_versions,
|
2021-01-24 18:27:23 -08:00
|
|
|
"installed" => versions.last,
|
2021-01-09 22:46:27 -05:00
|
|
|
"outdated" => outdated?,
|
2018-11-02 17:18:07 +00:00
|
|
|
"sha256" => sha256,
|
2019-03-27 15:59:38 +00:00
|
|
|
"artifacts" => artifacts.map(&method(:to_h_gsubs)),
|
2019-09-14 10:08:42 -04:00
|
|
|
"caveats" => (to_h_string_gsubs(caveats) unless caveats.empty?),
|
2018-11-02 17:18:07 +00:00
|
|
|
"depends_on" => depends_on,
|
2019-01-27 21:34:24 +01:00
|
|
|
"conflicts_with" => conflicts_with,
|
2018-11-02 17:18:07 +00:00
|
|
|
"container" => container,
|
|
|
|
"auto_updates" => auto_updates,
|
2018-07-02 09:05:49 +01:00
|
|
|
}
|
|
|
|
end
|
2019-03-27 15:59:38 +00:00
|
|
|
|
2022-06-23 17:19:27 -04:00
|
|
|
def to_h
|
|
|
|
hash = to_hash
|
|
|
|
variations = {}
|
|
|
|
|
|
|
|
hash_keys_to_skip = %w[outdated installed versions]
|
|
|
|
|
|
|
|
if @dsl.on_system_blocks_exist?
|
|
|
|
[:arm, :intel].each do |arch|
|
|
|
|
MacOS::Version::SYMBOLS.each_key do |os_name|
|
|
|
|
# Big Sur is the first version of macOS that supports arm
|
|
|
|
next if arch == :arm && MacOS::Version.from_symbol(os_name) < MacOS::Version.from_symbol(:big_sur)
|
|
|
|
|
|
|
|
Homebrew::Override.os = os_name
|
|
|
|
Homebrew::Override.arch = arch
|
|
|
|
|
|
|
|
refresh
|
|
|
|
|
|
|
|
bottle_tag = ::Utils::Bottles::Tag.new(system: os_name, arch: arch).to_sym
|
|
|
|
to_hash.each do |key, value|
|
|
|
|
next if hash_keys_to_skip.include? key
|
|
|
|
next if value.to_s == hash[key].to_s
|
|
|
|
|
|
|
|
variations[bottle_tag] ||= {}
|
|
|
|
variations[bottle_tag][key] = value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
Homebrew::Override.clear
|
|
|
|
refresh
|
|
|
|
|
|
|
|
hash["variations"] = variations
|
|
|
|
hash
|
|
|
|
end
|
|
|
|
|
2019-03-27 15:59:38 +00:00
|
|
|
private
|
|
|
|
|
|
|
|
def to_h_string_gsubs(string)
|
|
|
|
string.to_s
|
2022-05-30 04:48:54 +01:00
|
|
|
.gsub(Dir.home, "$HOME")
|
2019-03-27 15:59:38 +00:00
|
|
|
.gsub(HOMEBREW_PREFIX, "$(brew --prefix)")
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_h_array_gsubs(array)
|
|
|
|
array.to_a.map do |value|
|
|
|
|
to_h_gsubs(value)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_h_hash_gsubs(hash)
|
2020-02-19 11:18:40 +00:00
|
|
|
hash.to_h.transform_values do |value|
|
|
|
|
to_h_gsubs(value)
|
2019-03-27 15:59:38 +00:00
|
|
|
end
|
|
|
|
rescue TypeError
|
|
|
|
to_h_array_gsubs(hash)
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_h_gsubs(value)
|
|
|
|
if value.respond_to? :to_h
|
|
|
|
to_h_hash_gsubs(value)
|
|
|
|
elsif value.respond_to? :to_a
|
|
|
|
to_h_array_gsubs(value)
|
|
|
|
else
|
|
|
|
to_h_string_gsubs(value)
|
|
|
|
end
|
|
|
|
end
|
2016-08-18 22:11:42 +03:00
|
|
|
end
|
|
|
|
end
|