1067 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			1067 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: true
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| require "attrable"
 | |
| require "cask/denylist"
 | |
| require "cask/download"
 | |
| require "digest"
 | |
| require "livecheck/livecheck"
 | |
| require "source_location"
 | |
| require "system_command"
 | |
| require "utils/backtrace"
 | |
| require "utils/curl"
 | |
| require "utils/git"
 | |
| require "utils/shared_audits"
 | |
| 
 | |
| module Cask
 | |
|   # Audit a cask for various problems.
 | |
|   class Audit
 | |
|     include SystemCommand::Mixin
 | |
|     include ::Utils::Curl
 | |
|     extend Attrable
 | |
| 
 | |
|     attr_reader :cask, :download
 | |
| 
 | |
|     attr_predicate :new_cask?, :strict?, :signing?, :online?, :token_conflicts?
 | |
| 
 | |
|     def initialize(
 | |
|       cask,
 | |
|       download: nil, quarantine: nil,
 | |
|       token_conflicts: nil, online: nil, strict: nil, signing: nil,
 | |
|       new_cask: nil, only: [], except: []
 | |
|     )
 | |
|       # `new_cask` implies `online`, `token_conflicts`, `strict` and `signing`
 | |
|       online = new_cask if online.nil?
 | |
|       strict = new_cask if strict.nil?
 | |
|       signing = new_cask if signing.nil?
 | |
|       token_conflicts = new_cask if token_conflicts.nil?
 | |
| 
 | |
|       # `online` and `signing` imply `download`
 | |
|       download = online || signing if download.nil?
 | |
| 
 | |
|       @cask = cask
 | |
|       @download = Download.new(cask, quarantine:) if download
 | |
|       @online = online
 | |
|       @strict = strict
 | |
|       @signing = signing
 | |
|       @new_cask = new_cask
 | |
|       @token_conflicts = token_conflicts
 | |
|       @only = only || []
 | |
|       @except = except || []
 | |
|     end
 | |
| 
 | |
|     def run!
 | |
|       only_audits = @only
 | |
|       except_audits = @except
 | |
| 
 | |
|       private_methods.map(&:to_s).grep(/^audit_/).each do |audit_method_name|
 | |
|         name = audit_method_name.delete_prefix("audit_")
 | |
|         next if !only_audits.empty? && only_audits&.exclude?(name)
 | |
|         next if except_audits&.include?(name)
 | |
| 
 | |
|         send(audit_method_name)
 | |
|       end
 | |
| 
 | |
|       self
 | |
|     rescue => e
 | |
|       odebug e, ::Utils::Backtrace.clean(e)
 | |
|       add_error "exception while auditing #{cask}: #{e.message}"
 | |
|       self
 | |
|     end
 | |
| 
 | |
|     def errors
 | |
|       @errors ||= []
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def errors?
 | |
|       errors.any?
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def success?
 | |
|       !errors?
 | |
|     end
 | |
| 
 | |
|     sig {
 | |
|       params(
 | |
|         message:     T.nilable(String),
 | |
|         location:    T.nilable(Homebrew::SourceLocation),
 | |
|         strict_only: T::Boolean,
 | |
|       ).void
 | |
|     }
 | |
|     def add_error(message, location: nil, strict_only: false)
 | |
|       # Only raise non-critical audits if the user specified `--strict`.
 | |
|       return if strict_only && !@strict
 | |
| 
 | |
|       errors << ({ message:, location:, corrected: false })
 | |
|     end
 | |
| 
 | |
|     def result
 | |
|       Formatter.error("failed") if errors?
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.nilable(String)) }
 | |
|     def summary
 | |
|       return if success?
 | |
| 
 | |
|       summary = ["audit for #{cask}: #{result}"]
 | |
| 
 | |
|       errors.each do |error|
 | |
|         summary << " #{Formatter.error("-")} #{error[:message]}"
 | |
|       end
 | |
| 
 | |
|       summary.join("\n")
 | |
|     end
 | |
| 
 | |
|     private
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_untrusted_pkg
 | |
|       odebug "Auditing pkg stanza: allow_untrusted"
 | |
| 
 | |
|       return if @cask.sourcefile_path.nil?
 | |
| 
 | |
|       tap = @cask.tap
 | |
|       return if tap.nil?
 | |
|       return if tap.user != "Homebrew"
 | |
| 
 | |
|       return if cask.artifacts.none? { |k| k.is_a?(Artifact::Pkg) && k.stanza_options.key?(:allow_untrusted) }
 | |
| 
 | |
|       add_error "allow_untrusted is not permitted in official Homebrew Cask taps"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_stanza_requires_uninstall
 | |
|       odebug "Auditing stanzas which require an uninstall"
 | |
| 
 | |
|       return if cask.artifacts.none? { |k| k.is_a?(Artifact::Pkg) || k.is_a?(Artifact::Installer) }
 | |
|       return if cask.artifacts.any?(Artifact::Uninstall)
 | |
| 
 | |
|       add_error "installer and pkg stanzas require an uninstall stanza"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_single_pre_postflight
 | |
|       odebug "Auditing preflight and postflight stanzas"
 | |
| 
 | |
|       if cask.artifacts.count { |k| k.is_a?(Artifact::PreflightBlock) && k.directives.key?(:preflight) } > 1
 | |
|         add_error "only a single preflight stanza is allowed"
 | |
|       end
 | |
| 
 | |
|       count = cask.artifacts.count do |k|
 | |
|         k.is_a?(Artifact::PostflightBlock) &&
 | |
|           k.directives.key?(:postflight)
 | |
|       end
 | |
|       return if count <= 1
 | |
| 
 | |
|       add_error "only a single postflight stanza is allowed"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_single_uninstall_zap
 | |
|       odebug "Auditing single uninstall_* and zap stanzas"
 | |
| 
 | |
|       count = cask.artifacts.count do |k|
 | |
|         k.is_a?(Artifact::PreflightBlock) &&
 | |
|           k.directives.key?(:uninstall_preflight)
 | |
|       end
 | |
| 
 | |
|       add_error "only a single uninstall_preflight stanza is allowed" if count > 1
 | |
| 
 | |
|       count = cask.artifacts.count do |k|
 | |
|         k.is_a?(Artifact::PostflightBlock) &&
 | |
|           k.directives.key?(:uninstall_postflight)
 | |
|       end
 | |
| 
 | |
|       add_error "only a single uninstall_postflight stanza is allowed" if count > 1
 | |
| 
 | |
|       return if cask.artifacts.count { |k| k.is_a?(Artifact::Zap) } <= 1
 | |
| 
 | |
|       add_error "only a single zap stanza is allowed"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_required_stanzas
 | |
|       odebug "Auditing required stanzas"
 | |
|       [:version, :sha256, :url, :homepage].each do |sym|
 | |
|         add_error "a #{sym} stanza is required" unless cask.send(sym)
 | |
|       end
 | |
|       add_error "at least one name stanza is required" if cask.name.empty?
 | |
|       # TODO: specific DSL knowledge should not be spread around in various files like this
 | |
|       rejected_artifacts = [:uninstall, :zap]
 | |
|       installable_artifacts = cask.artifacts.reject { |k| rejected_artifacts.include?(k) }
 | |
|       add_error "at least one activatable artifact stanza is required" if installable_artifacts.empty?
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_description
 | |
|       # Fonts seldom benefit from descriptions and requiring them disproportionately
 | |
|       # increases the maintenance burden.
 | |
|       return if cask.tap == "homebrew/cask" && cask.token.include?("font-")
 | |
| 
 | |
|       add_error("Cask should have a description. Please add a `desc` stanza.", strict_only: true) if cask.desc.blank?
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_version_special_characters
 | |
|       return unless cask.version
 | |
| 
 | |
|       return if cask.version.latest?
 | |
| 
 | |
|       raw_version = cask.version.raw_version
 | |
|       return if raw_version.exclude?(":") && raw_version.exclude?("/")
 | |
| 
 | |
|       add_error "version should not contain colons or slashes"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_no_string_version_latest
 | |
|       return unless cask.version
 | |
| 
 | |
|       odebug "Auditing version :latest does not appear as a string ('latest')"
 | |
|       return if cask.version.raw_version != "latest"
 | |
| 
 | |
|       add_error "you should use version :latest instead of version 'latest'"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_sha256_no_check_if_latest
 | |
|       return unless cask.sha256
 | |
|       return unless cask.version
 | |
| 
 | |
|       odebug "Auditing sha256 :no_check with version :latest"
 | |
|       return unless cask.version.latest?
 | |
|       return if cask.sha256 == :no_check
 | |
| 
 | |
|       add_error "you should use sha256 :no_check when version is :latest"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_sha256_no_check_if_unversioned
 | |
|       return unless cask.sha256
 | |
|       return if cask.sha256 == :no_check
 | |
| 
 | |
|       return unless cask.url&.unversioned?
 | |
| 
 | |
|       add_error "Use `sha256 :no_check` when URL is unversioned."
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_sha256_actually_256
 | |
|       return unless cask.sha256
 | |
| 
 | |
|       odebug "Auditing sha256 string is a legal SHA-256 digest"
 | |
|       return unless cask.sha256.is_a?(Checksum)
 | |
|       return if cask.sha256.length == 64 && cask.sha256[/^[0-9a-f]+$/i]
 | |
| 
 | |
|       add_error "sha256 string must be of 64 hexadecimal characters"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_sha256_invalid
 | |
|       return unless cask.sha256
 | |
| 
 | |
|       odebug "Auditing sha256 is not a known invalid value"
 | |
|       empty_sha256 = "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855"
 | |
|       return if cask.sha256 != empty_sha256
 | |
| 
 | |
|       add_error "cannot use the sha256 for an empty string: #{empty_sha256}"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_latest_with_livecheck
 | |
|       return unless cask.version&.latest?
 | |
|       return unless cask.livecheckable?
 | |
|       return if cask.livecheck.skip?
 | |
| 
 | |
|       add_error "Casks with a `livecheck` should not use `version :latest`."
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_latest_with_auto_updates
 | |
|       return unless cask.version&.latest?
 | |
|       return unless cask.auto_updates
 | |
| 
 | |
|       add_error "Casks with `version :latest` should not use `auto_updates`."
 | |
|     end
 | |
| 
 | |
|     LIVECHECK_REFERENCE_URL = "https://docs.brew.sh/Cask-Cookbook#stanza-livecheck"
 | |
| 
 | |
|     sig { params(livecheck_result: T.any(NilClass, T::Boolean, Symbol)).void }
 | |
|     def audit_hosting_with_livecheck(livecheck_result: audit_livecheck_version)
 | |
|       return if cask.deprecated? || cask.disabled?
 | |
|       return if cask.version&.latest?
 | |
|       return unless cask.url
 | |
|       return if block_url_offline?
 | |
|       return if cask.livecheckable?
 | |
|       return if livecheck_result == :auto_detected
 | |
| 
 | |
|       add_livecheck = "please add a livecheck. See #{Formatter.url(LIVECHECK_REFERENCE_URL)}"
 | |
| 
 | |
|       case cask.url.to_s
 | |
|       when %r{sourceforge.net/(\S+)}
 | |
|         return unless online?
 | |
| 
 | |
|         add_error "Download is hosted on SourceForge, #{add_livecheck}", location: cask.url.location
 | |
|       when %r{dl.devmate.com/(\S+)}
 | |
|         add_error "Download is hosted on DevMate, #{add_livecheck}", location: cask.url.location
 | |
|       when %r{rink.hockeyapp.net/(\S+)}
 | |
|         add_error "Download is hosted on HockeyApp, #{add_livecheck}", location: cask.url.location
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     SOURCEFORGE_OSDN_REFERENCE_URL = "https://docs.brew.sh/Cask-Cookbook#sourceforgeosdn-urls"
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_download_url_format
 | |
|       return unless cask.url
 | |
|       return if block_url_offline?
 | |
| 
 | |
|       odebug "Auditing URL format"
 | |
|       return unless bad_sourceforge_url?
 | |
| 
 | |
|       add_error "SourceForge URL format incorrect. See #{Formatter.url(SOURCEFORGE_OSDN_REFERENCE_URL)}",
 | |
|                 location: cask.url.location
 | |
|     end
 | |
| 
 | |
|     def audit_download_url_is_osdn
 | |
|       return unless cask.url
 | |
|       return if block_url_offline?
 | |
|       return unless bad_osdn_url?
 | |
| 
 | |
|       add_error "OSDN download urls are disabled.", location: cask.url.location, strict_only: true
 | |
|     end
 | |
| 
 | |
|     VERIFIED_URL_REFERENCE_URL = "https://docs.brew.sh/Cask-Cookbook#when-url-and-homepage-domains-differ-add-verified"
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_unnecessary_verified
 | |
|       return unless cask.url
 | |
|       return if block_url_offline?
 | |
|       return unless verified_present?
 | |
|       return unless url_match_homepage?
 | |
|       return unless verified_matches_url?
 | |
| 
 | |
|       add_error "The URL's domain #{Formatter.url(domain)} matches the homepage domain " \
 | |
|                 "#{Formatter.url(homepage)}, the 'verified' parameter of the 'url' stanza is unnecessary. " \
 | |
|                 "See #{Formatter.url(VERIFIED_URL_REFERENCE_URL)}"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_missing_verified
 | |
|       return unless cask.url
 | |
|       return if block_url_offline?
 | |
|       return if file_url?
 | |
|       return if url_match_homepage?
 | |
|       return if verified_present?
 | |
| 
 | |
|       add_error "The URL's domain #{Formatter.url(domain)} does not match the homepage domain " \
 | |
|                 "#{Formatter.url(homepage)}, a 'verified' parameter has to be added to the 'url' stanza. " \
 | |
|                 "See #{Formatter.url(VERIFIED_URL_REFERENCE_URL)}"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_no_match
 | |
|       return unless cask.url
 | |
|       return if block_url_offline?
 | |
|       return unless verified_present?
 | |
|       return if verified_matches_url?
 | |
| 
 | |
|       add_error "Verified URL #{Formatter.url(url_from_verified)} does not match URL " \
 | |
|                 "#{Formatter.url(strip_url_scheme(cask.url.to_s))}. " \
 | |
|                 "See #{Formatter.url(VERIFIED_URL_REFERENCE_URL)}",
 | |
|                 location: cask.url.location
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_generic_artifacts
 | |
|       cask.artifacts.select { |a| a.is_a?(Artifact::Artifact) }.each do |artifact|
 | |
|         unless artifact.target.absolute?
 | |
|           add_error "target must be absolute path for #{artifact.class.english_name} #{artifact.source}"
 | |
|         end
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_languages
 | |
|       @cask.languages.each do |language|
 | |
|         Locale.parse(language)
 | |
|       rescue Locale::ParserError
 | |
|         add_error "Locale '#{language}' is invalid."
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_token_conflicts
 | |
|       return unless token_conflicts?
 | |
| 
 | |
|       Homebrew.with_no_api_env do
 | |
|         return unless core_formula_names.include?(cask.token)
 | |
| 
 | |
|         add_error(
 | |
|           "possible duplicate, cask token conflicts with Homebrew core formula: #{Formatter.url(core_formula_url)}",
 | |
|           strict_only: true,
 | |
|         )
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_token_valid
 | |
|       add_error "cask token contains non-ascii characters" unless cask.token.ascii_only?
 | |
|       add_error "cask token + should be replaced by -plus-" if cask.token.include? "+"
 | |
|       add_error "cask token whitespace should be replaced by hyphens" if cask.token.include? " "
 | |
|       add_error "cask token underscores should be replaced by hyphens" if cask.token.include? "_"
 | |
|       add_error "cask token should not contain double hyphens" if cask.token.include? "--"
 | |
| 
 | |
|       if cask.token.match?(/[^@a-z0-9-]/)
 | |
|         add_error "cask token should only contain lowercase alphanumeric characters, hyphens and @"
 | |
|       end
 | |
| 
 | |
|       if cask.token.start_with?("-", "@") || cask.token.end_with?("-", "@")
 | |
|         add_error "cask token should not have leading or trailing hyphens and/or @"
 | |
|       end
 | |
| 
 | |
|       add_error "cask token @ unrelated to versioning should be replaced by -at-" if cask.token.count("@") > 1
 | |
|       add_error "cask token should not contain a hyphen followed by @" if cask.token.include? "-@"
 | |
|       add_error "cask token should not contain @ followed by a hyphen" if cask.token.include? "@-"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_token_bad_words
 | |
|       return unless new_cask?
 | |
| 
 | |
|       token = cask.token
 | |
| 
 | |
|       add_error "cask token contains .app" if token.end_with? ".app"
 | |
| 
 | |
|       match_data = /-(?<designation>alpha|beta|rc|release-candidate)$/.match(cask.token)
 | |
|       if match_data && cask.tap&.official?
 | |
|         add_error "cask token contains version designation '#{match_data[:designation]}'"
 | |
|       end
 | |
| 
 | |
|       add_error("cask token mentions launcher", strict_only: true) if token.end_with? "launcher"
 | |
| 
 | |
|       add_error("cask token mentions desktop", strict_only: true) if token.end_with? "desktop"
 | |
| 
 | |
|       add_error("cask token mentions platform", strict_only: true) if token.end_with? "mac", "osx", "macos"
 | |
| 
 | |
|       add_error("cask token mentions architecture", strict_only: true) if token.end_with? "x86", "32_bit", "x86_64",
 | |
|                                                                                           "64_bit"
 | |
| 
 | |
|       frameworks = %w[cocoa qt gtk wx java]
 | |
|       return if frameworks.include?(token) || !token.end_with?(*frameworks)
 | |
| 
 | |
|       add_error("cask token mentions framework", strict_only: true)
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_download
 | |
|       return if download.blank? || cask.url.blank?
 | |
| 
 | |
|       begin
 | |
|         download.fetch
 | |
|       rescue => e
 | |
|         add_error "download not possible: #{e}", location: cask.url.location
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_livecheck_unneeded_long_version
 | |
|       return if cask.version.nil? || cask.url.nil?
 | |
|       return if cask.livecheck.strategy != :sparkle
 | |
|       return unless cask.version.csv.second
 | |
|       return if cask.url.to_s.include? cask.version.csv.second
 | |
|       return if cask.version.csv.third.present? && cask.url.to_s.include?(cask.version.csv.third)
 | |
| 
 | |
|       add_error "Download does not require additional version components. Use `&:short_version` in the livecheck",
 | |
|                 location:    cask.url.location,
 | |
|                 strict_only: true
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_signing
 | |
|       return if !signing? || download.blank? || cask.url.blank?
 | |
| 
 | |
|       odebug "Auditing signing"
 | |
| 
 | |
|       extract_artifacts do |artifacts, tmpdir|
 | |
|         is_container = artifacts.any? { |a| a.is_a?(Artifact::App) || a.is_a?(Artifact::Pkg) }
 | |
| 
 | |
|         artifacts.each do |artifact|
 | |
|           next if artifact.is_a?(Artifact::Binary) && is_container == true
 | |
| 
 | |
|           artifact_path = artifact.is_a?(Artifact::Pkg) ? artifact.path : artifact.source
 | |
| 
 | |
|           path = tmpdir/artifact_path.relative_path_from(cask.staged_path)
 | |
| 
 | |
|           result = case artifact
 | |
|           when Artifact::Pkg
 | |
|             system_command("spctl", args: ["--assess", "--type", "install", path], print_stderr: false)
 | |
|           when Artifact::App
 | |
|             system_command("spctl", args: ["--assess", "--type", "execute", path], print_stderr: false)
 | |
|           when Artifact::Binary
 | |
|             system_command("codesign",  args: ["--verify", path], print_stderr: false)
 | |
|           else
 | |
|             add_error "Unknown artifact type: #{artifact.class}", location: cask.url.location
 | |
|           end
 | |
| 
 | |
|           next if result.success?
 | |
| 
 | |
|           add_error <<~EOS, location: cask.url.location, strict_only: true
 | |
|             Signature verification failed:
 | |
|             #{result.merged_output}
 | |
|             macOS on ARM requires software to be signed.
 | |
|             Please contact the upstream developer to let them know they should sign and notarize their software.
 | |
|           EOS
 | |
|         end
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def extract_artifacts
 | |
|       return unless online?
 | |
| 
 | |
|       artifacts = cask.artifacts.select do |artifact|
 | |
|         artifact.is_a?(Artifact::Pkg) || artifact.is_a?(Artifact::App) || artifact.is_a?(Artifact::Binary)
 | |
|       end
 | |
| 
 | |
|       if @artifacts_extracted && @tmpdir
 | |
|         yield artifacts, @tmpdir if block_given?
 | |
|         return
 | |
|       end
 | |
| 
 | |
|       return if artifacts.empty?
 | |
| 
 | |
|       @tmpdir ||= Pathname(Dir.mktmpdir("cask-audit", HOMEBREW_TEMP))
 | |
| 
 | |
|       # Clean up tmp dir when @tmpdir object is destroyed
 | |
|       ObjectSpace.define_finalizer(
 | |
|         @tmpdir,
 | |
|         proc { FileUtils.remove_entry(@tmpdir) },
 | |
|       )
 | |
| 
 | |
|       ohai "Downloading and extracting artifacts"
 | |
| 
 | |
|       downloaded_path = download.fetch
 | |
| 
 | |
|       primary_container = UnpackStrategy.detect(downloaded_path, type: @cask.container&.type, merge_xattrs: true)
 | |
|       return if primary_container.nil?
 | |
| 
 | |
|       # Extract the container to the temporary directory.
 | |
|       primary_container.extract_nestedly(to: @tmpdir, basename: downloaded_path.basename, verbose: false)
 | |
| 
 | |
|       if (nested_container = @cask.container&.nested)
 | |
|         FileUtils.chmod_R "+rw", @tmpdir/nested_container, force: true, verbose: false
 | |
|         UnpackStrategy.detect(@tmpdir/nested_container, merge_xattrs: true)
 | |
|                       .extract_nestedly(to: @tmpdir, verbose: false)
 | |
|       end
 | |
| 
 | |
|       @artifacts_extracted = true # Set the flag to indicate that extraction has occurred.
 | |
| 
 | |
|       # Yield the artifacts and temp directory to the block if provided.
 | |
|       yield artifacts, @tmpdir if block_given?
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_rosetta
 | |
|       return unless online?
 | |
|       return if Homebrew::SimulateSystem.current_arch != :arm
 | |
| 
 | |
|       odebug "Auditing Rosetta 2 requirement"
 | |
| 
 | |
|       extract_artifacts do |artifacts, tmpdir|
 | |
|         is_container = artifacts.any? { |a| a.is_a?(Artifact::App) || a.is_a?(Artifact::Pkg) }
 | |
| 
 | |
|         artifacts.filter { |a| a.is_a?(Artifact::App) || a.is_a?(Artifact::Binary) }
 | |
|                  .each do |artifact|
 | |
|           next if artifact.is_a?(Artifact::Binary) && is_container
 | |
| 
 | |
|           path = tmpdir/artifact.source.relative_path_from(cask.staged_path)
 | |
| 
 | |
|           result = case artifact
 | |
|           when Artifact::App
 | |
|             files = Dir[path/"Contents/MacOS/*"].select do |f|
 | |
|               File.executable?(f) && !File.directory?(f) && !f.end_with?(".dylib")
 | |
|             end
 | |
|             add_error "No binaries in App: #{artifact.source}", location: cask.url.location if files.empty?
 | |
|             system_command("lipo", args: ["-archs", files.first], print_stderr: false)
 | |
|           when Artifact::Binary
 | |
|             binary_path = path.to_s.gsub(cask.appdir, tmpdir)
 | |
|             system_command("lipo", args: ["-archs", binary_path], print_stderr: true)
 | |
|           else
 | |
|             add_error "Unknown artifact type: #{artifact.class}", location: cask.url.location
 | |
|           end
 | |
| 
 | |
|           # binary stanza can contain shell scripts, so we just continue if lipo fails.
 | |
|           next unless result.success?
 | |
| 
 | |
|           odebug result.merged_output
 | |
| 
 | |
|           unless /arm64|x86_64/.match?(result.merged_output)
 | |
|             add_error "Artifacts architecture is no longer supported by macOS!",
 | |
|                       location: cask.url.location
 | |
|             next
 | |
|           end
 | |
| 
 | |
|           supports_arm = result.merged_output.include?("arm64")
 | |
|           mentions_rosetta = cask.caveats.include?("requires Rosetta 2")
 | |
| 
 | |
|           if supports_arm && mentions_rosetta
 | |
|             add_error "Artifacts does not require Rosetta 2 but the caveats say otherwise!",
 | |
|                       location: cask.url.location
 | |
|           elsif !supports_arm && !mentions_rosetta
 | |
|             add_error "Artifacts require Rosetta 2 but this is not indicated by the caveats!",
 | |
|                       location: cask.url.location
 | |
|           end
 | |
|         end
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.any(NilClass, T::Boolean, Symbol)) }
 | |
|     def audit_livecheck_version
 | |
|       return unless online?
 | |
|       return unless cask.version
 | |
| 
 | |
|       referenced_cask, = Homebrew::Livecheck.resolve_livecheck_reference(cask)
 | |
| 
 | |
|       # Respect skip conditions for a referenced cask
 | |
|       if referenced_cask
 | |
|         skip_info = Homebrew::Livecheck::SkipConditions.referenced_skip_information(
 | |
|           referenced_cask,
 | |
|           Homebrew::Livecheck.cask_name(cask),
 | |
|         )
 | |
|       end
 | |
| 
 | |
|       # Respect cask skip conditions (e.g. deprecated, disabled, latest, unversioned)
 | |
|       skip_info ||= Homebrew::Livecheck::SkipConditions.skip_information(cask)
 | |
|       return :skip if skip_info.present?
 | |
| 
 | |
|       latest_version = Homebrew::Livecheck.latest_version(
 | |
|         cask,
 | |
|         referenced_formula_or_cask: referenced_cask,
 | |
|       )&.fetch(:latest)
 | |
| 
 | |
|       return :auto_detected if cask.version.to_s == latest_version.to_s
 | |
| 
 | |
|       add_error "Version '#{cask.version}' differs from '#{latest_version}' retrieved by livecheck."
 | |
| 
 | |
|       false
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_min_os
 | |
|       return unless online?
 | |
|       return unless strict?
 | |
| 
 | |
|       odebug "Auditing minimum OS version"
 | |
| 
 | |
|       plist_min_os = cask_plist_min_os
 | |
|       sparkle_min_os = livecheck_min_os
 | |
| 
 | |
|       debug_messages = []
 | |
|       debug_messages << "Plist #{plist_min_os}" if plist_min_os
 | |
|       debug_messages << "Sparkle #{sparkle_min_os}" if sparkle_min_os
 | |
|       odebug "Detected minimum OS version: #{debug_messages.join(" | ")}" unless debug_messages.empty?
 | |
|       min_os = [plist_min_os, sparkle_min_os].compact.max
 | |
| 
 | |
|       return if min_os.nil? || min_os <= HOMEBREW_MACOS_OLDEST_ALLOWED
 | |
| 
 | |
|       cask_min_os = [cask.on_system_block_min_os, cask.depends_on.macos&.minimum_version].compact.max
 | |
|       odebug "Declared minimum OS version: #{cask_min_os&.to_sym}"
 | |
|       return if cask_min_os&.to_sym == min_os.to_sym
 | |
|       return if cask.on_system_blocks_exist? &&
 | |
|                 OnSystem.arch_condition_met?(:arm) &&
 | |
|                 cask_min_os.present? &&
 | |
|                 cask_min_os < MacOSVersion.new("11")
 | |
| 
 | |
|       min_os_definition = if cask_min_os.present?
 | |
|         if cask.on_system_block_min_os.present? &&
 | |
|            cask.on_system_block_min_os > cask.depends_on.macos&.minimum_version
 | |
|           "a block with a minimum OS version of #{cask_min_os.to_sym.inspect}"
 | |
|         else
 | |
|           cask_min_os.to_sym.inspect
 | |
|         end
 | |
|       else
 | |
|         "no minimum OS version"
 | |
|       end
 | |
|       add_error "Upstream defined #{min_os.to_sym.inspect} as the minimum OS version " \
 | |
|                 "but the cask declared #{min_os_definition}",
 | |
|                 strict_only: true
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.nilable(MacOSVersion)) }
 | |
|     def livecheck_min_os
 | |
|       return unless online?
 | |
|       return unless cask.livecheckable?
 | |
|       return if cask.livecheck.strategy != :sparkle
 | |
| 
 | |
|       # `Sparkle` strategy blocks that use the `items` argument (instead of
 | |
|       # `item`) contain arbitrary logic that ignores/overrides the strategy's
 | |
|       # sorting, so we can't identify which item would be first/newest here.
 | |
|       return if cask.livecheck.strategy_block.present? &&
 | |
|                 cask.livecheck.strategy_block.parameters[0] == [:opt, :items]
 | |
| 
 | |
|       content = Homebrew::Livecheck::Strategy.page_content(cask.livecheck.url)[:content]
 | |
|       return if content.blank?
 | |
| 
 | |
|       begin
 | |
|         items = Homebrew::Livecheck::Strategy::Sparkle.sort_items(
 | |
|           Homebrew::Livecheck::Strategy::Sparkle.filter_items(
 | |
|             Homebrew::Livecheck::Strategy::Sparkle.items_from_content(content),
 | |
|           ),
 | |
|         )
 | |
|       rescue
 | |
|         return
 | |
|       end
 | |
|       return if items.blank?
 | |
| 
 | |
|       min_os = items[0]&.minimum_system_version&.strip_patch
 | |
|       # Big Sur is sometimes identified as 10.16, so we override it to the
 | |
|       # expected macOS version (11).
 | |
|       min_os = MacOSVersion.new("11") if min_os == "10.16"
 | |
|       min_os
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.nilable(MacOSVersion)) }
 | |
|     def cask_plist_min_os
 | |
|       return unless online?
 | |
| 
 | |
|       plist_min_os = T.let(nil, T.untyped)
 | |
|       @staged_path ||= cask.staged_path
 | |
| 
 | |
|       extract_artifacts do |artifacts, tmpdir|
 | |
|         artifacts.each do |artifact|
 | |
|           artifact_path = artifact.is_a?(Artifact::Pkg) ? artifact.path : artifact.source
 | |
|           path = tmpdir/artifact_path.relative_path_from(cask.staged_path)
 | |
|           plist_path = "#{path}/Contents/Info.plist"
 | |
|           next unless File.exist?(plist_path)
 | |
| 
 | |
|           plist = system_command!("plutil", args: ["-convert", "xml1", "-o", "-", plist_path]).plist
 | |
|           plist_min_os = plist["LSMinimumSystemVersion"].presence
 | |
|           break if plist_min_os
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       begin
 | |
|         MacOSVersion.new(plist_min_os).strip_patch
 | |
|       rescue MacOSVersion::Error
 | |
|         nil
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_github_prerelease_version
 | |
|       odebug "Auditing GitHub prerelease"
 | |
|       user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*}) if online?
 | |
|       return if user.nil?
 | |
| 
 | |
|       tag = SharedAudits.github_tag_from_url(cask.url)
 | |
|       tag ||= cask.version
 | |
|       error = SharedAudits.github_release(user, repo, tag, cask:)
 | |
|       add_error error, location: cask.url.location if error
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_gitlab_prerelease_version
 | |
|       user, repo = get_repo_data(%r{https?://gitlab\.com/([^/]+)/([^/]+)/?.*}) if online?
 | |
|       return if user.nil?
 | |
| 
 | |
|       odebug "Auditing GitLab prerelease"
 | |
| 
 | |
|       tag = SharedAudits.gitlab_tag_from_url(cask.url)
 | |
|       tag ||= cask.version
 | |
|       error = SharedAudits.gitlab_release(user, repo, tag, cask:)
 | |
|       add_error error, location: cask.url.location if error
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_github_repository_archived
 | |
|       # Deprecated/disabled casks may have an archived repository.
 | |
|       return if cask.deprecated? || cask.disabled?
 | |
| 
 | |
|       user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*}) if online?
 | |
|       return if user.nil?
 | |
| 
 | |
|       metadata = SharedAudits.github_repo_data(user, repo)
 | |
|       return if metadata.nil?
 | |
| 
 | |
|       add_error "GitHub repo is archived", location: cask.url.location if metadata["archived"]
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_gitlab_repository_archived
 | |
|       # Deprecated/disabled casks may have an archived repository.
 | |
|       return if cask.deprecated? || cask.disabled?
 | |
| 
 | |
|       user, repo = get_repo_data(%r{https?://gitlab\.com/([^/]+)/([^/]+)/?.*}) if online?
 | |
|       return if user.nil?
 | |
| 
 | |
|       odebug "Auditing GitLab repo archived"
 | |
| 
 | |
|       metadata = SharedAudits.gitlab_repo_data(user, repo)
 | |
|       return if metadata.nil?
 | |
| 
 | |
|       add_error "GitLab repo is archived", location: cask.url.location if metadata["archived"]
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_github_repository
 | |
|       return unless new_cask?
 | |
| 
 | |
|       user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*})
 | |
|       return if user.nil?
 | |
| 
 | |
|       odebug "Auditing GitHub repo"
 | |
| 
 | |
|       error = SharedAudits.github(user, repo)
 | |
|       add_error error, location: cask.url.location if error
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_gitlab_repository
 | |
|       return unless new_cask?
 | |
| 
 | |
|       user, repo = get_repo_data(%r{https?://gitlab\.com/([^/]+)/([^/]+)/?.*})
 | |
|       return if user.nil?
 | |
| 
 | |
|       odebug "Auditing GitLab repo"
 | |
| 
 | |
|       error = SharedAudits.gitlab(user, repo)
 | |
|       add_error error, location: cask.url.location if error
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_bitbucket_repository
 | |
|       return unless new_cask?
 | |
| 
 | |
|       user, repo = get_repo_data(%r{https?://bitbucket\.org/([^/]+)/([^/]+)/?.*})
 | |
|       return if user.nil?
 | |
| 
 | |
|       odebug "Auditing Bitbucket repo"
 | |
| 
 | |
|       error = SharedAudits.bitbucket(user, repo)
 | |
|       add_error error, location: cask.url.location if error
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_denylist
 | |
|       return unless cask.tap
 | |
|       return unless cask.tap.official?
 | |
|       return unless (reason = Denylist.reason(cask.token))
 | |
| 
 | |
|       add_error "#{cask.token} is not allowed: #{reason}"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_reverse_migration
 | |
|       return unless new_cask?
 | |
|       return unless cask.tap
 | |
|       return unless cask.tap.official?
 | |
|       return unless cask.tap.tap_migrations.key?(cask.token)
 | |
| 
 | |
|       add_error "#{cask.token} is listed in tap_migrations.json"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_homepage_https_availability
 | |
|       return unless online?
 | |
|       return unless (homepage = cask.homepage)
 | |
| 
 | |
|       user_agents = if cask.tap&.audit_exception(:simple_user_agent_for_homepage, cask.token)
 | |
|         ["curl"]
 | |
|       else
 | |
|         [:browser, :default]
 | |
|       end
 | |
| 
 | |
|       validate_url_for_https_availability(
 | |
|         homepage, SharedAudits::URL_TYPE_HOMEPAGE,
 | |
|         user_agents:,
 | |
|         check_content: true,
 | |
|         strict:        strict?
 | |
|       )
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_url_https_availability
 | |
|       return unless online?
 | |
|       return unless (url = cask.url)
 | |
|       return if url.using
 | |
| 
 | |
|       validate_url_for_https_availability(
 | |
|         url, "binary URL",
 | |
|         location:    cask.url.location,
 | |
|         user_agents: [cask.url.user_agent],
 | |
|         referer:     cask.url&.referer
 | |
|       )
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_livecheck_https_availability
 | |
|       return unless online?
 | |
|       return unless cask.livecheckable?
 | |
|       return unless (url = cask.livecheck.url)
 | |
|       return if url.is_a?(Symbol)
 | |
| 
 | |
|       validate_url_for_https_availability(
 | |
|         url, "livecheck URL",
 | |
|         check_content: true,
 | |
|         user_agents:   [:default, :browser]
 | |
|       )
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def audit_cask_path
 | |
|       return unless cask.tap.core_cask_tap?
 | |
| 
 | |
|       expected_path = cask.tap.new_cask_path(cask.token)
 | |
| 
 | |
|       return if cask.sourcefile_path.to_s.end_with?(expected_path)
 | |
| 
 | |
|       add_error "Cask should be located in '#{expected_path}'"
 | |
|     end
 | |
| 
 | |
|     sig {
 | |
|       params(
 | |
|         url_to_check: T.any(String, URL),
 | |
|         url_type:     String,
 | |
|         location:     T.nilable(Homebrew::SourceLocation),
 | |
|         options:      T.untyped,
 | |
|       ).void
 | |
|     }
 | |
|     def validate_url_for_https_availability(url_to_check, url_type, location: nil, **options)
 | |
|       problem = curl_check_http_content(url_to_check.to_s, url_type, **options)
 | |
|       exception = cask.tap&.audit_exception(:secure_connection_audit_skiplist, cask.token, url_to_check.to_s)
 | |
| 
 | |
|       if problem
 | |
|         add_error problem, location: location unless exception
 | |
|       elsif exception
 | |
|         add_error "#{url_to_check} is in the secure connection audit skiplist but does not need to be skipped",
 | |
|                   location:
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { params(regex: T.any(String, Regexp)).returns(T.nilable(T::Array[String])) }
 | |
|     def get_repo_data(regex)
 | |
|       return unless online?
 | |
| 
 | |
|       _, user, repo = *regex.match(cask.url.to_s)
 | |
|       _, user, repo = *regex.match(cask.homepage) unless user
 | |
|       return if !user || !repo
 | |
| 
 | |
|       repo.gsub!(/.git$/, "")
 | |
| 
 | |
|       [user, repo]
 | |
|     end
 | |
| 
 | |
|     sig {
 | |
|       params(regex: T.any(String, Regexp), valid_formats_array: T::Array[T.any(String, Regexp)]).returns(T::Boolean)
 | |
|     }
 | |
|     def bad_url_format?(regex, valid_formats_array)
 | |
|       return false unless cask.url.to_s.match?(regex)
 | |
| 
 | |
|       valid_formats_array.none? { |format| cask.url.to_s.match?(format) }
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def bad_sourceforge_url?
 | |
|       bad_url_format?(%r{((downloads|\.dl)\.|//)sourceforge},
 | |
|                       [
 | |
|                         %r{\Ahttps://sourceforge\.net/projects/[^/]+/files/latest/download\Z},
 | |
|                         %r{\Ahttps://downloads\.sourceforge\.net/(?!(project|sourceforge)/)},
 | |
|                       ])
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def bad_osdn_url?
 | |
|       domain.match?(%r{^(?:\w+\.)*osdn\.jp(?=/|$)})
 | |
|     end
 | |
| 
 | |
|     # sig { returns(String) }
 | |
|     def homepage
 | |
|       URI(cask.homepage.to_s).host
 | |
|     end
 | |
| 
 | |
|     # sig { returns(String) }
 | |
|     def domain
 | |
|       URI(cask.url.to_s).host
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def url_match_homepage?
 | |
|       host = cask.url.to_s
 | |
|       host_uri = URI(host)
 | |
|       host = if host.match?(/:\d/) && host_uri.port != 80
 | |
|         "#{host_uri.host}:#{host_uri.port}"
 | |
|       else
 | |
|         host_uri.host
 | |
|       end
 | |
| 
 | |
|       return false if homepage.blank?
 | |
| 
 | |
|       home = homepage.downcase
 | |
|       if (split_host = T.must(host).split(".")).length >= 3
 | |
|         host = T.must(split_host[-2..]).join(".")
 | |
|       end
 | |
|       if (split_home = homepage.split(".")).length >= 3
 | |
|         home = split_home[-2..].join(".")
 | |
|       end
 | |
|       host == home
 | |
|     end
 | |
| 
 | |
|     # sig { params(url: String).returns(String) }
 | |
|     def strip_url_scheme(url)
 | |
|       url.sub(%r{^[^:/]+://(www\.)?}, "")
 | |
|     end
 | |
| 
 | |
|     # sig { returns(String) }
 | |
|     def url_from_verified
 | |
|       strip_url_scheme(cask.url.verified)
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def verified_matches_url?
 | |
|       url_domain, url_path = strip_url_scheme(cask.url.to_s).split("/", 2)
 | |
|       verified_domain, verified_path = url_from_verified.split("/", 2)
 | |
| 
 | |
|       (url_domain == verified_domain || (verified_domain && url_domain&.end_with?(".#{verified_domain}"))) &&
 | |
|         (!verified_path || url_path&.start_with?(verified_path))
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def verified_present?
 | |
|       cask.url.verified.present?
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def file_url?
 | |
|       URI(cask.url.to_s).scheme == "file"
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def block_url_offline?
 | |
|       return false if online?
 | |
| 
 | |
|       cask.url.from_block?
 | |
|     end
 | |
| 
 | |
|     sig { returns(Tap) }
 | |
|     def core_tap
 | |
|       @core_tap ||= CoreTap.instance
 | |
|     end
 | |
| 
 | |
|     # sig { returns(T::Array[String]) }
 | |
|     def core_formula_names
 | |
|       core_tap.formula_names
 | |
|     end
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def core_formula_url
 | |
|       formula_path = Formulary.core_path(cask.token)
 | |
|                               .to_s
 | |
|                               .delete_prefix(core_tap.path.to_s)
 | |
|       "#{core_tap.default_remote}/blob/HEAD#{formula_path}"
 | |
|     end
 | |
|   end
 | |
| end
 | 
