 45978435e7
			
		
	
	
		45978435e7
		
			
		
	
	
	
	
		
			
			- Previously I thought that comments were fine to discourage people from wasting their time trying to bump things that used `undef` that Sorbet didn't support. But RuboCop is better at this since it'll complain if the comments are unnecessary. - Suggested in https://github.com/Homebrew/brew/pull/18018#issuecomment-2283369501. - I've gone for a mixture of `rubocop:disable` for the files that can't be `typed: strict` (use of undef, required before everything else, etc) and `rubocop:todo` for everything else that should be tried to make strictly typed. There's no functional difference between the two as `rubocop:todo` is `rubocop:disable` with a different name. - And I entirely disabled the cop for the docs/ directory since `typed: strict` isn't going to gain us anything for some Markdown linting config files. - This means that now it's easier to track what needs to be done rather than relying on checklists of files in our big Sorbet issue: ```shell $ git grep 'typed: true # rubocop:todo Sorbet/StrictSigil' | wc -l 268 ``` - And this is confirmed working for new files: ```shell $ git status On branch use-rubocop-for-sorbet-strict-sigils Untracked files: (use "git add <file>..." to include in what will be committed) Library/Homebrew/bad.rb Library/Homebrew/good.rb nothing added to commit but untracked files present (use "git add" to track) $ brew style Offenses: bad.rb:1:1: C: Sorbet/StrictSigil: Sorbet sigil should be at least strict got true. ^^^^^^^^^^^^^ 1340 files inspected, 1 offense detected ```
		
			
				
	
	
		
			536 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			536 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: true # rubocop:todo Sorbet/StrictSigil
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| require "attrable"
 | |
| require "bundle_version"
 | |
| require "cask/cask_loader"
 | |
| require "cask/config"
 | |
| require "cask/dsl"
 | |
| require "cask/metadata"
 | |
| require "cask/tab"
 | |
| require "utils/bottles"
 | |
| require "extend/api_hashable"
 | |
| 
 | |
| module Cask
 | |
|   # An instance of a cask.
 | |
|   class Cask
 | |
|     extend Forwardable
 | |
|     extend Attrable
 | |
|     extend APIHashable
 | |
|     include Metadata
 | |
| 
 | |
|     # The token of this {Cask}.
 | |
|     #
 | |
|     # @api internal
 | |
|     attr_reader :token
 | |
| 
 | |
|     # The configuration of this {Cask}.
 | |
|     #
 | |
|     # @api internal
 | |
|     attr_reader :config
 | |
| 
 | |
|     attr_reader :sourcefile_path, :source, :default_config, :loader
 | |
|     attr_accessor :download, :allow_reassignment
 | |
| 
 | |
|     attr_predicate :loaded_from_api?
 | |
| 
 | |
|     def self.all(eval_all: false)
 | |
|       if !eval_all && !Homebrew::EnvConfig.eval_all?
 | |
|         raise ArgumentError, "Cask::Cask#all cannot be used without `--eval-all` or HOMEBREW_EVAL_ALL"
 | |
|       end
 | |
| 
 | |
|       # Load core casks from tokens so they load from the API when the core cask is not tapped.
 | |
|       tokens_and_files = CoreCaskTap.instance.cask_tokens
 | |
|       tokens_and_files += Tap.reject(&:core_cask_tap?).flat_map(&:cask_files)
 | |
|       tokens_and_files.filter_map do |token_or_file|
 | |
|         CaskLoader.load(token_or_file)
 | |
|       rescue CaskUnreadableError => e
 | |
|         opoo e.message
 | |
| 
 | |
|         nil
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     def tap
 | |
|       return super if block_given? # Object#tap
 | |
| 
 | |
|       @tap
 | |
|     end
 | |
| 
 | |
|     sig {
 | |
|       params(
 | |
|         token:              String,
 | |
|         sourcefile_path:    T.nilable(Pathname),
 | |
|         source:             T.nilable(String),
 | |
|         tap:                T.nilable(Tap),
 | |
|         loaded_from_api:    T::Boolean,
 | |
|         config:             T.nilable(Config),
 | |
|         allow_reassignment: T::Boolean,
 | |
|         loader:             T.nilable(CaskLoader::ILoader),
 | |
|         block:              T.nilable(T.proc.bind(DSL).void),
 | |
|       ).void
 | |
|     }
 | |
|     def initialize(token, sourcefile_path: nil, source: nil, tap: nil, loaded_from_api: false,
 | |
|                    config: nil, allow_reassignment: false, loader: nil, &block)
 | |
|       @token = token
 | |
|       @sourcefile_path = sourcefile_path
 | |
|       @source = source
 | |
|       @tap = tap
 | |
|       @allow_reassignment = allow_reassignment
 | |
|       @loaded_from_api = loaded_from_api
 | |
|       @loader = loader
 | |
|       # Sorbet has trouble with bound procs assigned to instance variables:
 | |
|       # https://github.com/sorbet/sorbet/issues/6843
 | |
|       instance_variable_set(:@block, block)
 | |
| 
 | |
|       @default_config = config || Config.new
 | |
| 
 | |
|       self.config = if config_path.exist?
 | |
|         Config.from_json(File.read(config_path), ignore_invalid_keys: true)
 | |
|       else
 | |
|         @default_config
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     # An old name for the cask.
 | |
|     sig { returns(T::Array[String]) }
 | |
|     def old_tokens
 | |
|       @old_tokens ||= if (tap = self.tap)
 | |
|         Tap.tap_migration_oldnames(tap, token) +
 | |
|           tap.cask_reverse_renames.fetch(token, [])
 | |
|       else
 | |
|         []
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     def config=(config)
 | |
|       @config = config
 | |
| 
 | |
|       refresh
 | |
|     end
 | |
| 
 | |
|     def refresh
 | |
|       @dsl = DSL.new(self)
 | |
|       return unless @block
 | |
| 
 | |
|       @dsl.instance_eval(&@block)
 | |
|       @dsl.language_eval
 | |
|     end
 | |
| 
 | |
|     DSL::DSL_METHODS.each do |method_name|
 | |
|       define_method(method_name) { |&block| @dsl.send(method_name, &block) }
 | |
|     end
 | |
| 
 | |
|     sig { params(caskroom_path: Pathname).returns(T::Array[[String, String]]) }
 | |
|     def timestamped_versions(caskroom_path: self.caskroom_path)
 | |
|       relative_paths = Pathname.glob(metadata_timestamped_path(
 | |
|                                        version: "*", timestamp: "*",
 | |
|                                        caskroom_path:
 | |
|                                      ))
 | |
|                                .map { |p| p.relative_path_from(p.parent.parent) }
 | |
|       # Sorbet is unaware that Pathname is sortable: https://github.com/sorbet/sorbet/issues/6844
 | |
|       T.unsafe(relative_paths).sort_by(&:basename) # sort by timestamp
 | |
|        .map { |p| p.split.map(&:to_s) }
 | |
|     end
 | |
| 
 | |
|     # The fully-qualified token of this {Cask}.
 | |
|     #
 | |
|     # @api internal
 | |
|     def full_token
 | |
|       return token if tap.nil?
 | |
|       return token if tap.core_cask_tap?
 | |
| 
 | |
|       "#{tap.name}/#{token}"
 | |
|     end
 | |
| 
 | |
|     # Alias for {#full_token}.
 | |
|     #
 | |
|     # @api internal
 | |
|     def full_name = full_token
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def installed?
 | |
|       installed_caskfile&.exist? || false
 | |
|     end
 | |
| 
 | |
|     # The caskfile is needed during installation when there are
 | |
|     # `*flight` blocks or the cask has multiple languages
 | |
|     def caskfile_only?
 | |
|       languages.any? || artifacts.any?(Artifact::AbstractFlightBlock)
 | |
|     end
 | |
| 
 | |
|     def uninstall_flight_blocks?
 | |
|       artifacts.any? do |artifact|
 | |
|         case artifact
 | |
|         when Artifact::PreflightBlock
 | |
|           artifact.directives.key?(:uninstall_preflight)
 | |
|         when Artifact::PostflightBlock
 | |
|           artifact.directives.key?(:uninstall_postflight)
 | |
|         end
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.nilable(Time)) }
 | |
|     def install_time
 | |
|       # <caskroom_path>/.metadata/<version>/<timestamp>/Casks/<token>.{rb,json} -> <timestamp>
 | |
|       time = installed_caskfile&.dirname&.dirname&.basename&.to_s
 | |
|       Time.strptime(time, Metadata::TIMESTAMP_FORMAT) if time
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.nilable(Pathname)) }
 | |
|     def installed_caskfile
 | |
|       installed_caskroom_path = caskroom_path
 | |
|       installed_token = token
 | |
| 
 | |
|       # Check if the cask is installed with an old name.
 | |
|       old_tokens.each do |old_token|
 | |
|         old_caskroom_path = Caskroom.path/old_token
 | |
|         next if !old_caskroom_path.directory? || old_caskroom_path.symlink?
 | |
| 
 | |
|         installed_caskroom_path = old_caskroom_path
 | |
|         installed_token = old_token
 | |
|         break
 | |
|       end
 | |
| 
 | |
|       installed_version = timestamped_versions(caskroom_path: installed_caskroom_path).last
 | |
|       return unless installed_version
 | |
| 
 | |
|       caskfile_dir = metadata_main_container_path(caskroom_path: installed_caskroom_path)
 | |
|                      .join(*installed_version, "Casks")
 | |
| 
 | |
|       ["json", "rb"]
 | |
|         .map { |ext| caskfile_dir.join("#{installed_token}.#{ext}") }
 | |
|         .find(&:exist?)
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.nilable(String)) }
 | |
|     def installed_version
 | |
|       return unless (installed_caskfile = self.installed_caskfile)
 | |
| 
 | |
|       # <caskroom_path>/.metadata/<version>/<timestamp>/Casks/<token>.{rb,json} -> <version>
 | |
|       installed_caskfile.dirname.dirname.dirname.basename.to_s
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.nilable(String)) }
 | |
|     def bundle_short_version
 | |
|       bundle_version&.short_version
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.nilable(String)) }
 | |
|     def bundle_long_version
 | |
|       bundle_version&.version
 | |
|     end
 | |
| 
 | |
|     def tab
 | |
|       Tab.for_cask(self)
 | |
|     end
 | |
| 
 | |
|     def config_path
 | |
|       metadata_main_container_path/"config.json"
 | |
|     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
 | |
| 
 | |
|     sig { returns(Pathname) }
 | |
|     def caskroom_path
 | |
|       @caskroom_path ||= Caskroom.path.join(token)
 | |
|     end
 | |
| 
 | |
|     # Check if the installed cask is outdated.
 | |
|     #
 | |
|     # @api internal
 | |
|     def outdated?(greedy: false, greedy_latest: false, greedy_auto_updates: false)
 | |
|       !outdated_version(greedy:, greedy_latest:,
 | |
|                         greedy_auto_updates:).nil?
 | |
|     end
 | |
| 
 | |
|     def outdated_version(greedy: false, greedy_latest: false, greedy_auto_updates: false)
 | |
|       # special case: tap version is not available
 | |
|       return if version.nil?
 | |
| 
 | |
|       if version.latest?
 | |
|         return installed_version if (greedy || greedy_latest) && outdated_download_sha?
 | |
| 
 | |
|         return
 | |
|       elsif auto_updates && !greedy && !greedy_auto_updates
 | |
|         return
 | |
|       end
 | |
| 
 | |
|       # not outdated unless there is a different version on tap
 | |
|       return if installed_version == version
 | |
| 
 | |
|       installed_version
 | |
|     end
 | |
| 
 | |
|     def outdated_info(greedy, verbose, json, greedy_latest, greedy_auto_updates)
 | |
|       return token if !verbose && !json
 | |
| 
 | |
|       installed_version = outdated_version(greedy:, greedy_latest:,
 | |
|                                            greedy_auto_updates:).to_s
 | |
| 
 | |
|       if json
 | |
|         {
 | |
|           name:               token,
 | |
|           installed_versions: [installed_version],
 | |
|           current_version:    version,
 | |
|         }
 | |
|       else
 | |
|         "#{token} (#{installed_version}) != #{version}"
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     def ruby_source_path
 | |
|       return @ruby_source_path if defined?(@ruby_source_path)
 | |
| 
 | |
|       return unless sourcefile_path
 | |
|       return unless tap
 | |
| 
 | |
|       @ruby_source_path = sourcefile_path.relative_path_from(tap.path)
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Hash[Symbol, String]) }
 | |
|     def ruby_source_checksum
 | |
|       @ruby_source_checksum ||= {
 | |
|         sha256: Digest::SHA256.file(sourcefile_path).hexdigest,
 | |
|       }.freeze
 | |
|     end
 | |
| 
 | |
|     def languages
 | |
|       @languages ||= @dsl.languages
 | |
|     end
 | |
| 
 | |
|     def tap_git_head
 | |
|       @tap_git_head ||= tap&.git_head
 | |
|     rescue TapUnavailableError
 | |
|       nil
 | |
|     end
 | |
| 
 | |
|     def populate_from_api!(json_cask)
 | |
|       raise ArgumentError, "Expected cask to be loaded from the API" unless loaded_from_api?
 | |
| 
 | |
|       @languages = json_cask.fetch(:languages, [])
 | |
|       @tap_git_head = json_cask.fetch(:tap_git_head, "HEAD")
 | |
| 
 | |
|       @ruby_source_path = json_cask[:ruby_source_path]
 | |
| 
 | |
|       # TODO: Clean this up when we deprecate the current JSON API and move to the internal JSON v3.
 | |
|       ruby_source_sha256 = json_cask.dig(:ruby_source_checksum, :sha256)
 | |
|       ruby_source_sha256 ||= json_cask[:ruby_source_sha256]
 | |
|       @ruby_source_checksum = { sha256: ruby_source_sha256 }
 | |
|     end
 | |
| 
 | |
|     # @api public
 | |
|     sig { returns(String) }
 | |
|     def to_s = token
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def inspect
 | |
|       "#<Cask #{token}#{sourcefile_path&.to_s&.prepend(" ")}>"
 | |
|     end
 | |
| 
 | |
|     def hash
 | |
|       token.hash
 | |
|     end
 | |
| 
 | |
|     def eql?(other)
 | |
|       instance_of?(other.class) && token == other.token
 | |
|     end
 | |
|     alias == eql?
 | |
| 
 | |
|     def to_h
 | |
|       {
 | |
|         "token"                => token,
 | |
|         "full_token"           => full_name,
 | |
|         "old_tokens"           => old_tokens,
 | |
|         "tap"                  => tap&.name,
 | |
|         "name"                 => name,
 | |
|         "desc"                 => desc,
 | |
|         "homepage"             => homepage,
 | |
|         "url"                  => url,
 | |
|         "url_specs"            => url_specs,
 | |
|         "version"              => version,
 | |
|         "installed"            => installed_version,
 | |
|         "installed_time"       => install_time&.to_i,
 | |
|         "bundle_version"       => bundle_long_version,
 | |
|         "bundle_short_version" => bundle_short_version,
 | |
|         "outdated"             => outdated?,
 | |
|         "sha256"               => sha256,
 | |
|         "artifacts"            => artifacts_list,
 | |
|         "caveats"              => (caveats unless caveats.empty?),
 | |
|         "depends_on"           => depends_on,
 | |
|         "conflicts_with"       => conflicts_with,
 | |
|         "container"            => container&.pairs,
 | |
|         "auto_updates"         => auto_updates,
 | |
|         "deprecated"           => deprecated?,
 | |
|         "deprecation_date"     => deprecation_date,
 | |
|         "deprecation_reason"   => deprecation_reason,
 | |
|         "disabled"             => disabled?,
 | |
|         "disable_date"         => disable_date,
 | |
|         "disable_reason"       => disable_reason,
 | |
|         "tap_git_head"         => tap_git_head,
 | |
|         "languages"            => languages,
 | |
|         "ruby_source_path"     => ruby_source_path,
 | |
|         "ruby_source_checksum" => ruby_source_checksum,
 | |
|       }
 | |
|     end
 | |
| 
 | |
|     def to_internal_api_hash
 | |
|       api_hash = {
 | |
|         "token"              => token,
 | |
|         "name"               => name,
 | |
|         "desc"               => desc,
 | |
|         "homepage"           => homepage,
 | |
|         "url"                => url,
 | |
|         "version"            => version,
 | |
|         "sha256"             => sha256,
 | |
|         "artifacts"          => artifacts_list(compact: true),
 | |
|         "ruby_source_path"   => ruby_source_path,
 | |
|         "ruby_source_sha256" => ruby_source_checksum.fetch(:sha256),
 | |
|       }
 | |
| 
 | |
|       if deprecation_date
 | |
|         api_hash["deprecation_date"] = deprecation_date
 | |
|         api_hash["deprecation_reason"] = deprecation_reason
 | |
|       end
 | |
| 
 | |
|       if disable_date
 | |
|         api_hash["disable_date"] = disable_date
 | |
|         api_hash["disable_reason"] = disable_reason
 | |
|       end
 | |
| 
 | |
|       if (url_specs_hash = url_specs).present?
 | |
|         api_hash["url_specs"] = url_specs_hash
 | |
|       end
 | |
| 
 | |
|       api_hash["caskfile_only"] = true if caskfile_only?
 | |
|       api_hash["conflicts_with"] = conflicts_with if conflicts_with.present?
 | |
|       api_hash["depends_on"] = depends_on if depends_on.present?
 | |
|       api_hash["container"] = container.pairs if container
 | |
|       api_hash["caveats"] = caveats if caveats.present?
 | |
|       api_hash["auto_updates"] = auto_updates if auto_updates
 | |
|       api_hash["languages"] = languages if languages.present?
 | |
| 
 | |
|       api_hash
 | |
|     end
 | |
| 
 | |
|     HASH_KEYS_TO_SKIP = %w[outdated installed versions].freeze
 | |
|     private_constant :HASH_KEYS_TO_SKIP
 | |
| 
 | |
|     def to_hash_with_variations(hash_method: :to_h)
 | |
|       case hash_method
 | |
|       when :to_h
 | |
|         if loaded_from_api? && !Homebrew::EnvConfig.no_install_from_api?
 | |
|           return api_to_local_hash(Homebrew::API::Cask.all_casks[token].dup)
 | |
|         end
 | |
|       when :to_internal_api_hash
 | |
|         raise ArgumentError, "API Hash must be generated from Ruby source files" if loaded_from_api?
 | |
|       else
 | |
|         raise ArgumentError, "Unknown hash method #{hash_method.inspect}"
 | |
|       end
 | |
| 
 | |
|       hash = public_send(hash_method)
 | |
|       variations = {}
 | |
| 
 | |
|       if @dsl.on_system_blocks_exist?
 | |
|         begin
 | |
|           MacOSVersion::SYMBOLS.keys.product(OnSystem::ARCH_OPTIONS).each do |os, arch|
 | |
|             bottle_tag = ::Utils::Bottles::Tag.new(system: os, arch:)
 | |
|             next unless bottle_tag.valid_combination?
 | |
|             next if depends_on.macos &&
 | |
|                     !@dsl.depends_on_set_in_block? &&
 | |
|                     !depends_on.macos.allows?(bottle_tag.to_macos_version)
 | |
| 
 | |
|             Homebrew::SimulateSystem.with(os:, arch:) do
 | |
|               refresh
 | |
| 
 | |
|               public_send(hash_method).each do |key, value|
 | |
|                 next if HASH_KEYS_TO_SKIP.include? key
 | |
|                 next if value.to_s == hash[key].to_s
 | |
| 
 | |
|                 variations[bottle_tag.to_sym] ||= {}
 | |
|                 variations[bottle_tag.to_sym][key] = value
 | |
|               end
 | |
|             end
 | |
|           end
 | |
|         ensure
 | |
|           refresh
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       hash["variations"] = variations if hash_method != :to_internal_api_hash || variations.present?
 | |
|       hash
 | |
|     end
 | |
| 
 | |
|     def artifacts_list(compact: false, uninstall_only: false)
 | |
|       artifacts.filter_map do |artifact|
 | |
|         case artifact
 | |
|         when Artifact::AbstractFlightBlock
 | |
|           uninstall_flight_block = artifact.directives.key?(:uninstall_preflight) ||
 | |
|                                    artifact.directives.key?(:uninstall_postflight)
 | |
|           next if uninstall_only && !uninstall_flight_block
 | |
| 
 | |
|           # Only indicate whether this block is used as we don't load it from the API
 | |
|           # We can skip this entirely once we move to internal JSON v3.
 | |
|           { artifact.summarize.to_sym => nil } unless compact
 | |
|         else
 | |
|           zap_artifact = artifact.is_a?(Artifact::Zap)
 | |
|           uninstall_artifact = artifact.respond_to?(:uninstall_phase) || artifact.respond_to?(:post_uninstall_phase)
 | |
|           next if uninstall_only && !zap_artifact && !uninstall_artifact
 | |
| 
 | |
|           { artifact.class.dsl_key => artifact.to_args }
 | |
|         end
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     private
 | |
| 
 | |
|     sig { returns(T.nilable(Homebrew::BundleVersion)) }
 | |
|     def bundle_version
 | |
|       @bundle_version ||= if (bundle = artifacts.find { |a| a.is_a?(Artifact::App) }&.target) &&
 | |
|                              (plist = Pathname("#{bundle}/Contents/Info.plist")) && plist.exist?
 | |
|         Homebrew::BundleVersion.from_info_plist(plist)
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     def api_to_local_hash(hash)
 | |
|       hash["token"] = token
 | |
|       hash["installed"] = installed_version
 | |
|       hash["outdated"] = outdated?
 | |
|       hash
 | |
|     end
 | |
| 
 | |
|     def url_specs
 | |
|       url&.specs.dup.tap do |url_specs|
 | |
|         case url_specs&.dig(:user_agent)
 | |
|         when :default
 | |
|           url_specs.delete(:user_agent)
 | |
|         when Symbol
 | |
|           url_specs[:user_agent] = ":#{url_specs[:user_agent]}"
 | |
|         end
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| end
 |