| 
									
										
										
										
											2024-08-12 10:30:59 +01:00
										 |  |  | # typed: true # rubocop:todo Sorbet/StrictSigil | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  | require "shellwords" | 
					
						
							| 
									
										
										
										
											2023-05-19 19:43:15 +02:00
										 |  |  | require "source_location" | 
					
						
							| 
									
										
										
										
											2024-01-26 17:33:55 -08:00
										 |  |  | require "system_command" | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  | module Homebrew | 
					
						
							| 
									
										
										
										
											2020-08-19 07:26:09 +02:00
										 |  |  |   # Helper module for running RuboCop. | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |   module Style | 
					
						
							| 
									
										
										
										
											2024-01-26 17:33:55 -08:00
										 |  |  |     extend SystemCommand::Mixin | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |     # Checks style for a list of files, printing simple RuboCop output. | 
					
						
							|  |  |  |     # Returns true if violations were found, false otherwise. | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.check_style_and_print(files, **options) | 
					
						
							| 
									
										
										
										
											2020-09-01 20:07:21 +02:00
										 |  |  |       success = check_style_impl(files, :print, **options) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-09 13:19:14 +01:00
										 |  |  |       if GitHub::Actions.env_set? && !success | 
					
						
							| 
									
										
										
										
											2020-09-09 20:41:29 +02:00
										 |  |  |         check_style_json(files, **options).each do |path, offenses| | 
					
						
							|  |  |  |           offenses.each do |o| | 
					
						
							|  |  |  |             line = o.location.line | 
					
						
							|  |  |  |             column = o.location.line | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |             annotation = GitHub::Actions::Annotation.new(:error, o.message, file: path, line:, column:) | 
					
						
							| 
									
										
										
										
											2020-09-09 20:41:29 +02:00
										 |  |  |             puts annotation if annotation.relevant? | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-09-01 20:07:21 +02:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       success | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-05 17:17:03 -05:00
										 |  |  |     # Checks style for a list of files, returning results as an {Offenses} | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |     # object parsed from its JSON output. | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.check_style_json(files, **options) | 
					
						
							| 
									
										
										
										
											2019-10-04 23:39:11 +02:00
										 |  |  |       check_style_impl(files, :json, **options) | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.check_style_impl(files, output_type, | 
					
						
							| 
									
										
										
										
											2023-03-15 18:21:41 -07:00
										 |  |  |                               fix: false, | 
					
						
							|  |  |  |                               except_cops: nil, only_cops: nil, | 
					
						
							|  |  |  |                               display_cop_names: false, | 
					
						
							|  |  |  |                               reset_cache: false, | 
					
						
							|  |  |  |                               debug: false, verbose: false) | 
					
						
							| 
									
										
										
										
											2021-09-30 11:06:04 +01:00
										 |  |  |       raise ArgumentError, "Invalid output type: #{output_type.inspect}" if [:print, :json].exclude?(output_type) | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-12 16:04:17 +01:00
										 |  |  |       ruby_files = T.let([], T::Array[Pathname]) | 
					
						
							|  |  |  |       shell_files = T.let([], T::Array[Pathname]) | 
					
						
							|  |  |  |       actionlint_files = T.let([], T::Array[Pathname]) | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |       Array(files).map(&method(:Pathname)) | 
					
						
							|  |  |  |                   .each do |path| | 
					
						
							|  |  |  |         case path.extname | 
					
						
							|  |  |  |         when ".rb" | 
					
						
							|  |  |  |           ruby_files << path | 
					
						
							|  |  |  |         when ".sh" | 
					
						
							|  |  |  |           shell_files << path | 
					
						
							|  |  |  |         when ".yml" | 
					
						
							|  |  |  |           actionlint_files << path if path.realpath.to_s.include?("/.github/workflows/") | 
					
						
							|  |  |  |         else | 
					
						
							| 
									
										
										
										
											2024-06-12 16:04:17 +01:00
										 |  |  |           ruby_files << path | 
					
						
							|  |  |  |           shell_files += if [HOMEBREW_PREFIX, HOMEBREW_REPOSITORY].include?(path) | 
					
						
							|  |  |  |             shell_scripts | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             path.glob("**/*.sh") | 
					
						
							| 
									
										
										
										
											2024-06-14 16:08:49 +01:00
										 |  |  |                 .reject { |path| path.to_s.include?("/vendor/") || path.directory? } | 
					
						
							| 
									
										
										
										
											2024-06-12 16:04:17 +01:00
										 |  |  |           end | 
					
						
							|  |  |  |           actionlint_files += (path/".github/workflows").glob("*.y{,a}ml") | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |       rubocop_result = if files.present? && ruby_files.empty? | 
					
						
							| 
									
										
										
										
											2022-08-10 14:19:33 +01:00
										 |  |  |         (output_type == :json) ? [] : true | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |       else | 
					
						
							|  |  |  |         run_rubocop(ruby_files, output_type, | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |                     fix:, | 
					
						
							|  |  |  |                     except_cops:, only_cops:, | 
					
						
							|  |  |  |                     display_cop_names:, | 
					
						
							|  |  |  |                     reset_cache:, | 
					
						
							|  |  |  |                     debug:, verbose:) | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |       shellcheck_result = if files.present? && shell_files.empty? | 
					
						
							| 
									
										
										
										
											2022-08-10 14:19:33 +01:00
										 |  |  |         (output_type == :json) ? [] : true | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |         run_shellcheck(shell_files, output_type, fix:) | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |       shfmt_result = if files.present? && shell_files.empty? | 
					
						
							| 
									
										
										
										
											2021-09-16 23:42:30 +08:00
										 |  |  |         true | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |         run_shfmt(shell_files, fix:) | 
					
						
							| 
									
										
										
										
											2021-09-16 23:42:30 +08:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2021-09-15 14:58:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-27 15:29:07 +08:00
										 |  |  |       has_actionlint_workflow = actionlint_files.any? do |path| | 
					
						
							|  |  |  |         path.to_s.end_with?("/.github/workflows/actionlint.yml") | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |       odebug "actionlint workflow detected. Skipping actionlint checks." if has_actionlint_workflow | 
					
						
							|  |  |  |       actionlint_result = if files.present? && (has_actionlint_workflow || actionlint_files.empty?) | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |         true | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         run_actionlint(actionlint_files) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |       if output_type == :json | 
					
						
							| 
									
										
										
										
											2020-09-01 20:07:21 +02:00
										 |  |  |         Offenses.new(rubocop_result + shellcheck_result) | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |         rubocop_result && shellcheck_result && shfmt_result && actionlint_result | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-09 18:59:29 -05:00
										 |  |  |     RUBOCOP = (HOMEBREW_LIBRARY_PATH/"utils/rubocop.rb").freeze | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.run_rubocop(files, output_type, | 
					
						
							| 
									
										
										
										
											2023-03-15 18:21:41 -07:00
										 |  |  |                          fix: false, except_cops: nil, only_cops: nil, display_cop_names: false, reset_cache: false, | 
					
						
							|  |  |  |                          debug: false, verbose: false) | 
					
						
							| 
									
										
										
										
											2021-02-14 11:56:32 -05:00
										 |  |  |       require "warnings" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       Warnings.ignore :parser_syntax do | 
					
						
							|  |  |  |         require "rubocop" | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 16:07:47 +01:00
										 |  |  |       require "rubocops/all" | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |       args = %w[
 | 
					
						
							|  |  |  |         --force-exclusion | 
					
						
							|  |  |  |       ] | 
					
						
							| 
									
										
										
										
											2020-03-13 21:15:06 +00:00
										 |  |  |       args << if fix | 
					
						
							| 
									
										
										
										
											2022-06-16 22:03:20 +01:00
										 |  |  |         "--autocorrect-all" | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2020-03-13 21:15:06 +00:00
										 |  |  |         "--parallel" | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-02 03:36:09 +02:00
										 |  |  |       args += ["--extra-details"] if verbose | 
					
						
							| 
									
										
										
										
											2018-07-08 20:08:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-04 23:39:11 +02:00
										 |  |  |       if except_cops | 
					
						
							| 
									
										
										
										
											2024-07-10 13:05:31 -07:00
										 |  |  |         except_cops.map! { |cop| RuboCop::Cop::Registry.global.qualified_cop_name(cop.to_s, "") } | 
					
						
							| 
									
										
										
										
											2019-10-04 23:39:11 +02:00
										 |  |  |         cops_to_exclude = except_cops.select do |cop| | 
					
						
							| 
									
										
										
										
											2024-07-10 13:05:31 -07:00
										 |  |  |           RuboCop::Cop::Registry.global.names.include?(cop) || | 
					
						
							|  |  |  |             RuboCop::Cop::Registry.global.departments.include?(cop.to_sym) | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         args << "--except" << cops_to_exclude.join(",") unless cops_to_exclude.empty? | 
					
						
							| 
									
										
										
										
											2019-10-04 23:39:11 +02:00
										 |  |  |       elsif only_cops | 
					
						
							| 
									
										
										
										
											2024-07-10 13:05:31 -07:00
										 |  |  |         only_cops.map! { |cop| RuboCop::Cop::Registry.global.qualified_cop_name(cop.to_s, "") } | 
					
						
							| 
									
										
										
										
											2019-10-04 23:39:11 +02:00
										 |  |  |         cops_to_include = only_cops.select do |cop| | 
					
						
							| 
									
										
										
										
											2024-07-10 13:05:31 -07:00
										 |  |  |           RuboCop::Cop::Registry.global.names.include?(cop) || | 
					
						
							|  |  |  |             RuboCop::Cop::Registry.global.departments.include?(cop.to_sym) | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-04 23:39:11 +02:00
										 |  |  |         odie "RuboCops #{only_cops.join(",")} were not found" if cops_to_include.empty? | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         args << "--only" << cops_to_include.join(",") | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 11:06:04 +01:00
										 |  |  |       files&.map!(&:expand_path) | 
					
						
							| 
									
										
										
										
											2024-06-30 01:21:54 +01:00
										 |  |  |       base_dir = Dir.pwd | 
					
						
							| 
									
										
										
										
											2021-09-30 11:06:04 +01:00
										 |  |  |       if files.blank? || files == [HOMEBREW_REPOSITORY] | 
					
						
							|  |  |  |         files = [HOMEBREW_LIBRARY_PATH] | 
					
						
							| 
									
										
										
										
											2024-06-30 01:21:54 +01:00
										 |  |  |         base_dir = HOMEBREW_LIBRARY_PATH | 
					
						
							| 
									
										
										
										
											2024-06-13 12:31:47 +01:00
										 |  |  |       elsif files.any? { |f| f.to_s.start_with?(HOMEBREW_REPOSITORY/"docs") || (f.basename.to_s == "docs") } | 
					
						
							|  |  |  |         args << "--config" << (HOMEBREW_REPOSITORY/"docs/docs_rubocop_style.yml") | 
					
						
							| 
									
										
										
										
											2024-06-30 01:21:54 +01:00
										 |  |  |       elsif files.any? { |f| f.to_s.start_with? HOMEBREW_LIBRARY_PATH } | 
					
						
							|  |  |  |         base_dir = HOMEBREW_LIBRARY_PATH | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2023-02-21 16:32:32 -08:00
										 |  |  |         args << "--config" << (HOMEBREW_LIBRARY/".rubocop.yml") | 
					
						
							| 
									
										
										
										
											2024-06-30 01:21:54 +01:00
										 |  |  |         base_dir = HOMEBREW_LIBRARY if files.any? { |f| f.to_s.start_with? HOMEBREW_LIBRARY } | 
					
						
							| 
									
										
										
										
											2018-10-04 18:09:23 +02:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-06-11 15:10:59 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 11:06:04 +01:00
										 |  |  |       args += files | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-14 18:57:45 +02:00
										 |  |  |       HOMEBREW_CACHE.mkpath | 
					
						
							|  |  |  |       cache_dir = HOMEBREW_CACHE.realpath | 
					
						
							|  |  |  |       cache_env = { "XDG_CACHE_HOME" => "#{cache_dir}/style" } | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-29 14:21:06 -05:00
										 |  |  |       FileUtils.rm_rf cache_env["XDG_CACHE_HOME"] if reset_cache | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-05 03:05:58 +00:00
										 |  |  |       ruby_args = HOMEBREW_RUBY_EXEC_ARGS.dup | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |       case output_type | 
					
						
							|  |  |  |       when :print | 
					
						
							| 
									
										
										
										
											2020-08-02 03:36:09 +02:00
										 |  |  |         args << "--debug" if debug | 
					
						
							| 
									
										
										
										
											2020-09-02 02:02:01 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-07 18:39:58 +02:00
										 |  |  |         # Don't show the default formatter's progress dots | 
					
						
							|  |  |  |         # on CI or if only checking a single file. | 
					
						
							|  |  |  |         args << "--format" << "clang" if ENV["CI"] || files.count { |f| !f.directory? } == 1
 | 
					
						
							| 
									
										
										
										
											2020-09-02 02:02:01 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-01 18:31:09 +02:00
										 |  |  |         args << "--color" if Tty.color? | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 01:21:54 +01:00
										 |  |  |         system cache_env, *ruby_args, "--", RUBOCOP, *args, chdir: base_dir | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |         $CHILD_STATUS.success? | 
					
						
							|  |  |  |       when :json | 
					
						
							| 
									
										
										
										
											2022-11-05 03:05:58 +00:00
										 |  |  |         result = system_command ruby_args.shift, | 
					
						
							| 
									
										
										
										
											2024-06-30 01:21:54 +01:00
										 |  |  |                                 args:  [*ruby_args, "--", RUBOCOP, "--format", "json", *args], | 
					
						
							|  |  |  |                                 env:   cache_env, | 
					
						
							|  |  |  |                                 chdir: base_dir | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |         json = json_result!(result) | 
					
						
							| 
									
										
										
										
											2024-06-30 01:21:54 +01:00
										 |  |  |         json["files"].each do |file| | 
					
						
							|  |  |  |           file["path"] = File.absolute_path(file["path"], base_dir) | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-10-03 15:52:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.run_shellcheck(files, output_type, fix: false) | 
					
						
							| 
									
										
										
										
											2021-09-16 19:35:03 +08:00
										 |  |  |       files = shell_scripts if files.blank? | 
					
						
							| 
									
										
										
										
											2018-10-03 15:52:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-08 22:33:53 +08:00
										 |  |  |       files = files.map(&:realpath) # use absolute file paths | 
					
						
							| 
									
										
										
										
											2021-11-07 21:00:01 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-08 03:11:39 +00:00
										 |  |  |       args = [ | 
					
						
							|  |  |  |         "--shell=bash", | 
					
						
							|  |  |  |         "--enable=all", | 
					
						
							|  |  |  |         "--external-sources", | 
					
						
							|  |  |  |         "--source-path=#{HOMEBREW_LIBRARY}", | 
					
						
							|  |  |  |         "--", | 
					
						
							|  |  |  |         *files, | 
					
						
							|  |  |  |       ] | 
					
						
							| 
									
										
										
										
											2021-11-07 21:00:01 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if fix | 
					
						
							| 
									
										
										
										
											2021-11-08 22:33:53 +08:00
										 |  |  |         # patch options: | 
					
						
							| 
									
										
										
										
											2022-01-03 22:08:49 +08:00
										 |  |  |         #   -g 0 (--get=0)       : suppress environment variable `PATCH_GET` | 
					
						
							|  |  |  |         #   -f   (--force)       : we know what we are doing, force apply patches | 
					
						
							|  |  |  |         #   -d / (--directory=/) : change to root directory, since we use absolute file paths | 
					
						
							|  |  |  |         #   -p0  (--strip=0)     : do not strip path prefixes, since we are at root directory | 
					
						
							| 
									
										
										
										
											2024-04-26 20:55:51 +02:00
										 |  |  |         # NOTE: We use short flags for compatibility. | 
					
						
							| 
									
										
										
										
											2022-01-03 22:08:49 +08:00
										 |  |  |         patch_command = %w[patch -g 0 -f -d / -p0] | 
					
						
							| 
									
										
										
										
											2021-11-08 20:10:30 +08:00
										 |  |  |         patches = system_command(shellcheck, args: ["--format=diff", *args]).stdout | 
					
						
							| 
									
										
										
										
											2022-01-03 22:08:49 +08:00
										 |  |  |         Utils.safe_popen_write(*patch_command) { |p| p.write(patches) } if patches.present? | 
					
						
							| 
									
										
										
										
											2021-11-07 21:00:01 +08:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |       case output_type | 
					
						
							|  |  |  |       when :print | 
					
						
							|  |  |  |         system shellcheck, "--format=tty", *args | 
					
						
							|  |  |  |         $CHILD_STATUS.success? | 
					
						
							|  |  |  |       when :json | 
					
						
							| 
									
										
										
										
											2020-09-01 20:07:21 +02:00
										 |  |  |         result = system_command shellcheck, args: ["--format=json", *args] | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |         json = json_result!(result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Convert to same format as RuboCop offenses. | 
					
						
							| 
									
										
										
										
											2020-09-11 10:29:21 +01:00
										 |  |  |         severity_hash = { "style" => "refactor", "info" => "convention" } | 
					
						
							| 
									
										
										
										
											2020-09-01 20:07:21 +02:00
										 |  |  |         json.group_by { |v| v["file"] } | 
					
						
							|  |  |  |             .map do |k, v| | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |           { | 
					
						
							|  |  |  |             "path"     => k, | 
					
						
							|  |  |  |             "offenses" => v.map do |o| | 
					
						
							|  |  |  |               o.delete("file") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               o["cop_name"] = "SC#{o.delete("code")}" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               level = o.delete("level") | 
					
						
							| 
									
										
										
										
											2020-09-11 10:29:21 +01:00
										 |  |  |               o["severity"] = severity_hash.fetch(level, level) | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |               line = o.delete("line") | 
					
						
							|  |  |  |               column = o.delete("column") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               o["corrected"] = false | 
					
						
							|  |  |  |               o["correctable"] = o.delete("fix").present? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               o["location"] = { | 
					
						
							|  |  |  |                 "start_line"   => line, | 
					
						
							|  |  |  |                 "start_column" => column, | 
					
						
							|  |  |  |                 "last_line"    => o.delete("endLine"), | 
					
						
							|  |  |  |                 "last_column"  => o.delete("endColumn"), | 
					
						
							|  |  |  |                 "line"         => line, | 
					
						
							|  |  |  |                 "column"       => column, | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               o | 
					
						
							|  |  |  |             end, | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.run_shfmt(files, fix: false) | 
					
						
							| 
									
										
										
										
											2021-09-16 19:35:03 +08:00
										 |  |  |       files = shell_scripts if files.blank? | 
					
						
							| 
									
										
										
										
											2021-09-15 22:31:56 +08:00
										 |  |  |       # Do not format completions and Dockerfile | 
					
						
							|  |  |  |       files.delete(HOMEBREW_REPOSITORY/"completions/bash/brew") | 
					
						
							|  |  |  |       files.delete(HOMEBREW_REPOSITORY/"Dockerfile") | 
					
						
							| 
									
										
										
										
											2021-09-15 14:58:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-15 18:28:41 -04:00
										 |  |  |       args = ["--language-dialect", "bash", "--indent", "2", "--case-indent", "--", *files] | 
					
						
							|  |  |  |       args.unshift("--write") if fix # need to add before "--" | 
					
						
							| 
									
										
										
										
											2021-09-15 22:31:56 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 14:58:09 +08:00
										 |  |  |       system shfmt, *args | 
					
						
							|  |  |  |       $CHILD_STATUS.success? | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |     def self.run_actionlint(files) | 
					
						
							|  |  |  |       files = github_workflow_files if files.blank? | 
					
						
							| 
									
										
										
										
											2024-06-13 09:41:16 +01:00
										 |  |  |       # the ignore is to avoid false positives in e.g. actions, homebrew-test-bot | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |       system actionlint, "-shellcheck", shellcheck, | 
					
						
							|  |  |  |              "-config-file", HOMEBREW_REPOSITORY/".github/actionlint.yaml", | 
					
						
							| 
									
										
										
										
											2024-06-13 09:41:16 +01:00
										 |  |  |              "-ignore", "image: string; options: string", | 
					
						
							| 
									
										
										
										
											2024-06-14 17:28:06 +00:00
										 |  |  |              "-ignore", "label .* is unknown", | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |              *files | 
					
						
							|  |  |  |       $CHILD_STATUS.success? | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.json_result!(result) | 
					
						
							| 
									
										
										
										
											2020-08-30 19:08:24 +02:00
										 |  |  |       # An exit status of 1 just means violations were found; other numbers mean | 
					
						
							|  |  |  |       # execution errors. | 
					
						
							|  |  |  |       # JSON needs to be at least 2 characters. | 
					
						
							|  |  |  |       result.assert_success! if !(0..1).cover?(result.status.exitstatus) || result.stdout.length < 2
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       JSON.parse(result.stdout) | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.shell_scripts | 
					
						
							| 
									
										
										
										
											2021-09-15 14:58:09 +08:00
										 |  |  |       [ | 
					
						
							|  |  |  |         HOMEBREW_BREW_FILE, | 
					
						
							|  |  |  |         HOMEBREW_REPOSITORY/"completions/bash/brew", | 
					
						
							|  |  |  |         HOMEBREW_REPOSITORY/"Dockerfile", | 
					
						
							| 
									
										
										
										
											2022-11-09 17:29:38 +00:00
										 |  |  |         *HOMEBREW_REPOSITORY.glob(".devcontainer/**/*.sh"), | 
					
						
							| 
									
										
										
										
											2022-12-23 09:10:01 +01:00
										 |  |  |         *HOMEBREW_REPOSITORY.glob("package/scripts/*"), | 
					
						
							| 
									
										
										
										
											2022-11-09 17:29:38 +00:00
										 |  |  |         *HOMEBREW_LIBRARY.glob("Homebrew/**/*.sh").reject { |path| path.to_s.include?("/vendor/") }, | 
					
						
							| 
									
										
										
										
											2021-09-15 14:58:09 +08:00
										 |  |  |         *HOMEBREW_LIBRARY.glob("Homebrew/shims/**/*").map(&:realpath).uniq | 
					
						
							| 
									
										
										
										
											2022-10-08 18:57:40 +01:00
										 |  |  |                          .reject(&:directory?) | 
					
						
							|  |  |  |                          .reject { |path| path.basename.to_s == "cc" } | 
					
						
							|  |  |  |                          .select do |path| | 
					
						
							| 
									
										
										
										
											2022-11-09 17:29:38 +00:00
										 |  |  |                            %r{^#! ?/bin/(?:ba)?sh( |$)}.match?(path.read(13)) | 
					
						
							| 
									
										
										
										
											2022-10-08 18:57:40 +01:00
										 |  |  |                          end, | 
					
						
							| 
									
										
										
										
											2021-09-15 14:58:09 +08:00
										 |  |  |         *HOMEBREW_LIBRARY.glob("Homebrew/{dev-,}cmd/*.sh"), | 
					
						
							|  |  |  |         *HOMEBREW_LIBRARY.glob("Homebrew/{cask/,}utils/*.sh"), | 
					
						
							|  |  |  |       ] | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |     def self.github_workflow_files | 
					
						
							|  |  |  |       HOMEBREW_REPOSITORY.glob(".github/workflows/*.yml") | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def self.rubocop | 
					
						
							|  |  |  |       ensure_formula_installed!("rubocop", latest: true, | 
					
						
							|  |  |  |                                            reason: "Ruby style checks").opt_bin/"rubocop" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.shellcheck | 
					
						
							| 
									
										
										
										
											2021-11-23 23:25:32 +08:00
										 |  |  |       ensure_formula_installed!("shellcheck", latest: true, | 
					
						
							| 
									
										
										
										
											2021-11-23 23:59:09 +08:00
										 |  |  |                                               reason: "shell style checks").opt_bin/"shellcheck" | 
					
						
							| 
									
										
										
										
											2021-09-15 14:58:09 +08:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-15 14:29:15 -07:00
										 |  |  |     def self.shfmt | 
					
						
							| 
									
										
										
										
											2021-11-23 23:25:32 +08:00
										 |  |  |       ensure_formula_installed!("shfmt", latest: true, | 
					
						
							| 
									
										
										
										
											2021-11-23 23:59:09 +08:00
										 |  |  |                                          reason: "formatting shell scripts") | 
					
						
							| 
									
										
										
										
											2021-09-15 14:58:09 +08:00
										 |  |  |       HOMEBREW_LIBRARY/"Homebrew/utils/shfmt.sh" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-24 17:18:41 +01:00
										 |  |  |     def self.actionlint | 
					
						
							|  |  |  |       ensure_formula_installed!("actionlint", latest: true, | 
					
						
							|  |  |  |                                               reason: "GitHub Actions checks").opt_bin/"actionlint" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-01 20:07:21 +02:00
										 |  |  |     # Collection of style offenses. | 
					
						
							|  |  |  |     class Offenses | 
					
						
							|  |  |  |       include Enumerable | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def initialize(paths) | 
					
						
							|  |  |  |         @offenses = {} | 
					
						
							|  |  |  |         paths.each do |f| | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |           next if f["offenses"].empty? | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-01 20:07:21 +02:00
										 |  |  |           path = Pathname(f["path"]).realpath | 
					
						
							|  |  |  |           @offenses[path] = f["offenses"].map { |x| Offense.new(x) } | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-01 20:07:21 +02:00
										 |  |  |       def for_path(path) | 
					
						
							|  |  |  |         @offenses.fetch(Pathname(path), []) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def each(*args, &block) | 
					
						
							|  |  |  |         @offenses.each(*args, &block) | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-01 20:07:21 +02:00
										 |  |  |     # A style offense. | 
					
						
							|  |  |  |     class Offense | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |       attr_reader :severity, :message, :corrected, :location, :cop_name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def initialize(json) | 
					
						
							|  |  |  |         @severity = json["severity"] | 
					
						
							|  |  |  |         @message = json["message"] | 
					
						
							|  |  |  |         @cop_name = json["cop_name"] | 
					
						
							|  |  |  |         @corrected = json["corrected"] | 
					
						
							| 
									
										
										
										
											2023-05-19 19:43:15 +02:00
										 |  |  |         location = json["location"] | 
					
						
							|  |  |  |         @location = SourceLocation.new(location.fetch("line"), location["column"]) | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def severity_code | 
					
						
							|  |  |  |         @severity[0].upcase | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-26 19:33:19 +05:30
										 |  |  |       def corrected? | 
					
						
							|  |  |  |         @corrected | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-06-05 23:19:18 -04:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end |