| 
									
										
										
										
											2023-03-25 08:36:56 -07:00
										 |  |  | # typed: true | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-24 12:31:02 +01:00
										 |  |  | require "deprecate_disable" | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | require "formula_text_auditor" | 
					
						
							| 
									
										
										
										
											2024-03-30 13:50:02 -07:00
										 |  |  | require "formula_versions" | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | require "resource_auditor" | 
					
						
							| 
									
										
										
										
											2024-03-31 13:14:56 -07:00
										 |  |  | require "utils/shared_audits" | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | module Homebrew | 
					
						
							|  |  |  |   # Auditor for checking common violations in {Formula}e. | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   # @api private | 
					
						
							|  |  |  |   class FormulaAuditor | 
					
						
							|  |  |  |     include FormulaCellarChecks | 
					
						
							| 
									
										
										
										
											2023-09-04 22:17:57 -04:00
										 |  |  |     include Utils::Curl | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     attr_reader :formula, :text, :problems, :new_formula_problems | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def initialize(formula, options = {}) | 
					
						
							|  |  |  |       @formula = formula | 
					
						
							|  |  |  |       @versioned_formula = formula.versioned_formula? | 
					
						
							|  |  |  |       @new_formula_inclusive = options[:new_formula] | 
					
						
							|  |  |  |       @new_formula = options[:new_formula] && !@versioned_formula | 
					
						
							|  |  |  |       @strict = options[:strict] | 
					
						
							|  |  |  |       @online = options[:online] | 
					
						
							|  |  |  |       @git = options[:git] | 
					
						
							|  |  |  |       @display_cop_names = options[:display_cop_names] | 
					
						
							|  |  |  |       @only = options[:only] | 
					
						
							|  |  |  |       @except = options[:except] | 
					
						
							|  |  |  |       # Accept precomputed style offense results, for efficiency | 
					
						
							|  |  |  |       @style_offenses = options[:style_offenses] | 
					
						
							|  |  |  |       # Allow the formula tap to be set as homebrew/core, for testing purposes | 
					
						
							|  |  |  |       @core_tap = formula.tap&.core_tap? || options[:core_tap] | 
					
						
							|  |  |  |       @problems = [] | 
					
						
							|  |  |  |       @new_formula_problems = [] | 
					
						
							|  |  |  |       @text = FormulaTextAuditor.new(formula.path) | 
					
						
							| 
									
										
										
										
											2024-02-22 23:29:55 +00:00
										 |  |  |       @specs = %w[stable head].filter_map { |s| formula.send(s) } | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       @spdx_license_data = options[:spdx_license_data] | 
					
						
							|  |  |  |       @spdx_exception_data = options[:spdx_exception_data] | 
					
						
							| 
									
										
										
										
											2023-06-20 22:36:15 +08:00
										 |  |  |       @tap_audit = options[:tap_audit] | 
					
						
							| 
									
										
										
										
											2023-12-13 17:35:02 -05:00
										 |  |  |       @previous_committed = {} | 
					
						
							|  |  |  |       @newest_committed = {} | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_style | 
					
						
							|  |  |  |       return unless @style_offenses | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       @style_offenses.each do |offense| | 
					
						
							|  |  |  |         cop_name = "#{offense.cop_name}: " if @display_cop_names | 
					
						
							| 
									
										
										
										
											2023-05-19 16:59:14 +02:00
										 |  |  |         message = "#{cop_name}#{offense.message}" | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-19 16:59:14 +02:00
										 |  |  |         problem message, location: offense.location, corrected: offense.corrected? | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_file | 
					
						
							|  |  |  |       if formula.core_formula? && @versioned_formula | 
					
						
							| 
									
										
										
										
											2022-09-02 23:02:07 -07:00
										 |  |  |         unversioned_name = formula.name.gsub(/@.*$/, "") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # ignore when an unversioned formula doesn't exist after an explicit rename | 
					
						
							|  |  |  |         return if formula.tap.formula_renames.key?(unversioned_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # build this ourselves as we want e.g. homebrew/core to be present | 
					
						
							|  |  |  |         full_name = "#{formula.tap}/#{unversioned_name}" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         unversioned_formula = begin | 
					
						
							| 
									
										
										
										
											2022-09-02 23:02:07 -07:00
										 |  |  |           Formulary.factory(full_name).path | 
					
						
							| 
									
										
										
										
											2024-02-16 21:27:02 +01:00
										 |  |  |         rescue FormulaUnavailableError, TapFormulaAmbiguityError | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |           Pathname.new formula.path.to_s.gsub(/@.*\.rb$/, ".rb") | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         unless unversioned_formula.exist? | 
					
						
							|  |  |  |           unversioned_name = unversioned_formula.basename(".rb") | 
					
						
							|  |  |  |           problem "#{formula} is versioned but no #{unversioned_name} formula exists" | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2022-03-21 14:29:36 +00:00
										 |  |  |       elsif formula.stable? && | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |             !@versioned_formula && | 
					
						
							|  |  |  |             (versioned_formulae = formula.versioned_formulae - [formula]) && | 
					
						
							|  |  |  |             versioned_formulae.present? | 
					
						
							| 
									
										
										
										
											2023-12-27 13:00:48 -08:00
										 |  |  |         versioned_aliases, unversioned_aliases = formula.aliases.partition { |a| /.@\d/.match?(a) } | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         _, last_alias_version = versioned_formulae.map(&:name).last.split("@") | 
					
						
							| 
									
										
										
										
											2022-03-21 14:29:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         alias_name_major = "#{formula.name}@#{formula.version.major}" | 
					
						
							| 
									
										
										
										
											2022-03-21 14:29:36 +00:00
										 |  |  |         alias_name_major_minor = "#{formula.name}@#{formula.version.major_minor}" | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         alias_name = if last_alias_version.split(".").length == 1
 | 
					
						
							|  |  |  |           alias_name_major | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           alias_name_major_minor | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2022-03-21 14:29:36 +00:00
										 |  |  |         valid_main_alias_names = [alias_name_major, alias_name_major_minor].uniq | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Also accept versioned aliases with names of other aliases, but do not require them. | 
					
						
							|  |  |  |         valid_other_alias_names = unversioned_aliases.flat_map do |name| | 
					
						
							|  |  |  |           %W[
 | 
					
						
							|  |  |  |             #{name}@#{formula.version.major} | 
					
						
							|  |  |  |             #{name}@#{formula.version.major_minor} | 
					
						
							|  |  |  |           ].uniq | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         unless @core_tap | 
					
						
							| 
									
										
										
										
											2022-03-21 14:29:36 +00:00
										 |  |  |           [versioned_aliases, valid_main_alias_names, valid_other_alias_names].each do |array| | 
					
						
							|  |  |  |             array.map! { |a| "#{formula.tap}/#{a}" } | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-21 14:29:36 +00:00
										 |  |  |         valid_versioned_aliases = versioned_aliases & valid_main_alias_names | 
					
						
							|  |  |  |         invalid_versioned_aliases = versioned_aliases - valid_main_alias_names - valid_other_alias_names | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         latest_versioned_formula = versioned_formulae.map(&:name).first | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-21 14:29:36 +00:00
										 |  |  |         if valid_versioned_aliases.empty? && alias_name != latest_versioned_formula | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |           if formula.tap | 
					
						
							|  |  |  |             problem <<~EOS | 
					
						
							|  |  |  |               Formula has other versions so create a versioned alias: | 
					
						
							|  |  |  |                 cd #{formula.tap.alias_dir} | 
					
						
							|  |  |  |                 ln -s #{formula.path.to_s.gsub(formula.tap.path, "..")} #{alias_name} | 
					
						
							|  |  |  |             EOS | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             problem "Formula has other versions so create an alias named #{alias_name}." | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if invalid_versioned_aliases.present? | 
					
						
							|  |  |  |           problem <<~EOS | 
					
						
							|  |  |  |             Formula has invalid versioned aliases: | 
					
						
							|  |  |  |               #{invalid_versioned_aliases.join("\n  ")} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2023-09-04 13:12:07 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-04 14:35:59 +01:00
										 |  |  |       return if !formula.core_formula? || formula.path == formula.tap.new_formula_path(formula.name) | 
					
						
							| 
									
										
										
										
											2023-09-04 13:12:07 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       problem <<~EOS | 
					
						
							|  |  |  |         Formula is in wrong path: | 
					
						
							|  |  |  |           Expected: #{formula.tap.new_formula_path(formula.name)} | 
					
						
							|  |  |  |             Actual: #{formula.path} | 
					
						
							|  |  |  |       EOS | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def self.aliases | 
					
						
							|  |  |  |       # core aliases + tap alias names + tap alias full name | 
					
						
							|  |  |  |       @aliases ||= Formula.aliases + Formula.tap_aliases | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-20 20:14:51 +01:00
										 |  |  |     def audit_synced_versions_formulae | 
					
						
							| 
									
										
										
										
											2024-03-03 01:55:56 -05:00
										 |  |  |       return unless formula.synced_with_other_formulae? | 
					
						
							| 
									
										
										
										
											2021-10-20 20:14:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       name = formula.name | 
					
						
							|  |  |  |       version = formula.version | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-03 01:55:56 -05:00
										 |  |  |       formula.tap.synced_versions_formulae.each do |synced_version_formulae| | 
					
						
							| 
									
										
										
										
											2024-01-23 21:17:30 +00:00
										 |  |  |         next unless synced_version_formulae.include?(name) | 
					
						
							| 
									
										
										
										
											2021-10-20 20:14:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         synced_version_formulae.each do |synced_formula| | 
					
						
							|  |  |  |           next if synced_formula == name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (synced_version = Formulary.factory(synced_formula).version) != version | 
					
						
							|  |  |  |             problem "Version of `#{synced_formula}` (#{synced_version}) should match version of `#{name}` (#{version})" | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         break | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |     def audit_formula_name | 
					
						
							| 
									
										
										
										
											2021-05-27 17:55:28 -07:00
										 |  |  |       name = formula.name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem "Formula name '#{name}' must not contain uppercase letters." if name != name.downcase | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       return unless @strict | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem "'#{name}' is not allowed in homebrew/core." if MissingFormula.disallowed_reason(name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if Formula.aliases.include? name | 
					
						
							|  |  |  |         problem "Formula name conflicts with existing aliases in homebrew/core." | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-12 18:33:37 +05:30
										 |  |  |       if (oldname = CoreTap.instance.formula_renames[name]) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         problem "'#{name}' is reserved as the old name of #{oldname} in homebrew/core." | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return if formula.core_formula? | 
					
						
							|  |  |  |       return unless Formula.core_names.include?(name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem "Formula name conflicts with existing core formula." | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PERMITTED_LICENSE_MISMATCHES = { | 
					
						
							|  |  |  |       "AGPL-3.0" => ["AGPL-3.0-only", "AGPL-3.0-or-later"], | 
					
						
							|  |  |  |       "GPL-2.0"  => ["GPL-2.0-only",  "GPL-2.0-or-later"], | 
					
						
							|  |  |  |       "GPL-3.0"  => ["GPL-3.0-only",  "GPL-3.0-or-later"], | 
					
						
							|  |  |  |       "LGPL-2.1" => ["LGPL-2.1-only", "LGPL-2.1-or-later"], | 
					
						
							|  |  |  |       "LGPL-3.0" => ["LGPL-3.0-only", "LGPL-3.0-or-later"], | 
					
						
							|  |  |  |     }.freeze | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_license | 
					
						
							|  |  |  |       if formula.license.present? | 
					
						
							|  |  |  |         licenses, exceptions = SPDX.parse_license_expression formula.license | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-13 19:47:46 +08:00
										 |  |  |         sspl_licensed = licenses.any? { |license| license.to_s.start_with?("SSPL") } | 
					
						
							| 
									
										
										
										
											2022-09-13 19:22:11 +08:00
										 |  |  |         if sspl_licensed && @core_tap | 
					
						
							|  |  |  |           problem <<~EOS | 
					
						
							|  |  |  |             Formula #{formula.name} is SSPL-licensed. Software under the SSPL must not be packaged in homebrew/core. | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         non_standard_licenses = licenses.reject { |license| SPDX.valid_license? license } | 
					
						
							|  |  |  |         if non_standard_licenses.present? | 
					
						
							|  |  |  |           problem <<~EOS | 
					
						
							|  |  |  |             Formula #{formula.name} contains non-standard SPDX licenses: #{non_standard_licenses}. | 
					
						
							|  |  |  |             For a list of valid licenses check: #{Formatter.url("https://spdx.org/licenses/")} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if @strict | 
					
						
							|  |  |  |           deprecated_licenses = licenses.select do |license| | 
					
						
							|  |  |  |             SPDX.deprecated_license? license | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |           if deprecated_licenses.present? | 
					
						
							|  |  |  |             problem <<~EOS | 
					
						
							|  |  |  |               Formula #{formula.name} contains deprecated SPDX licenses: #{deprecated_licenses}. | 
					
						
							|  |  |  |               You may need to add `-only` or `-or-later` for GNU licenses (e.g. `GPL`, `LGPL`, `AGPL`, `GFDL`). | 
					
						
							|  |  |  |               For a list of valid licenses check: #{Formatter.url("https://spdx.org/licenses/")} | 
					
						
							|  |  |  |             EOS | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         invalid_exceptions = exceptions.reject { |exception| SPDX.valid_license_exception? exception } | 
					
						
							|  |  |  |         if invalid_exceptions.present? | 
					
						
							|  |  |  |           problem <<~EOS | 
					
						
							|  |  |  |             Formula #{formula.name} contains invalid or deprecated SPDX license exceptions: #{invalid_exceptions}. | 
					
						
							|  |  |  |             For a list of valid license exceptions check: | 
					
						
							|  |  |  |               #{Formatter.url("https://spdx.org/licenses/exceptions-index.html")} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return unless @online | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*}) | 
					
						
							|  |  |  |         return if user.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 16:53:54 +00:00
										 |  |  |         tag = SharedAudits.github_tag_from_url(formula.stable.url) | 
					
						
							| 
									
										
										
										
											2024-03-04 11:56:19 -05:00
										 |  |  |         tag ||= formula.stable.specs[:tag] | 
					
						
							| 
									
										
										
										
											2024-02-27 16:53:54 +00:00
										 |  |  |         github_license = GitHub.get_repo_license(user, repo, ref: tag) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         return unless github_license | 
					
						
							|  |  |  |         return if (licenses + ["NOASSERTION"]).include?(github_license) | 
					
						
							|  |  |  |         return if PERMITTED_LICENSE_MISMATCHES[github_license]&.any? { |license| licenses.include? license } | 
					
						
							| 
									
										
										
										
											2021-10-04 21:45:20 -04:00
										 |  |  |         return if formula.tap&.audit_exception :permitted_formula_license_mismatches, formula.name | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         problem "Formula license #{licenses} does not match GitHub license #{Array(github_license)}." | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       elsif @new_formula && @core_tap | 
					
						
							|  |  |  |         problem "Formulae in homebrew/core must specify a license." | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_deps | 
					
						
							|  |  |  |       @specs.each do |spec| | 
					
						
							|  |  |  |         # Check for things we don't like to depend on. | 
					
						
							|  |  |  |         # We allow non-Homebrew installs whenever possible. | 
					
						
							| 
									
										
										
										
											2023-06-19 06:03:31 +01:00
										 |  |  |         spec.declared_deps.each do |dep| | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |           begin | 
					
						
							|  |  |  |             dep_f = dep.to_formula | 
					
						
							|  |  |  |           rescue TapFormulaUnavailableError | 
					
						
							|  |  |  |             # Don't complain about missing cross-tap dependencies | 
					
						
							|  |  |  |             next | 
					
						
							|  |  |  |           rescue FormulaUnavailableError | 
					
						
							| 
									
										
										
										
											2024-02-15 22:43:29 +01:00
										 |  |  |             problem "Can't find dependency '#{dep.name}'." | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |             next | 
					
						
							|  |  |  |           rescue TapFormulaAmbiguityError | 
					
						
							| 
									
										
										
										
											2024-02-15 22:43:29 +01:00
										 |  |  |             problem "Ambiguous dependency '#{dep.name}'." | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |             next | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-06 16:47:09 +01:00
										 |  |  |           if dep_f.oldnames.include?(dep.name.split("/").last) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |             problem "Dependency '#{dep.name}' was renamed; use new name '#{dep_f.name}'." | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if @core_tap && | 
					
						
							|  |  |  |              @new_formula && | 
					
						
							| 
									
										
										
										
											2023-07-07 23:45:03 +01:00
										 |  |  |              !dep.uses_from_macos? && | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |              dep_f.keg_only? && | 
					
						
							|  |  |  |              dep_f.keg_only_reason.provided_by_macos? && | 
					
						
							|  |  |  |              dep_f.keg_only_reason.applicable? && | 
					
						
							| 
									
										
										
										
											2021-04-08 22:58:13 +02:00
										 |  |  |              formula.requirements.none?(LinuxRequirement) && | 
					
						
							| 
									
										
										
										
											2021-10-04 21:45:20 -04:00
										 |  |  |              !formula.tap&.audit_exception(:provided_by_macos_depends_on_allowlist, dep.name) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |             new_formula_problem( | 
					
						
							|  |  |  |               "Dependency '#{dep.name}' is provided by macOS; " \ | 
					
						
							|  |  |  |               "please replace 'depends_on' with 'uses_from_macos'.", | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           dep.options.each do |opt| | 
					
						
							|  |  |  |             next if @core_tap | 
					
						
							|  |  |  |             next if dep_f.option_defined?(opt) | 
					
						
							|  |  |  |             next if dep_f.requirements.find do |r| | 
					
						
							|  |  |  |               if r.recommended? | 
					
						
							|  |  |  |                 opt.name == "with-#{r.name}" | 
					
						
							|  |  |  |               elsif r.optional? | 
					
						
							|  |  |  |                 opt.name == "without-#{r.name}" | 
					
						
							|  |  |  |               end | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 21:55:35 -05:00
										 |  |  |             problem "Dependency '#{dep}' does not define option #{opt.name.inspect}" | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 21:55:35 -05:00
										 |  |  |           problem "Don't use 'git' as a dependency (it's always available)" if @new_formula && dep.name == "git" | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |           problem "Dependency '#{dep.name}' is marked as :run. Remove :run; it is a no-op." if dep.tags.include?(:run) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           next unless @core_tap | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-05 00:57:33 +00:00
										 |  |  |           if dep_f.tap.nil? | 
					
						
							|  |  |  |             problem <<~EOS | 
					
						
							|  |  |  |               Dependency '#{dep.name}' does not exist in any tap. | 
					
						
							|  |  |  |             EOS | 
					
						
							|  |  |  |           elsif !dep_f.tap.core_tap? | 
					
						
							| 
									
										
										
										
											2022-02-10 06:57:46 +08:00
										 |  |  |             problem <<~EOS | 
					
						
							|  |  |  |               Dependency '#{dep.name}' is not in homebrew/core. Formulae in homebrew/core | 
					
						
							|  |  |  |               should not have dependencies in external taps. | 
					
						
							|  |  |  |             EOS | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-19 03:00:55 +08:00
										 |  |  |           if dep_f.deprecated? && !formula.deprecated? && !formula.disabled? | 
					
						
							|  |  |  |             problem <<~EOS | 
					
						
							|  |  |  |               Dependency '#{dep.name}' is deprecated but has un-deprecated dependents. Either | 
					
						
							| 
									
										
										
										
											2022-07-29 20:10:05 +08:00
										 |  |  |               un-deprecate '#{dep.name}' or deprecate it and all of its dependents. | 
					
						
							| 
									
										
										
										
											2022-01-19 03:00:55 +08:00
										 |  |  |             EOS | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 23:47:28 +01:00
										 |  |  |           if dep_f.disabled? && !formula.disabled? | 
					
						
							|  |  |  |             problem <<~EOS | 
					
						
							|  |  |  |               Dependency '#{dep.name}' is disabled but has un-disabled dependents. Either | 
					
						
							|  |  |  |               un-disable '#{dep.name}' or disable it and all of its dependents. | 
					
						
							|  |  |  |             EOS | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-09 10:24:35 +01:00
										 |  |  |           # we want to allow uses_from_macos for aliases but not bare dependencies | 
					
						
							| 
									
										
										
										
											2023-06-19 06:03:31 +01:00
										 |  |  |           if self.class.aliases.include?(dep.name) && !dep.uses_from_macos? | 
					
						
							| 
									
										
										
										
											2020-12-24 13:59:59 +00:00
										 |  |  |             problem "Dependency '#{dep.name}' is an alias; use the canonical name '#{dep.to_formula.full_name}'." | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |           if dep.tags.include?(:recommended) || dep.tags.include?(:optional) | 
					
						
							|  |  |  |             problem "Formulae in homebrew/core should not have optional or recommended dependencies" | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         next unless @core_tap | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if spec.requirements.map(&:recommended?).any? || spec.requirements.map(&:optional?).any? | 
					
						
							|  |  |  |           problem "Formulae in homebrew/core should not have optional or recommended requirements" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							| 
									
										
										
										
											2021-10-04 21:45:20 -04:00
										 |  |  |       return if formula.tap&.audit_exception :versioned_dependencies_conflicts_allowlist, formula.name | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       # The number of conflicts on Linux is absurd. | 
					
						
							|  |  |  |       # TODO: remove this and check these there too. | 
					
						
							| 
									
										
										
										
											2022-07-28 14:52:19 -04:00
										 |  |  |       return if Homebrew::SimulateSystem.simulating_or_running_on_linux? | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-27 14:44:12 +08:00
										 |  |  |       # Skip the versioned dependencies conflict audit for *-staging branches. | 
					
						
							|  |  |  |       # This will allow us to migrate dependents of formulae like Python or OpenSSL | 
					
						
							|  |  |  |       # gradually over separate PRs which target a *-staging branch. See: | 
					
						
							|  |  |  |       #   https://github.com/Homebrew/homebrew-core/pull/134260 | 
					
						
							|  |  |  |       ignore_formula_conflict, staging_formula = | 
					
						
							|  |  |  |         if @tap_audit && (github_event_path = ENV.fetch("GITHUB_EVENT_PATH", nil)).present? | 
					
						
							|  |  |  |           event_payload = JSON.parse(File.read(github_event_path)) | 
					
						
							|  |  |  |           base_info = event_payload.dig("pull_request", "base").to_h # handle `nil` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           # We need to read the head ref from `GITHUB_EVENT_PATH` because | 
					
						
							|  |  |  |           # `git branch --show-current` returns `master` on PR branches. | 
					
						
							|  |  |  |           staging_branch = base_info["ref"]&.end_with?("-staging") | 
					
						
							|  |  |  |           homebrew_owned_repo = base_info.dig("repo", "owner", "login") == "Homebrew" | 
					
						
							|  |  |  |           homebrew_core_pr = base_info.dig("repo", "name") == "homebrew-core" | 
					
						
							|  |  |  |           # Support staging branches named `formula-staging` or `formula@version-staging`. | 
					
						
							|  |  |  |           base_formula = base_info["ref"]&.split(/-|@/, 2)&.first | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           [staging_branch && homebrew_owned_repo && homebrew_core_pr, base_formula] | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2023-06-10 00:46:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       recursive_runtime_formulae = formula.runtime_formula_dependencies(undeclared: false) | 
					
						
							|  |  |  |       version_hash = {} | 
					
						
							|  |  |  |       version_conflicts = Set.new | 
					
						
							|  |  |  |       recursive_runtime_formulae.each do |f| | 
					
						
							|  |  |  |         name = f.name | 
					
						
							|  |  |  |         unversioned_name, = name.split("@") | 
					
						
							| 
									
										
										
										
											2023-06-27 14:44:12 +08:00
										 |  |  |         next if ignore_formula_conflict && unversioned_name == staging_formula | 
					
						
							| 
									
										
										
										
											2022-08-18 15:27:23 +08:00
										 |  |  |         # Allow use of the full versioned name (e.g. `python@3.99`) or an unversioned alias (`python`). | 
					
						
							|  |  |  |         next if formula.tap&.audit_exception :versioned_formula_dependent_conflicts_allowlist, name | 
					
						
							|  |  |  |         next if formula.tap&.audit_exception :versioned_formula_dependent_conflicts_allowlist, unversioned_name | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         version_hash[unversioned_name] ||= Set.new | 
					
						
							|  |  |  |         version_hash[unversioned_name] << name | 
					
						
							|  |  |  |         next if version_hash[unversioned_name].length < 2
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         version_conflicts += version_hash[unversioned_name] | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return if version_conflicts.empty? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-24 12:31:02 +01:00
										 |  |  |       return if formula.disabled? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return if formula.deprecated? && | 
					
						
							| 
									
										
										
										
											2023-12-03 21:59:03 -05:00
										 |  |  |                 formula.deprecation_reason != DeprecateDisable::FORMULA_DEPRECATE_DISABLE_REASONS[:versioned_formula] | 
					
						
							| 
									
										
										
										
											2020-11-24 12:31:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       problem <<~EOS | 
					
						
							|  |  |  |         #{formula.full_name} contains conflicting version recursive dependencies: | 
					
						
							|  |  |  |           #{version_conflicts.to_a.join ", "} | 
					
						
							|  |  |  |         View these with `brew deps --tree #{formula.full_name}`. | 
					
						
							|  |  |  |       EOS | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_conflicts | 
					
						
							| 
									
										
										
										
											2021-06-18 16:40:12 +01:00
										 |  |  |       tap = formula.tap | 
					
						
							| 
									
										
										
										
											2021-06-18 16:57:53 +01:00
										 |  |  |       formula.conflicts.each do |conflict| | 
					
						
							|  |  |  |         conflicting_formula = Formulary.factory(conflict.name) | 
					
						
							| 
									
										
										
										
											2021-06-18 16:40:12 +01:00
										 |  |  |         next if tap != conflicting_formula.tap | 
					
						
							| 
									
										
										
										
											2021-06-14 14:14:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-18 16:40:12 +01:00
										 |  |  |         problem "Formula should not conflict with itself" if formula == conflicting_formula | 
					
						
							| 
									
										
										
										
											2021-06-16 22:52:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-24 14:01:53 -07:00
										 |  |  |         if T.must(tap).formula_renames.key?(conflict.name) || T.must(tap).aliases.include?(conflict.name) | 
					
						
							| 
									
										
										
										
											2021-06-16 22:52:09 +01:00
										 |  |  |           problem "Formula conflict should be declared using " \ | 
					
						
							| 
									
										
										
										
											2021-06-18 16:57:53 +01:00
										 |  |  |                   "canonical name (#{conflicting_formula.name}) instead of #{conflict.name}" | 
					
						
							| 
									
										
										
										
											2021-06-16 22:52:09 +01:00
										 |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-25 08:36:56 -07:00
										 |  |  |         reverse_conflict_found = T.let(false, T::Boolean) | 
					
						
							| 
									
										
										
										
											2021-06-18 16:57:53 +01:00
										 |  |  |         conflicting_formula.conflicts.each do |reverse_conflict| | 
					
						
							|  |  |  |           reverse_conflict_formula = Formulary.factory(reverse_conflict.name) | 
					
						
							| 
									
										
										
										
											2023-07-24 14:01:53 -07:00
										 |  |  |           if T.must(tap).formula_renames.key?(reverse_conflict.name) || | 
					
						
							|  |  |  |              T.must(tap).aliases.include?(reverse_conflict.name) | 
					
						
							| 
									
										
										
										
											2021-06-16 22:52:09 +01:00
										 |  |  |             problem "Formula #{conflicting_formula.name} conflict should be declared using " \ | 
					
						
							| 
									
										
										
										
											2021-06-18 16:57:53 +01:00
										 |  |  |                     "canonical name (#{reverse_conflict_formula.name}) instead of #{reverse_conflict.name}" | 
					
						
							| 
									
										
										
										
											2021-06-16 22:52:09 +01:00
										 |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-18 16:57:53 +01:00
										 |  |  |           reverse_conflict_found ||= reverse_conflict_formula == formula | 
					
						
							| 
									
										
										
										
											2021-06-16 22:52:09 +01:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-06-18 16:57:53 +01:00
										 |  |  |         unless reverse_conflict_found | 
					
						
							| 
									
										
										
										
											2021-06-14 14:14:36 +01:00
										 |  |  |           problem "Formula #{conflicting_formula.name} should also have a conflict declared with #{formula.name}" | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       rescue TapFormulaUnavailableError | 
					
						
							|  |  |  |         # Don't complain about missing cross-tap conflicts. | 
					
						
							|  |  |  |         next | 
					
						
							|  |  |  |       rescue FormulaUnavailableError | 
					
						
							| 
									
										
										
										
											2021-06-18 16:57:53 +01:00
										 |  |  |         problem "Can't find conflicting formula #{conflict.name.inspect}." | 
					
						
							| 
									
										
										
										
											2024-02-16 21:27:02 +01:00
										 |  |  |       rescue TapFormulaAmbiguityError | 
					
						
							| 
									
										
										
										
											2021-06-18 16:57:53 +01:00
										 |  |  |         problem "Ambiguous conflicting formula #{conflict.name.inspect}." | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-24 13:22:00 +08:00
										 |  |  |     def audit_gcc_dependency | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							|  |  |  |       return unless Homebrew::SimulateSystem.simulating_or_running_on_linux? | 
					
						
							|  |  |  |       return unless linux_only_gcc_dep?(formula) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem "Formulae in homebrew/core should not have a Linux-only dependency on GCC." | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |     def audit_postgresql | 
					
						
							| 
									
										
										
										
											2023-04-18 15:06:50 -07:00
										 |  |  |       return if formula.name != "postgresql" | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       return unless @core_tap | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       major_version = formula.version.major.to_i | 
					
						
							|  |  |  |       previous_major_version = major_version - 1
 | 
					
						
							|  |  |  |       previous_formula_name = "postgresql@#{previous_major_version}" | 
					
						
							|  |  |  |       begin | 
					
						
							|  |  |  |         Formula[previous_formula_name] | 
					
						
							|  |  |  |       rescue FormulaUnavailableError | 
					
						
							|  |  |  |         problem "Versioned #{previous_formula_name} in homebrew/core must be created for " \ | 
					
						
							| 
									
										
										
										
											2021-02-01 21:26:24 +09:00
										 |  |  |                 "`brew postgresql-upgrade-database` and `pg_upgrade` to work." | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-13 17:13:50 +01:00
										 |  |  |     def audit_glibc | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							| 
									
										
										
										
											2022-07-29 14:06:51 -07:00
										 |  |  |       return if formula.name != "glibc" | 
					
						
							| 
									
										
										
										
											2022-08-23 12:42:02 +01:00
										 |  |  |       # Also allow LINUX_GLIBC_NEXT_CI_VERSION for when we're upgrading. | 
					
						
							|  |  |  |       return if [OS::LINUX_GLIBC_CI_VERSION, OS::LINUX_GLIBC_NEXT_CI_VERSION].include?(formula.version.to_s) | 
					
						
							| 
									
										
										
										
											2021-02-13 17:13:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-23 12:42:02 +01:00
										 |  |  |       problem "The glibc version must be #{OS::LINUX_GLIBC_CI_VERSION}, as needed by our CI on Linux. " \ | 
					
						
							|  |  |  |               "The glibc formula is for users who have a system glibc with a lower version, " \ | 
					
						
							|  |  |  |               "which allows them to use our Linux bottles, which were compiled against system glibc on CI." | 
					
						
							| 
									
										
										
										
											2021-02-13 17:13:50 +01:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-20 20:48:51 -07:00
										 |  |  |     RELICENSED_FORMULAE_VERSIONS = { | 
					
						
							|  |  |  |       "boundary"           => "0.14", | 
					
						
							|  |  |  |       "consul"             => "1.17", | 
					
						
							|  |  |  |       "elasticsearch"      => "7.11", | 
					
						
							|  |  |  |       "kibana"             => "7.11", | 
					
						
							|  |  |  |       "nomad"              => "1.7", | 
					
						
							|  |  |  |       "packer"             => "1.10", | 
					
						
							|  |  |  |       "redis"              => "7.4", | 
					
						
							|  |  |  |       "terraform"          => "1.6", | 
					
						
							|  |  |  |       "vagrant"            => "2.4", | 
					
						
							|  |  |  |       "vagrant-completion" => "2.4", | 
					
						
							|  |  |  |       "vault"              => "1.15", | 
					
						
							|  |  |  |       "waypoint"           => "0.12", | 
					
						
							| 
									
										
										
										
											2023-09-08 22:29:19 +08:00
										 |  |  |     }.freeze | 
					
						
							| 
									
										
										
										
											2023-09-06 21:26:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-20 20:48:51 -07:00
										 |  |  |     def audit_relicensed_formulae | 
					
						
							|  |  |  |       return unless RELICENSED_FORMULAE_VERSIONS.key? formula.name | 
					
						
							| 
									
										
										
										
											2023-09-06 21:26:39 +02:00
										 |  |  |       return unless @core_tap | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-20 20:48:51 -07:00
										 |  |  |       relicensed_version = Version.new(RELICENSED_FORMULAE_VERSIONS[formula.name]) | 
					
						
							| 
									
										
										
										
											2023-09-08 22:29:19 +08:00
										 |  |  |       return if formula.version < relicensed_version | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem "#{formula.name} was relicensed to a non-open-source license from version #{relicensed_version}. " \ | 
					
						
							|  |  |  |               "It must not be upgraded to version #{relicensed_version} or newer." | 
					
						
							| 
									
										
										
										
											2023-09-06 21:26:39 +02:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-18 14:46:13 +00:00
										 |  |  |     def audit_keg_only_reason | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							|  |  |  |       return unless formula.keg_only? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-22 13:54:43 +00:00
										 |  |  |       keg_only_message = text.to_s.match(/keg_only\s+["'](.*)["']/)&.captures&.first | 
					
						
							| 
									
										
										
										
											2023-03-18 14:46:13 +00:00
										 |  |  |       return unless keg_only_message&.include?("HOMEBREW_PREFIX") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-18 15:27:35 +00:00
										 |  |  |       problem "`keg_only` reason should not include `HOMEBREW_PREFIX` as it creates confusing `brew info` output." | 
					
						
							| 
									
										
										
										
											2023-03-18 14:46:13 +00:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |     def audit_versioned_keg_only | 
					
						
							|  |  |  |       return unless @versioned_formula | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if formula.keg_only? | 
					
						
							|  |  |  |         return if formula.keg_only_reason.versioned_formula? | 
					
						
							| 
									
										
										
										
											2021-07-13 21:20:49 +09:00
										 |  |  |         return if formula.name.start_with?("openssl", "libressl") && formula.keg_only_reason.by_macos? | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-04 21:45:20 -04:00
										 |  |  |       return if formula.tap&.audit_exception :versioned_keg_only_allowlist, formula.name | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       problem "Versioned formulae in homebrew/core should use `keg_only :versioned_formula`" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_homepage | 
					
						
							|  |  |  |       homepage = formula.homepage | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-01 17:04:59 +00:00
										 |  |  |       return if homepage.blank? | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       return unless @online | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-04 21:45:20 -04:00
										 |  |  |       return if formula.tap&.audit_exception :cert_error_allowlist, formula.name, homepage | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       return unless DevelopmentTools.curl_handles_most_https_certificates? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-01 22:22:19 -05:00
										 |  |  |       use_homebrew_curl = [:stable, :head].any? do |spec_name| | 
					
						
							|  |  |  |         next false unless (spec = formula.send(spec_name)) | 
					
						
							| 
									
										
										
										
											2021-08-20 10:14:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-01 22:22:19 -05:00
										 |  |  |         spec.using == :homebrew_curl | 
					
						
							| 
									
										
										
										
											2021-08-20 10:14:50 +02:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-04 22:17:57 -04:00
										 |  |  |       if (http_content_problem = curl_check_http_content( | 
					
						
							|  |  |  |         homepage, | 
					
						
							|  |  |  |         SharedAudits::URL_TYPE_HOMEPAGE, | 
					
						
							|  |  |  |         user_agents:       [:browser, :default], | 
					
						
							|  |  |  |         check_content:     true, | 
					
						
							|  |  |  |         strict:            @strict, | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |         use_homebrew_curl:, | 
					
						
							| 
									
										
										
										
											2023-09-04 22:17:57 -04:00
										 |  |  |       )) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         problem http_content_problem | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_bottle_spec | 
					
						
							|  |  |  |       # special case: new versioned formulae should be audited | 
					
						
							|  |  |  |       return unless @new_formula_inclusive | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return unless formula.bottle_defined? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       new_formula_problem "New formulae in homebrew/core should not have a `bottle do` block" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-01 23:14:40 +02:00
										 |  |  |     def audit_eol | 
					
						
							|  |  |  |       return unless @online | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return if formula.deprecated? || formula.disabled? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       name = if formula.versioned_formula? | 
					
						
							|  |  |  |         formula.name.split("@").first | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         formula.name | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return if formula.tap&.audit_exception :eol_date_blocklist, name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       metadata = SharedAudits.eol_data(name, formula.version.major) | 
					
						
							|  |  |  |       metadata ||= SharedAudits.eol_data(name, formula.version.major_minor) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return if metadata.blank? || metadata["eol"] == false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       see_url = "see #{Formatter.url("https://endoflife.date/#{name}")}" | 
					
						
							|  |  |  |       if metadata["eol"] == true | 
					
						
							|  |  |  |         problem "Product is EOL, #{see_url}" | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem "Product is EOL since #{metadata["eol"]}, #{see_url}" if Date.parse(metadata["eol"]) <= Date.today | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |     def audit_github_repository_archived | 
					
						
							| 
									
										
										
										
											2021-02-03 10:12:36 -05:00
										 |  |  |       return if formula.deprecated? || formula.disabled? | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*}) if @online | 
					
						
							|  |  |  |       return if user.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       metadata = SharedAudits.github_repo_data(user, repo) | 
					
						
							|  |  |  |       return if metadata.nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem "GitHub repo is archived" if metadata["archived"] | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_gitlab_repository_archived | 
					
						
							| 
									
										
										
										
											2021-02-03 10:12:36 -05:00
										 |  |  |       return if formula.deprecated? || formula.disabled? | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       user, repo = get_repo_data(%r{https?://gitlab\.com/([^/]+)/([^/]+)/?.*}) if @online | 
					
						
							|  |  |  |       return if user.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       metadata = SharedAudits.gitlab_repo_data(user, repo) | 
					
						
							|  |  |  |       return if metadata.nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem "GitLab repo is archived" if metadata["archived"] | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_github_repository | 
					
						
							|  |  |  |       user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*}) if @new_formula | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return if user.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       warning = SharedAudits.github(user, repo) | 
					
						
							|  |  |  |       return if warning.nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       new_formula_problem warning | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_gitlab_repository | 
					
						
							|  |  |  |       user, repo = get_repo_data(%r{https?://gitlab\.com/([^/]+)/([^/]+)/?.*}) if @new_formula | 
					
						
							|  |  |  |       return if user.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       warning = SharedAudits.gitlab(user, repo) | 
					
						
							|  |  |  |       return if warning.nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       new_formula_problem warning | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_bitbucket_repository | 
					
						
							|  |  |  |       user, repo = get_repo_data(%r{https?://bitbucket\.org/([^/]+)/([^/]+)/?.*}) if @new_formula | 
					
						
							|  |  |  |       return if user.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       warning = SharedAudits.bitbucket(user, repo) | 
					
						
							|  |  |  |       return if warning.nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       new_formula_problem warning | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_repo_data(regex) | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							|  |  |  |       return unless @online | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       _, user, repo = *regex.match(formula.stable.url) if formula.stable | 
					
						
							|  |  |  |       _, user, repo = *regex.match(formula.homepage) unless user | 
					
						
							|  |  |  |       _, user, repo = *regex.match(formula.head.url) if !user && formula.head | 
					
						
							|  |  |  |       return if !user || !repo | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       repo.delete_suffix!(".git") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       [user, repo] | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_specs | 
					
						
							|  |  |  |       problem "Head-only (no stable download)" if head_only?(formula) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       %w[Stable HEAD].each do |name| | 
					
						
							|  |  |  |         spec_name = name.downcase.to_sym | 
					
						
							| 
									
										
										
										
											2021-02-12 18:33:37 +05:30
										 |  |  |         next unless (spec = formula.send(spec_name)) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-23 15:33:44 +05:30
										 |  |  |         except = @except.to_a | 
					
						
							|  |  |  |         if spec_name == :head && | 
					
						
							| 
									
										
										
										
											2021-10-04 21:45:20 -04:00
										 |  |  |            formula.tap&.audit_exception(:head_non_default_branch_allowlist, formula.name, spec.specs[:branch]) | 
					
						
							| 
									
										
										
										
											2021-07-23 15:33:44 +05:30
										 |  |  |           except << "head_branch" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-15 09:55:28 -04:00
										 |  |  |         ra = ResourceAuditor.new( | 
					
						
							|  |  |  |           spec, spec_name, | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |           online: @online, strict: @strict, only: @only, except:, | 
					
						
							| 
									
										
										
										
											2021-07-26 12:39:25 +02:00
										 |  |  |           use_homebrew_curl: spec.using == :homebrew_curl | 
					
						
							| 
									
										
										
										
											2021-06-15 09:55:28 -04:00
										 |  |  |         ).audit | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         ra.problems.each do |message| | 
					
						
							|  |  |  |           problem "#{name}: #{message}" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         spec.resources.each_value do |resource| | 
					
						
							|  |  |  |           problem "Resource name should be different from the formula name" if resource.name == formula.name | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-15 09:55:28 -04:00
										 |  |  |           ra = ResourceAuditor.new( | 
					
						
							|  |  |  |             resource, spec_name, | 
					
						
							| 
									
										
										
										
											2021-10-30 20:04:13 +08:00
										 |  |  |             online: @online, strict: @strict, only: @only, except: @except, | 
					
						
							|  |  |  |             use_homebrew_curl: resource.using == :homebrew_curl | 
					
						
							| 
									
										
										
										
											2021-06-15 09:55:28 -04:00
										 |  |  |           ).audit | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |           ra.problems.each do |message| | 
					
						
							|  |  |  |             problem "#{name} resource #{resource.name.inspect}: #{message}" | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         next if spec.patches.empty? | 
					
						
							| 
									
										
										
										
											2021-01-04 14:07:21 -05:00
										 |  |  |         next if !@new_formula || !@core_tap | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         new_formula_problem( | 
					
						
							|  |  |  |           "Formulae should not require patches to build. " \ | 
					
						
							|  |  |  |           "Patches should be submitted and accepted upstream first.", | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if formula.head && @versioned_formula && | 
					
						
							| 
									
										
										
										
											2021-10-04 21:45:20 -04:00
										 |  |  |          !formula.tap&.audit_exception(:versioned_head_spec_allowlist, formula.name) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         problem "Versioned formulae should not have a `HEAD` spec" | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       stable = formula.stable | 
					
						
							|  |  |  |       return unless stable | 
					
						
							|  |  |  |       return unless stable.url | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-13 21:55:47 +09:00
										 |  |  |       version = stable.version | 
					
						
							| 
									
										
										
										
											2023-12-27 13:00:48 -08:00
										 |  |  |       problem "Stable: version (#{version}) is set to a string without a digit" unless /\d/.match?(version.to_s) | 
					
						
							| 
									
										
										
										
											2021-07-13 21:55:47 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |       stable_version_string = version.to_s | 
					
						
							|  |  |  |       if stable_version_string.start_with?("HEAD") | 
					
						
							|  |  |  |         problem "Stable: non-HEAD version name (#{stable_version_string}) should not begin with HEAD" | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       stable_url_version = Version.parse(stable.url) | 
					
						
							|  |  |  |       stable_url_minor_version = stable_url_version.minor.to_i | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       formula_suffix = stable.version.patch.to_i | 
					
						
							| 
									
										
										
										
											2024-04-01 00:54:13 -04:00
										 |  |  |       throttled_rate = formula.livecheck.throttle | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       if throttled_rate && formula_suffix.modulo(throttled_rate).nonzero? | 
					
						
							|  |  |  |         problem "should only be updated every #{throttled_rate} releases on multiples of #{throttled_rate}" | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case (url = stable.url) | 
					
						
							|  |  |  |       when /[\d._-](alpha|beta|rc\d)/ | 
					
						
							|  |  |  |         matched = Regexp.last_match(1) | 
					
						
							|  |  |  |         version_prefix = stable_version_string.sub(/\d+$/, "") | 
					
						
							| 
									
										
										
										
											2021-10-04 21:45:20 -04:00
										 |  |  |         return if formula.tap&.audit_exception :unstable_allowlist, formula.name, version_prefix | 
					
						
							|  |  |  |         return if formula.tap&.audit_exception :unstable_devel_allowlist, formula.name, version_prefix | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         problem "Stable version URLs should not contain #{matched}" | 
					
						
							|  |  |  |       when %r{download\.gnome\.org/sources}, %r{ftp\.gnome\.org/pub/GNOME/sources}i | 
					
						
							|  |  |  |         version_prefix = stable.version.major_minor | 
					
						
							| 
									
										
										
										
											2021-10-04 21:45:20 -04:00
										 |  |  |         return if formula.tap&.audit_exception :gnome_devel_allowlist, formula.name, version_prefix | 
					
						
							| 
									
										
										
										
											2023-07-06 16:47:09 +01:00
										 |  |  |         return if stable_url_version < Version.new("1.0") | 
					
						
							| 
									
										
										
										
											2021-03-29 16:37:23 -04:00
										 |  |  |         # All minor versions are stable in the new GNOME version scheme (which starts at version 40.0) | 
					
						
							|  |  |  |         # https://discourse.gnome.org/t/new-gnome-versioning-scheme/4235 | 
					
						
							| 
									
										
										
										
											2023-07-06 16:47:09 +01:00
										 |  |  |         return if stable_url_version >= Version.new("40.0") | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         return if stable_url_minor_version.even? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         problem "#{stable.version} is a development release" | 
					
						
							|  |  |  |       when %r{isc.org/isc/bind\d*/}i | 
					
						
							|  |  |  |         return if stable_url_minor_version.even? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         problem "#{stable.version} is a development release" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       when %r{https?://gitlab\.com/([\w-]+)/([\w-]+)} | 
					
						
							|  |  |  |         owner = Regexp.last_match(1) | 
					
						
							|  |  |  |         repo = Regexp.last_match(2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         tag = SharedAudits.gitlab_tag_from_url(url) | 
					
						
							|  |  |  |         tag ||= stable.specs[:tag] | 
					
						
							|  |  |  |         tag ||= stable.version | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if @online | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |           error = SharedAudits.gitlab_release(owner, repo, tag, formula:) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |           problem error if error | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       when %r{^https://github.com/([\w-]+)/([\w-]+)} | 
					
						
							|  |  |  |         owner = Regexp.last_match(1) | 
					
						
							|  |  |  |         repo = Regexp.last_match(2) | 
					
						
							|  |  |  |         tag = SharedAudits.github_tag_from_url(url) | 
					
						
							|  |  |  |         tag ||= formula.stable.specs[:tag] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if @online | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |           error = SharedAudits.github_release(owner, repo, tag, formula:) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |           problem error if error | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-13 14:53:35 -05:00
										 |  |  |     def audit_stable_version | 
					
						
							|  |  |  |       return unless @git | 
					
						
							|  |  |  |       return unless formula.tap # skip formula not from core or any taps | 
					
						
							|  |  |  |       return unless formula.tap.git? # git log is required | 
					
						
							|  |  |  |       return if formula.stable.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       current_version = formula.stable.version | 
					
						
							|  |  |  |       current_version_scheme = formula.version_scheme | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-13 17:35:02 -05:00
										 |  |  |       previous_committed, newest_committed = committed_version_info | 
					
						
							| 
									
										
										
										
											2023-12-13 14:53:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-13 17:35:02 -05:00
										 |  |  |       if !newest_committed[:version].nil? && | 
					
						
							|  |  |  |          current_version < newest_committed[:version] && | 
					
						
							|  |  |  |          current_version_scheme == previous_committed[:version_scheme] | 
					
						
							|  |  |  |         problem "stable version should not decrease (from #{newest_committed[:version]} to #{current_version})" | 
					
						
							| 
									
										
										
										
											2023-12-13 14:53:35 -05:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:18:00 +00:00
										 |  |  |     def audit_revision | 
					
						
							| 
									
										
										
										
											2022-02-09 04:24:58 -05:00
										 |  |  |       new_formula_problem("New formulae should not define a revision.") if @new_formula && !formula.revision.zero? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       return unless @git | 
					
						
							|  |  |  |       return unless formula.tap # skip formula not from core or any taps | 
					
						
							|  |  |  |       return unless formula.tap.git? # git log is required | 
					
						
							|  |  |  |       return if formula.stable.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       current_version = formula.stable.version | 
					
						
							|  |  |  |       current_revision = formula.revision | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-13 17:35:02 -05:00
										 |  |  |       previous_committed, newest_committed = committed_version_info | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-13 17:35:02 -05:00
										 |  |  |       if (previous_committed[:version] != newest_committed[:version] || | 
					
						
							|  |  |  |          current_version != newest_committed[:version]) && | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |          !current_revision.zero? && | 
					
						
							| 
									
										
										
										
											2023-12-13 17:35:02 -05:00
										 |  |  |          current_revision == newest_committed[:revision] && | 
					
						
							|  |  |  |          current_revision == previous_committed[:revision] | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         problem "'revision #{current_revision}' should be removed" | 
					
						
							| 
									
										
										
										
											2023-12-13 17:35:02 -05:00
										 |  |  |       elsif current_version == previous_committed[:version] && | 
					
						
							|  |  |  |             !previous_committed[:revision].nil? && | 
					
						
							|  |  |  |             current_revision < previous_committed[:revision] | 
					
						
							|  |  |  |         problem "revision should not decrease (from #{previous_committed[:revision]} to #{current_revision})" | 
					
						
							|  |  |  |       elsif newest_committed[:revision] && | 
					
						
							|  |  |  |             current_revision > (newest_committed[:revision] + 1) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         problem "revisions should only increment by 1" | 
					
						
							| 
									
										
										
										
											2024-01-12 14:18:00 +00:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_version_scheme | 
					
						
							|  |  |  |       return unless @git | 
					
						
							|  |  |  |       return unless formula.tap # skip formula not from core or any taps | 
					
						
							|  |  |  |       return unless formula.tap.git? # git log is required | 
					
						
							|  |  |  |       return if formula.stable.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       current_version_scheme = formula.version_scheme | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       previous_committed, = committed_version_info | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return if previous_committed[:version_scheme].nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if current_version_scheme < previous_committed[:version_scheme] | 
					
						
							|  |  |  |         problem "version_scheme should not decrease (from #{previous_committed[:version_scheme]} " \ | 
					
						
							|  |  |  |                 "to #{current_version_scheme})" | 
					
						
							|  |  |  |       elsif current_version_scheme > (previous_committed[:version_scheme] + 1) | 
					
						
							|  |  |  |         problem "version_schemes should only increment by 1" | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-19 13:19:40 -05:00
										 |  |  |     def audit_unconfirmed_checksum_change | 
					
						
							|  |  |  |       return unless @git | 
					
						
							|  |  |  |       return unless formula.tap # skip formula not from core or any taps | 
					
						
							|  |  |  |       return unless formula.tap.git? # git log is required | 
					
						
							|  |  |  |       return if formula.stable.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       current_version = formula.stable.version | 
					
						
							|  |  |  |       current_checksum = formula.stable.checksum | 
					
						
							|  |  |  |       current_url = formula.stable.url | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       _, newest_committed = committed_version_info | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if current_version == newest_committed[:version] && | 
					
						
							|  |  |  |          current_url == newest_committed[:url] && | 
					
						
							|  |  |  |          current_checksum != newest_committed[:checksum] && | 
					
						
							|  |  |  |          current_checksum.present? && newest_committed[:checksum].present? | 
					
						
							|  |  |  |         problem( | 
					
						
							|  |  |  |           "stable sha256 changed without the url/version also changing; " \ | 
					
						
							|  |  |  |           "please create an issue upstream to rule out malicious " \ | 
					
						
							|  |  |  |           "circumstances and to find out why the file changed.", | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |     def audit_text | 
					
						
							|  |  |  |       bin_names = Set.new | 
					
						
							|  |  |  |       bin_names << formula.name | 
					
						
							|  |  |  |       bin_names += formula.aliases | 
					
						
							|  |  |  |       [formula.bin, formula.sbin].each do |dir| | 
					
						
							|  |  |  |         next unless dir.exist? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-14 02:52:30 +00:00
										 |  |  |         bin_names += dir.children.map { |child| child.basename.to_s } | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |       end | 
					
						
							|  |  |  |       shell_commands = ["system", "shell_output", "pipe_output"] | 
					
						
							|  |  |  |       bin_names.each do |name| | 
					
						
							|  |  |  |         shell_commands.each do |cmd| | 
					
						
							|  |  |  |           if text.to_s.match?(/test do.*#{cmd}[(\s]+['"]#{Regexp.escape(name)}[\s'"]/m) | 
					
						
							|  |  |  |             problem %Q(fully scope test #{cmd} calls, e.g. #{cmd} "\#{bin}/#{name}") | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_reverse_migration | 
					
						
							|  |  |  |       # Only enforce for new formula being re-added to core | 
					
						
							|  |  |  |       return unless @strict | 
					
						
							|  |  |  |       return unless @core_tap | 
					
						
							|  |  |  |       return unless formula.tap.tap_migrations.key?(formula.name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem <<~EOS | 
					
						
							|  |  |  |         #{formula.name} seems to be listed in tap_migrations.json! | 
					
						
							|  |  |  |         Please remove #{formula.name} from present tap & tap_migrations.json | 
					
						
							|  |  |  |         before submitting it to Homebrew/homebrew-#{formula.tap.repo}. | 
					
						
							|  |  |  |       EOS | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit_prefix_has_contents | 
					
						
							|  |  |  |       return unless formula.prefix.directory? | 
					
						
							|  |  |  |       return unless Keg.new(formula.prefix).empty_installation? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       problem <<~EOS | 
					
						
							|  |  |  |         The installation seems to be empty. Please ensure the prefix | 
					
						
							|  |  |  |         is set correctly and expected files are installed. | 
					
						
							|  |  |  |         The prefix configure/make argument may be case-sensitive. | 
					
						
							|  |  |  |       EOS | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def quote_dep(dep) | 
					
						
							|  |  |  |       dep.is_a?(Symbol) ? dep.inspect : "'#{dep}'" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def problem_if_output(output) | 
					
						
							|  |  |  |       problem(output) if output | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def audit | 
					
						
							|  |  |  |       only_audits = @only | 
					
						
							|  |  |  |       except_audits = @except | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       methods.map(&:to_s).grep(/^audit_/).each do |audit_method_name| | 
					
						
							|  |  |  |         name = audit_method_name.delete_prefix("audit_") | 
					
						
							| 
									
										
										
										
											2021-06-15 09:55:28 -04:00
										 |  |  |         next if only_audits&.exclude?(name) | 
					
						
							|  |  |  |         next if except_audits&.include?(name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |         send(audit_method_name) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-19 16:59:14 +02:00
										 |  |  |     def problem(message, location: nil, corrected: false) | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |       @problems << ({ message:, location:, corrected: }) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-19 16:59:14 +02:00
										 |  |  |     def new_formula_problem(message, location: nil, corrected: false) | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |       @new_formula_problems << ({ message:, location:, corrected: }) | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def head_only?(formula) | 
					
						
							|  |  |  |       formula.head && formula.stable.nil? | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2022-08-22 14:54:52 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def linux_only_gcc_dep?(formula) | 
					
						
							| 
									
										
										
										
											2022-08-24 13:22:00 +08:00
										 |  |  |       odie "`#linux_only_gcc_dep?` works only on Linux!" if Homebrew::SimulateSystem.simulating_or_running_on_macos? | 
					
						
							| 
									
										
										
										
											2022-08-25 20:44:27 +08:00
										 |  |  |       return false if formula.deps.map(&:name).exclude?("gcc") | 
					
						
							| 
									
										
										
										
											2022-08-22 14:54:52 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-25 20:44:27 +08:00
										 |  |  |       variations = formula.to_hash_with_variations["variations"] | 
					
						
							| 
									
										
										
										
											2022-08-24 15:12:58 +08:00
										 |  |  |       # The formula has no variations, so all OS-version-arch triples depend on GCC. | 
					
						
							| 
									
										
										
										
											2022-08-24 15:09:53 +08:00
										 |  |  |       return false if variations.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-13 22:35:08 +02:00
										 |  |  |       MacOSVersion::SYMBOLS.keys.product(OnSystem::ARCH_OPTIONS).each do |os, arch| | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |         bottle_tag = Utils::Bottles::Tag.new(system: os, arch:) | 
					
						
							| 
									
										
										
										
											2023-05-13 22:35:08 +02:00
										 |  |  |         next unless bottle_tag.valid_combination? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         variation_dependencies = variations.dig(bottle_tag.to_sym, "dependencies") | 
					
						
							|  |  |  |         # This variation either: | 
					
						
							|  |  |  |         #   1. does not exist | 
					
						
							|  |  |  |         #   2. has no variation-specific dependencies | 
					
						
							|  |  |  |         # In either case, it matches Linux. We must check for `nil` because an empty | 
					
						
							|  |  |  |         # array indicates that this variation does not depend on GCC. | 
					
						
							|  |  |  |         return false if variation_dependencies.nil? | 
					
						
							|  |  |  |         # We found a non-Linux variation that depends on GCC. | 
					
						
							|  |  |  |         return false if variation_dependencies.include?("gcc") | 
					
						
							| 
									
										
										
										
											2022-08-24 20:34:43 +08:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2022-08-22 14:54:52 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-24 20:34:43 +08:00
										 |  |  |       true | 
					
						
							| 
									
										
										
										
											2022-08-22 14:54:52 +08:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2023-12-13 17:35:02 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def committed_version_info | 
					
						
							|  |  |  |       return [] unless @git | 
					
						
							|  |  |  |       return [] unless formula.tap # skip formula not from core or any taps | 
					
						
							|  |  |  |       return [] unless formula.tap.git? # git log is required | 
					
						
							|  |  |  |       return [] if formula.stable.blank? | 
					
						
							|  |  |  |       return [@previous_committed, @newest_committed] if @previous_committed.present? || @newest_committed.present? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       current_version = formula.stable.version | 
					
						
							|  |  |  |       current_revision = formula.revision | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       fv = FormulaVersions.new(formula) | 
					
						
							|  |  |  |       fv.rev_list("origin/HEAD") do |revision, path| | 
					
						
							|  |  |  |         begin | 
					
						
							|  |  |  |           fv.formula_at_revision(revision, path) do |f| | 
					
						
							|  |  |  |             stable = f.stable | 
					
						
							|  |  |  |             next if stable.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             @previous_committed[:version] = stable.version | 
					
						
							|  |  |  |             @previous_committed[:checksum] = stable.checksum | 
					
						
							|  |  |  |             @previous_committed[:version_scheme] = f.version_scheme | 
					
						
							|  |  |  |             @previous_committed[:revision] = f.revision | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             @newest_committed[:version] ||= @previous_committed[:version] | 
					
						
							|  |  |  |             @newest_committed[:checksum] ||= @previous_committed[:checksum] | 
					
						
							|  |  |  |             @newest_committed[:revision] ||= @previous_committed[:revision] | 
					
						
							|  |  |  |             @newest_committed[:url] ||= stable.url | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         rescue MacOSVersion::Error | 
					
						
							|  |  |  |           break | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         break if @previous_committed[:version] && current_version != @previous_committed[:version] | 
					
						
							|  |  |  |         break if @previous_committed[:revision] && current_revision != @previous_committed[:revision] | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       @previous_committed.compact! | 
					
						
							|  |  |  |       @newest_committed.compact! | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       [@previous_committed, @newest_committed] | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2020-11-18 10:25:12 +01:00
										 |  |  |   end | 
					
						
							|  |  |  | end |