328 lines
8.4 KiB
Ruby
Raw Normal View History

2020-10-10 14:16:11 +02:00
# typed: false
# frozen_string_literal: true
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
include Metadata
2016-08-18 22:11:42 +03:00
attr_reader :token, :sourcefile_path, :source, :config, :default_config
2017-07-29 16:27:54 +02:00
attr_accessor :download, :allow_reassignment
def self.all
Tap.flat_map(&:cask_files).map do |f|
CaskLoader::FromTapPathLoader.new(f).load(config: nil)
rescue CaskUnreadableError => e
opoo e.message
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
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
@source = source
2017-07-29 16:27:54 +02:00
@tap = tap
@allow_reassignment = allow_reassignment
@block = block
2020-09-29 23:46:30 +02:00
@default_config = config || Config.new
self.config = if config_path.exist?
Config.from_json(File.read(config_path), ignore_invalid_keys: true)
2020-09-29 23:46:30 +02:00
else
@default_config
end
end
def config=(config)
@config = config
2022-06-23 17:19:27 -04:00
refresh
end
def refresh
@dsl = DSL.new(self)
return unless @block
2018-09-17 02:45:00 +02: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|
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
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
.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
def os_versions
2022-08-08 19:58:50 +09:00
# TODO: use #to_hash_with_variations instead once all casks use on_system blocks
@os_versions ||= begin
version_os_hash = {}
actual_version = MacOS.full_version.to_s
2022-06-29 11:29:46 -04:00
MacOSVersions::SYMBOLS.each do |os_name, os_version|
MacOS.full_version = os_version
2022-08-07 15:43:51 +09:00
cask = CaskLoader.load(full_name)
version_os_hash[os_name] = cask.version if cask.version != version
end
version_os_hash
ensure
MacOS.full_version = actual_version
end
end
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}"
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
def installed_caskfile
installed_version = timestamped_versions.last
metadata_main_container_path.join(*installed_version, "Casks", "#{token}.rb")
end
2019-02-02 17:11:37 +01:00
def config_path
metadata_main_container_path/"config.json"
2019-02-02 17:11:37 +01:00
end
def checksumable?
DownloadStrategyDetector.detect(url.to_s, url.using) <= AbstractFileDownloadStrategy
end
def download_sha_path
metadata_main_container_path/"LATEST_DOWNLOAD_SHA256"
end
def new_download_sha
require "cask/installer"
# Call checksumable? before hashing
@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?
return true unless checksumable?
current_download_sha = download_sha_path.read if download_sha_path.exist?
current_download_sha.blank? || current_download_sha != new_download_sha
end
def caskroom_path
@caskroom_path ||= Caskroom.path.join(token)
end
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
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-24 22:38:30 +10:00
return []
elsif auto_updates && !greedy && !greedy_auto_updates
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
def outdated_info(greedy, verbose, json, greedy_latest, greedy_auto_updates)
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
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,
}
else
2020-04-28 12:21:51 +08:00
"#{token} (#{installed_versions}) != #{version}"
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?
def to_h
2018-07-17 10:04:17 +01:00
{
2019-07-09 20:43:45 +02:00
"token" => token,
"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,
"versions" => os_versions,
"installed" => versions.last,
"outdated" => outdated?,
2018-11-02 17:18:07 +00:00
"sha256" => sha256,
2022-08-25 02:52:40 -04:00
"artifacts" => artifacts_list,
"caveats" => (to_h_string_gsubs(caveats) unless caveats.empty?),
2018-11-02 17:18:07 +00:00
"depends_on" => depends_on,
"conflicts_with" => conflicts_with,
2018-11-02 17:18:07 +00:00
"container" => container,
"auto_updates" => auto_updates,
}
end
def to_hash_with_variations
hash = to_h
2022-06-23 17:19:27 -04:00
variations = {}
hash_keys_to_skip = %w[outdated installed versions]
if @dsl.on_system_blocks_exist?
[:arm, :intel].each do |arch|
2022-06-29 11:29:46 -04:00
MacOSVersions::SYMBOLS.each_key do |os_name|
2022-07-21 15:32:51 +02:00
bottle_tag = ::Utils::Bottles::Tag.new(system: os_name, arch: arch)
next unless bottle_tag.valid_combination?
2022-06-23 17:19:27 -04:00
Homebrew::SimulateSystem.os = os_name
Homebrew::SimulateSystem.arch = arch
2022-06-23 17:19:27 -04:00
refresh
to_h.each do |key, value|
2022-06-23 17:19:27 -04:00
next if hash_keys_to_skip.include? key
next if value.to_s == hash[key].to_s
2022-07-21 15:32:51 +02:00
variations[bottle_tag.to_sym] ||= {}
variations[bottle_tag.to_sym][key] = value
2022-06-23 17:19:27 -04:00
end
end
end
end
Homebrew::SimulateSystem.clear
2022-06-23 17:19:27 -04:00
refresh
hash["variations"] = variations
hash
end
private
2022-08-25 02:52:40 -04:00
def artifacts_list
artifacts.map do |artifact|
if artifact.is_a? Artifact::AbstractFlightBlock
{ type: artifact.summarize }
else
{
type: artifact.class.dsl_key,
args: to_h_gsubs(artifact.to_args),
}
end
end
end
def to_h_string_gsubs(string)
string.to_s
2022-05-30 04:48:54 +01:00
.gsub(Dir.home, "$HOME")
.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)
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