rubocop 1.30.1 → 1.64.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (613) hide show
  1. checksums.yaml +4 -4
  2. data/LICENSE.txt +1 -1
  3. data/README.md +8 -6
  4. data/assets/output.css.erb +159 -0
  5. data/assets/output.html.erb +1 -160
  6. data/config/default.yml +584 -73
  7. data/config/obsoletion.yml +30 -1
  8. data/exe/rubocop +15 -7
  9. data/lib/rubocop/arguments_env.rb +17 -0
  10. data/lib/rubocop/arguments_file.rb +17 -0
  11. data/lib/rubocop/cache_config.rb +29 -0
  12. data/lib/rubocop/cached_data.rb +11 -3
  13. data/lib/rubocop/cli/command/{auto_genenerate_config.rb → auto_generate_config.rb} +30 -9
  14. data/lib/rubocop/cli/command/execute_runner.rb +14 -9
  15. data/lib/rubocop/cli/command/init_dotfile.rb +1 -1
  16. data/lib/rubocop/cli/command/lsp.rb +19 -0
  17. data/lib/rubocop/cli/command/show_docs_url.rb +2 -2
  18. data/lib/rubocop/cli/command/suggest_extensions.rb +61 -16
  19. data/lib/rubocop/cli.rb +70 -11
  20. data/lib/rubocop/comment_config.rb +60 -1
  21. data/lib/rubocop/config.rb +84 -28
  22. data/lib/rubocop/config_finder.rb +78 -0
  23. data/lib/rubocop/config_loader.rb +46 -68
  24. data/lib/rubocop/config_loader_resolver.rb +8 -8
  25. data/lib/rubocop/config_obsoletion/changed_parameter.rb +5 -0
  26. data/lib/rubocop/config_obsoletion/parameter_rule.rb +13 -1
  27. data/lib/rubocop/config_obsoletion.rb +15 -7
  28. data/lib/rubocop/config_validator.rb +16 -9
  29. data/lib/rubocop/cop/autocorrect_logic.rb +37 -14
  30. data/lib/rubocop/cop/badge.rb +9 -4
  31. data/lib/rubocop/cop/base.rb +181 -98
  32. data/lib/rubocop/cop/bundler/duplicated_gem.rb +2 -1
  33. data/lib/rubocop/cop/bundler/duplicated_group.rb +127 -0
  34. data/lib/rubocop/cop/bundler/gem_comment.rb +4 -4
  35. data/lib/rubocop/cop/bundler/gem_filename.rb +4 -4
  36. data/lib/rubocop/cop/bundler/gem_version.rb +5 -7
  37. data/lib/rubocop/cop/bundler/ordered_gems.rb +11 -3
  38. data/lib/rubocop/cop/commissioner.rb +19 -6
  39. data/lib/rubocop/cop/cop.rb +54 -34
  40. data/lib/rubocop/cop/corrector.rb +33 -13
  41. data/lib/rubocop/cop/correctors/alignment_corrector.rb +3 -3
  42. data/lib/rubocop/cop/correctors/each_to_for_corrector.rb +4 -8
  43. data/lib/rubocop/cop/correctors/for_to_each_corrector.rb +6 -14
  44. data/lib/rubocop/cop/correctors/lambda_literal_to_method_corrector.rb +7 -4
  45. data/lib/rubocop/cop/correctors/line_break_corrector.rb +1 -1
  46. data/lib/rubocop/cop/correctors/multiline_literal_brace_corrector.rb +26 -10
  47. data/lib/rubocop/cop/correctors/ordered_gem_corrector.rb +2 -7
  48. data/lib/rubocop/cop/correctors/parentheses_corrector.rb +58 -0
  49. data/lib/rubocop/cop/correctors/percent_literal_corrector.rb +2 -2
  50. data/lib/rubocop/cop/correctors/unused_arg_corrector.rb +1 -1
  51. data/lib/rubocop/cop/documentation.rb +16 -6
  52. data/lib/rubocop/cop/exclude_limit.rb +1 -1
  53. data/lib/rubocop/cop/gemspec/dependency_version.rb +22 -26
  54. data/lib/rubocop/cop/gemspec/deprecated_attribute_assignment.rb +33 -18
  55. data/lib/rubocop/cop/gemspec/development_dependencies.rb +107 -0
  56. data/lib/rubocop/cop/gemspec/duplicated_assignment.rb +1 -1
  57. data/lib/rubocop/cop/gemspec/ordered_dependencies.rb +10 -2
  58. data/lib/rubocop/cop/gemspec/require_mfa.rb +21 -21
  59. data/lib/rubocop/cop/gemspec/required_ruby_version.rb +5 -1
  60. data/lib/rubocop/cop/generator/require_file_injector.rb +3 -3
  61. data/lib/rubocop/cop/generator.rb +6 -3
  62. data/lib/rubocop/cop/internal_affairs/cop_description.rb +38 -12
  63. data/lib/rubocop/cop/internal_affairs/create_empty_file.rb +37 -0
  64. data/lib/rubocop/cop/internal_affairs/example_description.rb +46 -24
  65. data/lib/rubocop/cop/internal_affairs/example_heredoc_delimiter.rb +111 -0
  66. data/lib/rubocop/cop/internal_affairs/inherit_deprecated_cop_class.rb +1 -1
  67. data/lib/rubocop/cop/internal_affairs/lambda_or_proc.rb +46 -0
  68. data/lib/rubocop/cop/internal_affairs/location_expression.rb +37 -0
  69. data/lib/rubocop/cop/internal_affairs/location_line_equality_comparison.rb +3 -1
  70. data/lib/rubocop/cop/internal_affairs/method_name_end_with.rb +8 -6
  71. data/lib/rubocop/cop/internal_affairs/method_name_equal.rb +19 -20
  72. data/lib/rubocop/cop/internal_affairs/node_first_or_last_argument.rb +53 -0
  73. data/lib/rubocop/cop/internal_affairs/node_matcher_directive.rb +128 -38
  74. data/lib/rubocop/cop/internal_affairs/node_type_predicate.rb +1 -1
  75. data/lib/rubocop/cop/internal_affairs/numblock_handler.rb +69 -0
  76. data/lib/rubocop/cop/internal_affairs/processed_source_buffer_name.rb +42 -0
  77. data/lib/rubocop/cop/internal_affairs/redundant_expect_offense_arguments.rb +34 -0
  78. data/lib/rubocop/cop/internal_affairs/redundant_let_rubocop_config_new.rb +11 -3
  79. data/lib/rubocop/cop/internal_affairs/redundant_location_argument.rb +1 -1
  80. data/lib/rubocop/cop/internal_affairs/redundant_message_argument.rb +1 -1
  81. data/lib/rubocop/cop/internal_affairs/redundant_method_dispatch_node.rb +11 -2
  82. data/lib/rubocop/cop/internal_affairs/redundant_source_range.rb +66 -0
  83. data/lib/rubocop/cop/internal_affairs/single_line_comparison.rb +62 -0
  84. data/lib/rubocop/cop/internal_affairs/useless_message_assertion.rb +2 -0
  85. data/lib/rubocop/cop/internal_affairs/useless_restrict_on_send.rb +60 -0
  86. data/lib/rubocop/cop/internal_affairs.rb +11 -0
  87. data/lib/rubocop/cop/layout/argument_alignment.rb +1 -1
  88. data/lib/rubocop/cop/layout/array_alignment.rb +1 -1
  89. data/lib/rubocop/cop/layout/block_alignment.rb +16 -12
  90. data/lib/rubocop/cop/layout/block_end_newline.rb +31 -9
  91. data/lib/rubocop/cop/layout/class_structure.rb +44 -27
  92. data/lib/rubocop/cop/layout/closing_heredoc_indentation.rb +2 -3
  93. data/lib/rubocop/cop/layout/closing_parenthesis_indentation.rb +2 -6
  94. data/lib/rubocop/cop/layout/comment_indentation.rb +4 -2
  95. data/lib/rubocop/cop/layout/dot_position.rb +1 -5
  96. data/lib/rubocop/cop/layout/empty_comment.rb +6 -4
  97. data/lib/rubocop/cop/layout/empty_line_after_guard_clause.rb +42 -9
  98. data/lib/rubocop/cop/layout/empty_line_after_magic_comment.rb +14 -7
  99. data/lib/rubocop/cop/layout/empty_line_between_defs.rb +28 -5
  100. data/lib/rubocop/cop/layout/empty_lines.rb +3 -1
  101. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +7 -2
  102. data/lib/rubocop/cop/layout/empty_lines_around_attribute_accessor.rb +25 -4
  103. data/lib/rubocop/cop/layout/empty_lines_around_block_body.rb +2 -0
  104. data/lib/rubocop/cop/layout/empty_lines_around_exception_handling_keywords.rb +2 -0
  105. data/lib/rubocop/cop/layout/end_alignment.rb +23 -3
  106. data/lib/rubocop/cop/layout/end_of_line.rb +4 -4
  107. data/lib/rubocop/cop/layout/extra_spacing.rb +12 -9
  108. data/lib/rubocop/cop/layout/first_argument_indentation.rb +15 -4
  109. data/lib/rubocop/cop/layout/first_array_element_indentation.rb +44 -20
  110. data/lib/rubocop/cop/layout/first_array_element_line_break.rb +35 -8
  111. data/lib/rubocop/cop/layout/first_hash_element_indentation.rb +51 -12
  112. data/lib/rubocop/cop/layout/first_hash_element_line_break.rb +36 -1
  113. data/lib/rubocop/cop/layout/first_method_argument_line_break.rb +57 -8
  114. data/lib/rubocop/cop/layout/first_method_parameter_line_break.rb +52 -19
  115. data/lib/rubocop/cop/layout/first_parameter_indentation.rb +1 -1
  116. data/lib/rubocop/cop/layout/heredoc_argument_closing_parenthesis.rb +12 -6
  117. data/lib/rubocop/cop/layout/heredoc_indentation.rb +12 -12
  118. data/lib/rubocop/cop/layout/indentation_style.rb +8 -3
  119. data/lib/rubocop/cop/layout/indentation_width.rb +9 -5
  120. data/lib/rubocop/cop/layout/initial_indentation.rb +2 -2
  121. data/lib/rubocop/cop/layout/leading_comment_space.rb +2 -2
  122. data/lib/rubocop/cop/layout/line_continuation_leading_space.rb +144 -0
  123. data/lib/rubocop/cop/layout/line_continuation_spacing.rb +140 -0
  124. data/lib/rubocop/cop/layout/line_end_string_concatenation_indentation.rb +2 -0
  125. data/lib/rubocop/cop/layout/line_length.rb +8 -1
  126. data/lib/rubocop/cop/layout/multiline_array_line_breaks.rb +31 -1
  127. data/lib/rubocop/cop/layout/multiline_assignment_layout.rb +1 -1
  128. data/lib/rubocop/cop/layout/multiline_block_layout.rb +4 -2
  129. data/lib/rubocop/cop/layout/multiline_hash_key_line_breaks.rb +29 -1
  130. data/lib/rubocop/cop/layout/multiline_method_argument_line_breaks.rb +39 -2
  131. data/lib/rubocop/cop/layout/multiline_method_call_indentation.rb +20 -5
  132. data/lib/rubocop/cop/layout/multiline_method_parameter_line_breaks.rb +77 -0
  133. data/lib/rubocop/cop/layout/multiline_operation_indentation.rb +2 -2
  134. data/lib/rubocop/cop/layout/redundant_line_break.rb +35 -13
  135. data/lib/rubocop/cop/layout/rescue_ensure_alignment.rb +6 -6
  136. data/lib/rubocop/cop/layout/single_line_block_chain.rb +5 -0
  137. data/lib/rubocop/cop/layout/space_after_comma.rb +9 -1
  138. data/lib/rubocop/cop/layout/space_after_not.rb +1 -1
  139. data/lib/rubocop/cop/layout/space_around_block_parameters.rb +2 -2
  140. data/lib/rubocop/cop/layout/space_around_keyword.rb +3 -3
  141. data/lib/rubocop/cop/layout/space_around_method_call_operator.rb +2 -2
  142. data/lib/rubocop/cop/layout/space_around_operators.rb +54 -22
  143. data/lib/rubocop/cop/layout/space_before_block_braces.rb +22 -11
  144. data/lib/rubocop/cop/layout/space_before_first_arg.rb +2 -2
  145. data/lib/rubocop/cop/layout/space_in_lambda_literal.rb +2 -2
  146. data/lib/rubocop/cop/layout/space_inside_array_literal_brackets.rb +22 -20
  147. data/lib/rubocop/cop/layout/space_inside_array_percent_literal.rb +3 -0
  148. data/lib/rubocop/cop/layout/space_inside_block_braces.rb +32 -12
  149. data/lib/rubocop/cop/layout/space_inside_hash_literal_braces.rb +31 -4
  150. data/lib/rubocop/cop/layout/space_inside_parens.rb +3 -3
  151. data/lib/rubocop/cop/layout/space_inside_percent_literal_delimiters.rb +34 -0
  152. data/lib/rubocop/cop/layout/space_inside_range_literal.rb +1 -1
  153. data/lib/rubocop/cop/layout/space_inside_reference_brackets.rb +10 -6
  154. data/lib/rubocop/cop/layout/space_inside_string_interpolation.rb +5 -4
  155. data/lib/rubocop/cop/layout/trailing_empty_lines.rb +6 -1
  156. data/lib/rubocop/cop/layout/trailing_whitespace.rb +13 -6
  157. data/lib/rubocop/cop/legacy/corrections_proxy.rb +1 -1
  158. data/lib/rubocop/cop/legacy/corrector.rb +1 -1
  159. data/lib/rubocop/cop/lint/ambiguous_block_association.rb +40 -8
  160. data/lib/rubocop/cop/lint/ambiguous_operator.rb +4 -0
  161. data/lib/rubocop/cop/lint/assignment_in_condition.rb +17 -7
  162. data/lib/rubocop/cop/lint/binary_operator_with_identical_operands.rb +2 -2
  163. data/lib/rubocop/cop/lint/constant_overwritten_in_rescue.rb +51 -0
  164. data/lib/rubocop/cop/lint/constant_resolution.rb +5 -1
  165. data/lib/rubocop/cop/lint/debugger.rb +69 -34
  166. data/lib/rubocop/cop/lint/deprecated_class_methods.rb +66 -110
  167. data/lib/rubocop/cop/lint/deprecated_constants.rb +8 -1
  168. data/lib/rubocop/cop/lint/deprecated_open_ssl_constant.rb +1 -1
  169. data/lib/rubocop/cop/lint/duplicate_branch.rb +0 -2
  170. data/lib/rubocop/cop/lint/duplicate_hash_key.rb +2 -1
  171. data/lib/rubocop/cop/lint/duplicate_magic_comment.rb +73 -0
  172. data/lib/rubocop/cop/lint/duplicate_match_pattern.rb +122 -0
  173. data/lib/rubocop/cop/lint/duplicate_methods.rb +49 -19
  174. data/lib/rubocop/cop/lint/duplicate_regexp_character_class_element.rb +55 -11
  175. data/lib/rubocop/cop/lint/duplicate_require.rb +1 -1
  176. data/lib/rubocop/cop/lint/else_layout.rb +3 -7
  177. data/lib/rubocop/cop/lint/empty_block.rb +4 -8
  178. data/lib/rubocop/cop/lint/empty_class.rb +3 -1
  179. data/lib/rubocop/cop/lint/empty_conditional_body.rb +110 -2
  180. data/lib/rubocop/cop/lint/empty_interpolation.rb +1 -1
  181. data/lib/rubocop/cop/lint/erb_new_arguments.rb +15 -16
  182. data/lib/rubocop/cop/lint/float_comparison.rb +10 -0
  183. data/lib/rubocop/cop/lint/format_parameter_mismatch.rb +14 -7
  184. data/lib/rubocop/cop/lint/hash_compare_by_identity.rb +2 -1
  185. data/lib/rubocop/cop/lint/heredoc_method_call_position.rb +16 -18
  186. data/lib/rubocop/cop/lint/identity_comparison.rb +0 -1
  187. data/lib/rubocop/cop/lint/implicit_string_concatenation.rb +1 -1
  188. data/lib/rubocop/cop/lint/incompatible_io_select_with_fiber_scheduler.rb +5 -3
  189. data/lib/rubocop/cop/lint/ineffective_access_modifier.rb +1 -1
  190. data/lib/rubocop/cop/lint/inherit_exception.rb +9 -0
  191. data/lib/rubocop/cop/lint/interpolation_check.rb +5 -4
  192. data/lib/rubocop/cop/lint/it_without_arguments_in_block.rb +56 -0
  193. data/lib/rubocop/cop/lint/lambda_without_literal_block.rb +1 -1
  194. data/lib/rubocop/cop/lint/literal_as_condition.rb +5 -0
  195. data/lib/rubocop/cop/lint/literal_assignment_in_condition.rb +85 -0
  196. data/lib/rubocop/cop/lint/literal_in_interpolation.rb +49 -3
  197. data/lib/rubocop/cop/lint/missing_cop_enable_directive.rb +18 -3
  198. data/lib/rubocop/cop/lint/missing_super.rb +63 -5
  199. data/lib/rubocop/cop/lint/mixed_case_range.rb +116 -0
  200. data/lib/rubocop/cop/lint/mixed_regexp_capture_types.rb +1 -0
  201. data/lib/rubocop/cop/lint/nested_method_definition.rb +53 -9
  202. data/lib/rubocop/cop/lint/next_without_accumulator.rb +11 -7
  203. data/lib/rubocop/cop/lint/non_atomic_file_operation.rb +165 -0
  204. data/lib/rubocop/cop/lint/non_deterministic_require_order.rb +15 -5
  205. data/lib/rubocop/cop/lint/number_conversion.rb +45 -13
  206. data/lib/rubocop/cop/lint/numbered_parameter_assignment.rb +2 -2
  207. data/lib/rubocop/cop/lint/or_assignment_to_constant.rb +2 -0
  208. data/lib/rubocop/cop/lint/ordered_magic_comments.rb +4 -6
  209. data/lib/rubocop/cop/lint/out_of_range_regexp_ref.rb +17 -2
  210. data/lib/rubocop/cop/lint/parentheses_as_grouped_expression.rb +6 -1
  211. data/lib/rubocop/cop/lint/percent_string_array.rb +1 -1
  212. data/lib/rubocop/cop/lint/percent_symbol_array.rb +1 -1
  213. data/lib/rubocop/cop/lint/redundant_cop_disable_directive.rb +54 -14
  214. data/lib/rubocop/cop/lint/redundant_cop_enable_directive.rb +7 -7
  215. data/lib/rubocop/cop/lint/redundant_dir_glob_sort.rb +7 -0
  216. data/lib/rubocop/cop/lint/redundant_regexp_quantifiers.rb +130 -0
  217. data/lib/rubocop/cop/lint/redundant_require_statement.rb +57 -10
  218. data/lib/rubocop/cop/lint/redundant_safe_navigation.rb +85 -8
  219. data/lib/rubocop/cop/lint/redundant_splat_expansion.rb +1 -1
  220. data/lib/rubocop/cop/lint/redundant_string_coercion.rb +35 -15
  221. data/lib/rubocop/cop/lint/redundant_with_index.rb +18 -11
  222. data/lib/rubocop/cop/lint/redundant_with_object.rb +13 -12
  223. data/lib/rubocop/cop/lint/refinement_import_methods.rb +2 -1
  224. data/lib/rubocop/cop/lint/regexp_as_condition.rb +8 -2
  225. data/lib/rubocop/cop/lint/require_parentheses.rb +3 -1
  226. data/lib/rubocop/cop/lint/require_range_parentheses.rb +57 -0
  227. data/lib/rubocop/cop/lint/rescue_type.rb +2 -4
  228. data/lib/rubocop/cop/lint/safe_navigation_chain.rb +50 -11
  229. data/lib/rubocop/cop/lint/safe_navigation_consistency.rb +1 -1
  230. data/lib/rubocop/cop/lint/script_permission.rb +3 -3
  231. data/lib/rubocop/cop/lint/self_assignment.rb +38 -0
  232. data/lib/rubocop/cop/lint/send_with_mixin_argument.rb +6 -6
  233. data/lib/rubocop/cop/lint/shadowed_argument.rb +1 -0
  234. data/lib/rubocop/cop/lint/shadowed_exception.rb +21 -22
  235. data/lib/rubocop/cop/lint/shadowing_outer_local_variable.rb +35 -4
  236. data/lib/rubocop/cop/lint/struct_new_override.rb +14 -14
  237. data/lib/rubocop/cop/lint/suppressed_exception.rb +3 -3
  238. data/lib/rubocop/cop/lint/symbol_conversion.rb +8 -3
  239. data/lib/rubocop/cop/lint/syntax.rb +10 -3
  240. data/lib/rubocop/cop/lint/to_enum_arguments.rb +23 -6
  241. data/lib/rubocop/cop/lint/top_level_return_with_argument.rb +23 -9
  242. data/lib/rubocop/cop/lint/trailing_comma_in_attribute_declaration.rb +2 -2
  243. data/lib/rubocop/cop/lint/unmodified_reduce_accumulator.rb +2 -2
  244. data/lib/rubocop/cop/lint/unreachable_code.rb +4 -2
  245. data/lib/rubocop/cop/lint/unreachable_loop.rb +20 -8
  246. data/lib/rubocop/cop/lint/unused_method_argument.rb +6 -1
  247. data/lib/rubocop/cop/lint/useless_access_modifier.rb +19 -14
  248. data/lib/rubocop/cop/lint/useless_assignment.rb +94 -10
  249. data/lib/rubocop/cop/lint/useless_else_without_rescue.rb +44 -0
  250. data/lib/rubocop/cop/lint/useless_method_definition.rb +12 -4
  251. data/lib/rubocop/cop/lint/useless_rescue.rb +89 -0
  252. data/lib/rubocop/cop/lint/useless_ruby2_keywords.rb +15 -5
  253. data/lib/rubocop/cop/lint/useless_times.rb +3 -3
  254. data/lib/rubocop/cop/lint/void.rb +126 -25
  255. data/lib/rubocop/cop/metrics/abc_size.rb +7 -5
  256. data/lib/rubocop/cop/metrics/block_length.rb +17 -12
  257. data/lib/rubocop/cop/metrics/block_nesting.rb +2 -2
  258. data/lib/rubocop/cop/metrics/class_length.rb +17 -6
  259. data/lib/rubocop/cop/metrics/collection_literal_length.rb +76 -0
  260. data/lib/rubocop/cop/metrics/cyclomatic_complexity.rb +1 -1
  261. data/lib/rubocop/cop/metrics/method_length.rb +18 -12
  262. data/lib/rubocop/cop/metrics/module_length.rb +10 -5
  263. data/lib/rubocop/cop/metrics/parameter_lists.rb +27 -0
  264. data/lib/rubocop/cop/metrics/perceived_complexity.rb +1 -1
  265. data/lib/rubocop/cop/metrics/utils/abc_size_calculator.rb +4 -8
  266. data/lib/rubocop/cop/metrics/utils/code_length_calculator.rb +43 -12
  267. data/lib/rubocop/cop/migration/department_name.rb +3 -3
  268. data/lib/rubocop/cop/mixin/alignment.rb +2 -2
  269. data/lib/rubocop/cop/mixin/allowed_identifiers.rb +2 -2
  270. data/lib/rubocop/cop/mixin/allowed_methods.rb +23 -2
  271. data/lib/rubocop/cop/mixin/allowed_pattern.rb +17 -1
  272. data/lib/rubocop/cop/mixin/allowed_receivers.rb +34 -0
  273. data/lib/rubocop/cop/mixin/annotation_comment.rb +14 -7
  274. data/lib/rubocop/cop/mixin/check_line_breakable.rb +5 -1
  275. data/lib/rubocop/cop/mixin/code_length.rb +12 -1
  276. data/lib/rubocop/cop/mixin/comments_help.rb +37 -11
  277. data/lib/rubocop/cop/mixin/configurable_enforced_style.rb +21 -9
  278. data/lib/rubocop/cop/mixin/configurable_formatting.rb +1 -0
  279. data/lib/rubocop/cop/mixin/def_node.rb +3 -8
  280. data/lib/rubocop/cop/mixin/documentation_comment.rb +1 -1
  281. data/lib/rubocop/cop/mixin/end_keyword_alignment.rb +1 -1
  282. data/lib/rubocop/cop/mixin/enforce_superclass.rb +2 -1
  283. data/lib/rubocop/cop/mixin/first_element_line_break.rb +11 -7
  284. data/lib/rubocop/cop/mixin/frozen_string_literal.rb +4 -0
  285. data/lib/rubocop/cop/mixin/hash_alignment_styles.rb +1 -1
  286. data/lib/rubocop/cop/mixin/hash_shorthand_syntax.rb +162 -12
  287. data/lib/rubocop/cop/mixin/hash_transform_method.rb +13 -9
  288. data/lib/rubocop/cop/mixin/heredoc.rb +6 -2
  289. data/lib/rubocop/cop/mixin/line_length_help.rb +11 -2
  290. data/lib/rubocop/cop/mixin/method_complexity.rb +28 -22
  291. data/lib/rubocop/cop/mixin/min_branches_count.rb +40 -0
  292. data/lib/rubocop/cop/mixin/multiline_element_indentation.rb +34 -12
  293. data/lib/rubocop/cop/mixin/multiline_element_line_breaks.rb +5 -6
  294. data/lib/rubocop/cop/mixin/multiline_expression_indentation.rb +4 -3
  295. data/lib/rubocop/cop/mixin/ordered_gem_node.rb +1 -1
  296. data/lib/rubocop/cop/mixin/percent_array.rb +58 -1
  297. data/lib/rubocop/cop/mixin/percent_literal.rb +1 -1
  298. data/lib/rubocop/cop/mixin/preceding_following_alignment.rb +7 -9
  299. data/lib/rubocop/cop/mixin/range_help.rb +28 -7
  300. data/lib/rubocop/cop/mixin/require_library.rb +2 -0
  301. data/lib/rubocop/cop/mixin/rescue_node.rb +5 -3
  302. data/lib/rubocop/cop/mixin/safe_assignment.rb +1 -1
  303. data/lib/rubocop/cop/mixin/space_after_punctuation.rb +1 -1
  304. data/lib/rubocop/cop/mixin/space_before_punctuation.rb +1 -1
  305. data/lib/rubocop/cop/mixin/statement_modifier.rb +18 -3
  306. data/lib/rubocop/cop/mixin/string_help.rb +4 -2
  307. data/lib/rubocop/cop/mixin/surrounding_space.rb +13 -11
  308. data/lib/rubocop/cop/mixin/trailing_comma.rb +3 -3
  309. data/lib/rubocop/cop/mixin/visibility_help.rb +40 -5
  310. data/lib/rubocop/cop/naming/ascii_identifiers.rb +1 -1
  311. data/lib/rubocop/cop/naming/block_forwarding.rb +39 -8
  312. data/lib/rubocop/cop/naming/block_parameter_name.rb +1 -1
  313. data/lib/rubocop/cop/naming/class_and_module_camel_case.rb +3 -1
  314. data/lib/rubocop/cop/naming/constant_name.rb +4 -5
  315. data/lib/rubocop/cop/naming/file_name.rb +3 -3
  316. data/lib/rubocop/cop/naming/heredoc_delimiter_case.rb +1 -1
  317. data/lib/rubocop/cop/naming/heredoc_delimiter_naming.rb +3 -1
  318. data/lib/rubocop/cop/naming/inclusive_language.rb +29 -8
  319. data/lib/rubocop/cop/naming/memoized_instance_variable_name.rb +26 -11
  320. data/lib/rubocop/cop/naming/method_name.rb +3 -3
  321. data/lib/rubocop/cop/naming/predicate_name.rb +33 -4
  322. data/lib/rubocop/cop/naming/rescued_exceptions_variable_name.rb +12 -4
  323. data/lib/rubocop/cop/naming/variable_name.rb +6 -1
  324. data/lib/rubocop/cop/registry.rb +74 -46
  325. data/lib/rubocop/cop/security/compound_hash.rb +4 -3
  326. data/lib/rubocop/cop/security/open.rb +2 -2
  327. data/lib/rubocop/cop/style/access_modifier_declarations.rb +142 -3
  328. data/lib/rubocop/cop/style/accessor_grouping.rb +50 -20
  329. data/lib/rubocop/cop/style/alias.rb +19 -9
  330. data/lib/rubocop/cop/style/arguments_forwarding.rb +414 -62
  331. data/lib/rubocop/cop/style/array_first_last.rb +64 -0
  332. data/lib/rubocop/cop/style/array_intersect.rb +119 -0
  333. data/lib/rubocop/cop/style/ascii_comments.rb +1 -1
  334. data/lib/rubocop/cop/style/attr.rb +11 -1
  335. data/lib/rubocop/cop/style/auto_resource_cleanup.rb +21 -14
  336. data/lib/rubocop/cop/style/begin_block.rb +1 -2
  337. data/lib/rubocop/cop/style/bisected_attr_accessor.rb +3 -3
  338. data/lib/rubocop/cop/style/block_comments.rb +3 -3
  339. data/lib/rubocop/cop/style/block_delimiters.rb +52 -15
  340. data/lib/rubocop/cop/style/case_equality.rb +40 -10
  341. data/lib/rubocop/cop/style/case_like_if.rb +25 -8
  342. data/lib/rubocop/cop/style/character_literal.rb +1 -1
  343. data/lib/rubocop/cop/style/class_and_module_children.rb +9 -16
  344. data/lib/rubocop/cop/style/class_check.rb +1 -0
  345. data/lib/rubocop/cop/style/class_equality_comparison.rb +79 -12
  346. data/lib/rubocop/cop/style/class_methods_definitions.rb +2 -1
  347. data/lib/rubocop/cop/style/class_vars.rb +3 -3
  348. data/lib/rubocop/cop/style/collection_compact.rb +44 -12
  349. data/lib/rubocop/cop/style/collection_methods.rb +4 -0
  350. data/lib/rubocop/cop/style/colon_method_call.rb +2 -2
  351. data/lib/rubocop/cop/style/combinable_loops.rb +38 -8
  352. data/lib/rubocop/cop/style/command_literal.rb +1 -1
  353. data/lib/rubocop/cop/style/comment_annotation.rb +1 -1
  354. data/lib/rubocop/cop/style/commented_keyword.rb +7 -4
  355. data/lib/rubocop/cop/style/comparable_clamp.rb +125 -0
  356. data/lib/rubocop/cop/style/concat_array_literals.rb +95 -0
  357. data/lib/rubocop/cop/style/conditional_assignment.rb +16 -20
  358. data/lib/rubocop/cop/style/copyright.rb +32 -22
  359. data/lib/rubocop/cop/style/data_inheritance.rb +75 -0
  360. data/lib/rubocop/cop/style/date_time.rb +5 -4
  361. data/lib/rubocop/cop/style/dir.rb +1 -1
  362. data/lib/rubocop/cop/style/dir_empty.rb +54 -0
  363. data/lib/rubocop/cop/style/disable_cops_within_source_code_directive.rb +2 -2
  364. data/lib/rubocop/cop/style/document_dynamic_eval_definition.rb +3 -3
  365. data/lib/rubocop/cop/style/documentation.rb +22 -10
  366. data/lib/rubocop/cop/style/documentation_method.rb +30 -4
  367. data/lib/rubocop/cop/style/double_negation.rb +4 -2
  368. data/lib/rubocop/cop/style/each_for_simple_loop.rb +45 -10
  369. data/lib/rubocop/cop/style/each_with_object.rb +40 -9
  370. data/lib/rubocop/cop/style/empty_block_parameter.rb +2 -2
  371. data/lib/rubocop/cop/style/empty_case_condition.rb +6 -1
  372. data/lib/rubocop/cop/style/empty_else.rb +37 -0
  373. data/lib/rubocop/cop/style/empty_heredoc.rb +73 -0
  374. data/lib/rubocop/cop/style/empty_lambda_parameter.rb +2 -2
  375. data/lib/rubocop/cop/style/empty_literal.rb +1 -1
  376. data/lib/rubocop/cop/style/empty_method.rb +17 -2
  377. data/lib/rubocop/cop/style/encoding.rb +1 -1
  378. data/lib/rubocop/cop/style/endless_method.rb +1 -1
  379. data/lib/rubocop/cop/style/eval_with_location.rb +14 -23
  380. data/lib/rubocop/cop/style/exact_regexp_match.rb +70 -0
  381. data/lib/rubocop/cop/style/explicit_block_argument.rb +8 -4
  382. data/lib/rubocop/cop/style/fetch_env_var.rb +10 -177
  383. data/lib/rubocop/cop/style/file_empty.rb +71 -0
  384. data/lib/rubocop/cop/style/file_read.rb +3 -3
  385. data/lib/rubocop/cop/style/file_write.rb +1 -1
  386. data/lib/rubocop/cop/style/for.rb +5 -1
  387. data/lib/rubocop/cop/style/format_string.rb +33 -12
  388. data/lib/rubocop/cop/style/format_string_token.rb +73 -23
  389. data/lib/rubocop/cop/style/frozen_string_literal_comment.rb +5 -3
  390. data/lib/rubocop/cop/style/guard_clause.rb +159 -42
  391. data/lib/rubocop/cop/style/hash_as_last_array_item.rb +2 -1
  392. data/lib/rubocop/cop/style/hash_conversion.rb +10 -0
  393. data/lib/rubocop/cop/style/hash_each_methods.rb +131 -22
  394. data/lib/rubocop/cop/style/hash_except.rb +101 -9
  395. data/lib/rubocop/cop/style/hash_like_case.rb +3 -9
  396. data/lib/rubocop/cop/style/hash_syntax.rb +51 -5
  397. data/lib/rubocop/cop/style/hash_transform_keys.rb +2 -2
  398. data/lib/rubocop/cop/style/hash_transform_values.rb +2 -2
  399. data/lib/rubocop/cop/style/identical_conditional_branches.rb +47 -3
  400. data/lib/rubocop/cop/style/if_inside_else.rb +6 -0
  401. data/lib/rubocop/cop/style/if_unless_modifier.rb +112 -16
  402. data/lib/rubocop/cop/style/if_with_boolean_literal_branches.rb +34 -5
  403. data/lib/rubocop/cop/style/if_with_semicolon.rb +6 -6
  404. data/lib/rubocop/cop/style/implicit_runtime_error.rb +2 -2
  405. data/lib/rubocop/cop/style/infinite_loop.rb +2 -5
  406. data/lib/rubocop/cop/style/inverse_methods.rb +21 -16
  407. data/lib/rubocop/cop/style/invertible_unless_condition.rb +160 -0
  408. data/lib/rubocop/cop/style/keyword_parameters_order.rb +1 -1
  409. data/lib/rubocop/cop/style/lambda.rb +3 -3
  410. data/lib/rubocop/cop/style/lambda_call.rb +5 -0
  411. data/lib/rubocop/cop/style/line_end_concatenation.rb +5 -2
  412. data/lib/rubocop/cop/style/magic_comment_format.rb +307 -0
  413. data/lib/rubocop/cop/style/map_compact_with_conditional_block.rb +8 -10
  414. data/lib/rubocop/cop/style/map_into_array.rb +175 -0
  415. data/lib/rubocop/cop/style/map_to_hash.rb +21 -8
  416. data/lib/rubocop/cop/style/map_to_set.rb +64 -0
  417. data/lib/rubocop/cop/style/method_call_with_args_parentheses/omit_parentheses.rb +63 -32
  418. data/lib/rubocop/cop/style/method_call_with_args_parentheses/require_parentheses.rb +5 -1
  419. data/lib/rubocop/cop/style/method_call_with_args_parentheses.rb +50 -45
  420. data/lib/rubocop/cop/style/method_call_without_args_parentheses.rb +41 -2
  421. data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +4 -1
  422. data/lib/rubocop/cop/style/method_def_parentheses.rb +12 -5
  423. data/lib/rubocop/cop/style/min_max.rb +3 -3
  424. data/lib/rubocop/cop/style/min_max_comparison.rb +83 -0
  425. data/lib/rubocop/cop/style/missing_else.rb +13 -1
  426. data/lib/rubocop/cop/style/missing_respond_to_missing.rb +2 -2
  427. data/lib/rubocop/cop/style/mixin_grouping.rb +5 -5
  428. data/lib/rubocop/cop/style/module_function.rb +30 -8
  429. data/lib/rubocop/cop/style/multiline_block_chain.rb +4 -2
  430. data/lib/rubocop/cop/style/multiline_if_modifier.rb +2 -6
  431. data/lib/rubocop/cop/style/multiline_if_then.rb +1 -1
  432. data/lib/rubocop/cop/style/multiline_in_pattern_then.rb +2 -4
  433. data/lib/rubocop/cop/style/multiline_memoization.rb +2 -2
  434. data/lib/rubocop/cop/style/multiline_method_signature.rb +18 -6
  435. data/lib/rubocop/cop/style/multiline_ternary_operator.rb +22 -5
  436. data/lib/rubocop/cop/style/multiline_when_then.rb +1 -3
  437. data/lib/rubocop/cop/style/multiple_comparison.rb +14 -0
  438. data/lib/rubocop/cop/style/negated_if_else_condition.rb +17 -10
  439. data/lib/rubocop/cop/style/nested_parenthesized_calls.rb +10 -1
  440. data/lib/rubocop/cop/style/nested_ternary_operator.rb +18 -14
  441. data/lib/rubocop/cop/style/next.rb +4 -6
  442. data/lib/rubocop/cop/style/nil_comparison.rb +2 -0
  443. data/lib/rubocop/cop/style/nil_lambda.rb +4 -4
  444. data/lib/rubocop/cop/style/not.rb +1 -1
  445. data/lib/rubocop/cop/style/numbered_parameters_limit.rb +11 -3
  446. data/lib/rubocop/cop/style/numeric_literal_prefix.rb +1 -1
  447. data/lib/rubocop/cop/style/numeric_literals.rb +16 -1
  448. data/lib/rubocop/cop/style/numeric_predicate.rb +53 -11
  449. data/lib/rubocop/cop/style/object_then.rb +10 -3
  450. data/lib/rubocop/cop/style/one_line_conditional.rb +4 -7
  451. data/lib/rubocop/cop/style/open_struct_use.rb +1 -1
  452. data/lib/rubocop/cop/style/operator_method_call.rb +73 -0
  453. data/lib/rubocop/cop/style/parallel_assignment.rb +32 -24
  454. data/lib/rubocop/cop/style/parentheses_around_condition.rb +8 -0
  455. data/lib/rubocop/cop/style/percent_literal_delimiters.rb +2 -3
  456. data/lib/rubocop/cop/style/percent_q_literals.rb +1 -1
  457. data/lib/rubocop/cop/style/perl_backrefs.rb +22 -1
  458. data/lib/rubocop/cop/style/preferred_hash_methods.rb +1 -1
  459. data/lib/rubocop/cop/style/proc.rb +4 -1
  460. data/lib/rubocop/cop/style/quoted_symbols.rb +1 -1
  461. data/lib/rubocop/cop/style/raise_args.rb +4 -1
  462. data/lib/rubocop/cop/style/redundant_argument.rb +37 -5
  463. data/lib/rubocop/cop/style/redundant_array_constructor.rb +77 -0
  464. data/lib/rubocop/cop/style/redundant_assignment.rb +10 -2
  465. data/lib/rubocop/cop/style/redundant_begin.rb +13 -2
  466. data/lib/rubocop/cop/style/redundant_condition.rb +41 -8
  467. data/lib/rubocop/cop/style/redundant_conditional.rb +2 -14
  468. data/lib/rubocop/cop/style/redundant_constant_base.rb +85 -0
  469. data/lib/rubocop/cop/style/redundant_current_directory_in_path.rb +39 -0
  470. data/lib/rubocop/cop/style/redundant_double_splat_hash_braces.rb +133 -0
  471. data/lib/rubocop/cop/style/redundant_each.rb +119 -0
  472. data/lib/rubocop/cop/style/redundant_exception.rb +32 -12
  473. data/lib/rubocop/cop/style/redundant_fetch_block.rb +10 -8
  474. data/lib/rubocop/cop/style/redundant_filter_chain.rb +118 -0
  475. data/lib/rubocop/cop/style/redundant_heredoc_delimiter_quotes.rb +58 -0
  476. data/lib/rubocop/cop/style/redundant_initialize.rb +3 -1
  477. data/lib/rubocop/cop/style/redundant_interpolation.rb +2 -2
  478. data/lib/rubocop/cop/style/redundant_line_continuation.rb +199 -0
  479. data/lib/rubocop/cop/style/redundant_parentheses.rb +91 -44
  480. data/lib/rubocop/cop/style/redundant_percent_q.rb +2 -2
  481. data/lib/rubocop/cop/style/redundant_regexp_argument.rb +100 -0
  482. data/lib/rubocop/cop/style/redundant_regexp_character_class.rb +9 -3
  483. data/lib/rubocop/cop/style/redundant_regexp_constructor.rb +46 -0
  484. data/lib/rubocop/cop/style/redundant_regexp_escape.rb +22 -4
  485. data/lib/rubocop/cop/style/redundant_return.rb +21 -3
  486. data/lib/rubocop/cop/style/redundant_self.rb +19 -2
  487. data/lib/rubocop/cop/style/redundant_self_assignment.rb +1 -1
  488. data/lib/rubocop/cop/style/redundant_self_assignment_branch.rb +8 -1
  489. data/lib/rubocop/cop/style/redundant_sort.rb +33 -17
  490. data/lib/rubocop/cop/style/redundant_sort_by.rb +24 -8
  491. data/lib/rubocop/cop/style/redundant_string_escape.rb +185 -0
  492. data/lib/rubocop/cop/style/regexp_literal.rb +11 -2
  493. data/lib/rubocop/cop/style/require_order.rb +139 -0
  494. data/lib/rubocop/cop/style/rescue_modifier.rb +2 -4
  495. data/lib/rubocop/cop/style/rescue_standard_error.rb +2 -2
  496. data/lib/rubocop/cop/style/return_nil.rb +6 -2
  497. data/lib/rubocop/cop/style/return_nil_in_predicate_method_definition.rb +95 -0
  498. data/lib/rubocop/cop/style/safe_navigation.rb +41 -10
  499. data/lib/rubocop/cop/style/sample.rb +3 -4
  500. data/lib/rubocop/cop/style/select_by_regexp.rb +30 -11
  501. data/lib/rubocop/cop/style/self_assignment.rb +3 -3
  502. data/lib/rubocop/cop/style/semicolon.rb +71 -8
  503. data/lib/rubocop/cop/style/send.rb +4 -4
  504. data/lib/rubocop/cop/style/send_with_literal_method_name.rb +83 -0
  505. data/lib/rubocop/cop/style/signal_exception.rb +9 -7
  506. data/lib/rubocop/cop/style/single_argument_dig.rb +7 -3
  507. data/lib/rubocop/cop/style/single_line_block_params.rb +1 -1
  508. data/lib/rubocop/cop/style/single_line_do_end_block.rb +67 -0
  509. data/lib/rubocop/cop/style/single_line_methods.rb +1 -1
  510. data/lib/rubocop/cop/style/slicing_with_range.rb +77 -11
  511. data/lib/rubocop/cop/style/sole_nested_conditional.rb +25 -12
  512. data/lib/rubocop/cop/style/special_global_vars.rb +4 -6
  513. data/lib/rubocop/cop/style/static_class.rb +32 -1
  514. data/lib/rubocop/cop/style/stderr_puts.rb +1 -1
  515. data/lib/rubocop/cop/style/string_chars.rb +1 -0
  516. data/lib/rubocop/cop/style/string_hash_keys.rb +4 -1
  517. data/lib/rubocop/cop/style/string_literals.rb +1 -5
  518. data/lib/rubocop/cop/style/string_literals_in_interpolation.rb +30 -5
  519. data/lib/rubocop/cop/style/strip.rb +7 -4
  520. data/lib/rubocop/cop/style/struct_inheritance.rb +3 -3
  521. data/lib/rubocop/cop/style/super_arguments.rb +137 -0
  522. data/lib/rubocop/cop/style/super_with_args_parentheses.rb +35 -0
  523. data/lib/rubocop/cop/style/swap_values.rb +1 -1
  524. data/lib/rubocop/cop/style/symbol_array.rb +40 -19
  525. data/lib/rubocop/cop/style/symbol_proc.rb +110 -15
  526. data/lib/rubocop/cop/style/ternary_parentheses.rb +1 -13
  527. data/lib/rubocop/cop/style/top_level_method_definition.rb +3 -1
  528. data/lib/rubocop/cop/style/trailing_body_on_class.rb +1 -0
  529. data/lib/rubocop/cop/style/trailing_comma_in_arguments.rb +4 -4
  530. data/lib/rubocop/cop/style/trailing_comma_in_block_args.rb +1 -1
  531. data/lib/rubocop/cop/style/trailing_underscore_variable.rb +1 -1
  532. data/lib/rubocop/cop/style/trivial_accessors.rb +4 -1
  533. data/lib/rubocop/cop/style/unless_logical_operators.rb +1 -0
  534. data/lib/rubocop/cop/style/unpack_first.rb +11 -14
  535. data/lib/rubocop/cop/style/word_array.rb +59 -5
  536. data/lib/rubocop/cop/style/yaml_file_read.rb +66 -0
  537. data/lib/rubocop/cop/style/yoda_condition.rb +17 -8
  538. data/lib/rubocop/cop/style/yoda_expression.rb +91 -0
  539. data/lib/rubocop/cop/style/zero_length_predicate.rb +40 -19
  540. data/lib/rubocop/cop/team.rb +66 -56
  541. data/lib/rubocop/cop/util.rb +46 -10
  542. data/lib/rubocop/cop/utils/regexp_ranges.rb +113 -0
  543. data/lib/rubocop/cop/variable_force/assignment.rb +45 -4
  544. data/lib/rubocop/cop/variable_force/scope.rb +3 -3
  545. data/lib/rubocop/cop/variable_force/variable.rb +5 -3
  546. data/lib/rubocop/cop/variable_force/variable_table.rb +6 -4
  547. data/lib/rubocop/cop/variable_force.rb +19 -30
  548. data/lib/rubocop/cops_documentation_generator.rb +60 -18
  549. data/lib/rubocop/directive_comment.rb +14 -12
  550. data/lib/rubocop/ext/comment.rb +18 -0
  551. data/lib/rubocop/ext/processed_source.rb +2 -0
  552. data/lib/rubocop/ext/range.rb +15 -0
  553. data/lib/rubocop/ext/regexp_node.rb +10 -5
  554. data/lib/rubocop/ext/regexp_parser.rb +5 -2
  555. data/lib/rubocop/feature_loader.rb +94 -0
  556. data/lib/rubocop/file_finder.rb +4 -7
  557. data/lib/rubocop/file_patterns.rb +43 -0
  558. data/lib/rubocop/formatter/clang_style_formatter.rb +4 -8
  559. data/lib/rubocop/formatter/disabled_config_formatter.rb +47 -15
  560. data/lib/rubocop/formatter/formatter_set.rb +27 -20
  561. data/lib/rubocop/formatter/{git_hub_actions_formatter.rb → github_actions_formatter.rb} +15 -2
  562. data/lib/rubocop/formatter/html_formatter.rb +38 -18
  563. data/lib/rubocop/formatter/json_formatter.rb +0 -1
  564. data/lib/rubocop/formatter/junit_formatter.rb +4 -1
  565. data/lib/rubocop/formatter/markdown_formatter.rb +1 -1
  566. data/lib/rubocop/formatter/offense_count_formatter.rb +21 -6
  567. data/lib/rubocop/formatter/simple_text_formatter.rb +7 -8
  568. data/lib/rubocop/formatter/tap_formatter.rb +4 -8
  569. data/lib/rubocop/formatter/worst_offenders_formatter.rb +6 -3
  570. data/lib/rubocop/formatter.rb +34 -0
  571. data/lib/rubocop/lockfile.rb +56 -7
  572. data/lib/rubocop/lsp/logger.rb +22 -0
  573. data/lib/rubocop/lsp/routes.rb +243 -0
  574. data/lib/rubocop/lsp/runtime.rb +99 -0
  575. data/lib/rubocop/lsp/server.rb +73 -0
  576. data/lib/rubocop/lsp/severity.rb +27 -0
  577. data/lib/rubocop/lsp.rb +36 -0
  578. data/lib/rubocop/magic_comment.rb +13 -11
  579. data/lib/rubocop/options.rb +101 -32
  580. data/lib/rubocop/path_util.rb +54 -22
  581. data/lib/rubocop/rake_task.rb +35 -10
  582. data/lib/rubocop/result_cache.rb +29 -24
  583. data/lib/rubocop/rspec/cop_helper.rb +32 -3
  584. data/lib/rubocop/rspec/expect_offense.rb +18 -8
  585. data/lib/rubocop/rspec/shared_contexts.rb +86 -33
  586. data/lib/rubocop/rspec/support.rb +19 -2
  587. data/lib/rubocop/runner.rb +91 -23
  588. data/lib/rubocop/server/cache.rb +155 -0
  589. data/lib/rubocop/server/cli.rb +147 -0
  590. data/lib/rubocop/server/client_command/base.rb +44 -0
  591. data/lib/rubocop/server/client_command/exec.rb +64 -0
  592. data/lib/rubocop/server/client_command/restart.rb +25 -0
  593. data/lib/rubocop/server/client_command/start.rb +48 -0
  594. data/lib/rubocop/server/client_command/status.rb +28 -0
  595. data/lib/rubocop/server/client_command/stop.rb +31 -0
  596. data/lib/rubocop/server/client_command.rb +26 -0
  597. data/lib/rubocop/server/core.rb +111 -0
  598. data/lib/rubocop/server/errors.rb +23 -0
  599. data/lib/rubocop/server/helper.rb +34 -0
  600. data/lib/rubocop/server/server_command/base.rb +50 -0
  601. data/lib/rubocop/server/server_command/exec.rb +33 -0
  602. data/lib/rubocop/server/server_command/stop.rb +24 -0
  603. data/lib/rubocop/server/server_command.rb +21 -0
  604. data/lib/rubocop/server/socket_reader.rb +69 -0
  605. data/lib/rubocop/server.rb +53 -0
  606. data/lib/rubocop/string_interpreter.rb +3 -3
  607. data/lib/rubocop/target_finder.rb +92 -82
  608. data/lib/rubocop/target_ruby.rb +86 -80
  609. data/lib/rubocop/version.rb +44 -12
  610. data/lib/rubocop.rb +73 -35
  611. metadata +145 -39
  612. data/lib/rubocop/cop/gemspec/date_assignment.rb +0 -49
  613. data/lib/rubocop/cop/mixin/ignored_methods.rb +0 -52
@@ -0,0 +1,165 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Lint
6
+ # Checks for non-atomic file operation.
7
+ # And then replace it with a nearly equivalent and atomic method.
8
+ #
9
+ # These can cause problems that are difficult to reproduce,
10
+ # especially in cases of frequent file operations in parallel,
11
+ # such as test runs with parallel_rspec.
12
+ #
13
+ # For examples: creating a directory if there is none, has the following problems
14
+ #
15
+ # An exception occurs when the directory didn't exist at the time of `exist?`,
16
+ # but someone else created it before `mkdir` was executed.
17
+ #
18
+ # Subsequent processes are executed without the directory that should be there
19
+ # when the directory existed at the time of `exist?`,
20
+ # but someone else deleted it shortly afterwards.
21
+ #
22
+ # @safety
23
+ # This cop is unsafe, because autocorrection change to atomic processing.
24
+ # The atomic processing of the replacement destination is not guaranteed
25
+ # to be strictly equivalent to that before the replacement.
26
+ #
27
+ # @example
28
+ # # bad - race condition with another process may result in an error in `mkdir`
29
+ # unless Dir.exist?(path)
30
+ # FileUtils.mkdir(path)
31
+ # end
32
+ #
33
+ # # good - atomic and idempotent creation
34
+ # FileUtils.mkdir_p(path)
35
+ #
36
+ # # bad - race condition with another process may result in an error in `remove`
37
+ # if File.exist?(path)
38
+ # FileUtils.remove(path)
39
+ # end
40
+ #
41
+ # # good - atomic and idempotent removal
42
+ # FileUtils.rm_f(path)
43
+ #
44
+ class NonAtomicFileOperation < Base
45
+ extend AutoCorrector
46
+
47
+ MSG_REMOVE_FILE_EXIST_CHECK = 'Remove unnecessary existence check ' \
48
+ '`%<receiver>s.%<method_name>s`.'
49
+ MSG_CHANGE_FORCE_METHOD = 'Use atomic file operation method `FileUtils.%<method_name>s`.'
50
+ MAKE_FORCE_METHODS = %i[makedirs mkdir_p mkpath].freeze
51
+ MAKE_METHODS = %i[mkdir].freeze
52
+ REMOVE_FORCE_METHODS = %i[rm_f rm_rf].freeze
53
+ REMOVE_METHODS = %i[remove delete unlink remove_file rm rmdir safe_unlink].freeze
54
+ RECURSIVE_REMOVE_METHODS = %i[remove_dir remove_entry remove_entry_secure].freeze
55
+ RESTRICT_ON_SEND = (
56
+ MAKE_METHODS + MAKE_FORCE_METHODS + REMOVE_METHODS + RECURSIVE_REMOVE_METHODS +
57
+ REMOVE_FORCE_METHODS
58
+ ).freeze
59
+
60
+ # @!method send_exist_node(node)
61
+ def_node_search :send_exist_node, <<~PATTERN
62
+ $(send (const nil? {:FileTest :File :Dir :Shell}) {:exist? :exists?} ...)
63
+ PATTERN
64
+
65
+ # @!method receiver_and_method_name(node)
66
+ def_node_matcher :receiver_and_method_name, <<~PATTERN
67
+ (send (const nil? $_) $_ ...)
68
+ PATTERN
69
+
70
+ # @!method force?(node)
71
+ def_node_search :force?, <<~PATTERN
72
+ (pair (sym :force) (:true))
73
+ PATTERN
74
+
75
+ # @!method explicit_not_force?(node)
76
+ def_node_search :explicit_not_force?, <<~PATTERN
77
+ (pair (sym :force) (:false))
78
+ PATTERN
79
+
80
+ def on_send(node)
81
+ return unless if_node_child?(node)
82
+ return if explicit_not_force?(node)
83
+ return unless (exist_node = send_exist_node(node.parent).first)
84
+ return unless exist_node.first_argument == node.first_argument
85
+
86
+ register_offense(node, exist_node)
87
+ end
88
+
89
+ private
90
+
91
+ def if_node_child?(node)
92
+ return false unless (parent = node.parent)
93
+
94
+ parent.if_type? && !allowable_use_with_if?(parent)
95
+ end
96
+
97
+ def allowable_use_with_if?(if_node)
98
+ if_node.condition.and_type? || if_node.condition.or_type? || if_node.else_branch
99
+ end
100
+
101
+ def register_offense(node, exist_node)
102
+ add_offense(node, message: message_change_force_method(node)) unless force_method?(node)
103
+
104
+ parent = node.parent
105
+ range = parent.loc.keyword.begin.join(parent.condition.source_range.end)
106
+
107
+ add_offense(range, message: message_remove_file_exist_check(exist_node)) do |corrector|
108
+ autocorrect(corrector, node, range) unless parent.elsif?
109
+ end
110
+ end
111
+
112
+ def message_change_force_method(node)
113
+ format(MSG_CHANGE_FORCE_METHOD, method_name: replacement_method(node))
114
+ end
115
+
116
+ def message_remove_file_exist_check(node)
117
+ receiver, method_name = receiver_and_method_name(node)
118
+ format(MSG_REMOVE_FILE_EXIST_CHECK, receiver: receiver, method_name: method_name)
119
+ end
120
+
121
+ def autocorrect(corrector, node, range)
122
+ corrector.remove(range)
123
+ autocorrect_replace_method(corrector, node)
124
+
125
+ if node.parent.modifier_form?
126
+ corrector.remove(node.source_range.end.join(node.parent.loc.keyword.begin))
127
+ else
128
+ corrector.remove(node.parent.loc.end)
129
+ end
130
+ end
131
+
132
+ def autocorrect_replace_method(corrector, node)
133
+ return if force_method?(node)
134
+
135
+ corrector.replace(node.child_nodes.first.loc.name, 'FileUtils')
136
+ corrector.replace(node.loc.selector, replacement_method(node))
137
+ end
138
+
139
+ def replacement_method(node)
140
+ if MAKE_METHODS.include?(node.method_name)
141
+ 'mkdir_p'
142
+ elsif REMOVE_METHODS.include?(node.method_name)
143
+ 'rm_f'
144
+ elsif RECURSIVE_REMOVE_METHODS.include?(node.method_name)
145
+ 'rm_rf'
146
+ else
147
+ node.method_name
148
+ end
149
+ end
150
+
151
+ def force_method?(node)
152
+ force_method_name?(node) || force_option?(node)
153
+ end
154
+
155
+ def force_option?(node)
156
+ node.arguments.any? { |arg| force?(arg) }
157
+ end
158
+
159
+ def force_method_name?(node)
160
+ (MAKE_FORCE_METHODS + REMOVE_FORCE_METHODS).include?(node.method_name)
161
+ end
162
+ end
163
+ end
164
+ end
165
+ end
@@ -74,6 +74,18 @@ module RuboCop
74
74
  end
75
75
  end
76
76
 
77
+ def on_numblock(node)
78
+ return if target_ruby_version >= 3.0
79
+ return unless node.body
80
+ return unless unsorted_dir_loop?(node.send_node)
81
+
82
+ node.argument_list
83
+ .filter { |argument| var_is_required?(node.body, argument.name) }
84
+ .each do
85
+ add_offense(node.send_node) { |corrector| correct_block(corrector, node.send_node) }
86
+ end
87
+ end
88
+
77
89
  def on_block_pass(node)
78
90
  return if target_ruby_version >= 3.0
79
91
  return unless method_require?(node)
@@ -82,7 +94,7 @@ module RuboCop
82
94
  parent_node = node.parent
83
95
 
84
96
  add_offense(parent_node) do |corrector|
85
- if parent_node.arguments.last&.block_pass_type?
97
+ if parent_node.last_argument&.block_pass_type?
86
98
  correct_block_pass(corrector, parent_node)
87
99
  else
88
100
  correct_block(corrector, parent_node)
@@ -104,7 +116,7 @@ module RuboCop
104
116
 
105
117
  def correct_block_pass(corrector, node)
106
118
  if unsorted_dir_glob_pass?(node)
107
- block_arg = node.arguments.last
119
+ block_arg = node.last_argument
108
120
 
109
121
  corrector.remove(last_arg_range(node))
110
122
  corrector.insert_after(node, ".sort.each(#{block_arg.source})")
@@ -118,9 +130,7 @@ module RuboCop
118
130
  # @return [Parser::Source::Range]
119
131
  #
120
132
  def last_arg_range(node)
121
- node.arguments.last.source_range.with(
122
- begin_pos: node.arguments[-2].source_range.end_pos
123
- )
133
+ node.last_argument.source_range.join(node.arguments[-2].source_range.end)
124
134
  end
125
135
 
126
136
  def unsorted_dir_loop?(node)
@@ -9,14 +9,15 @@ module RuboCop
9
9
  #
10
10
  # Conversion with `Integer`, `Float`, etc. will raise an `ArgumentError`
11
11
  # if given input that is not numeric (eg. an empty string), whereas
12
- # `to_i`, etc. will try to convert regardless of input (`''.to_i => 0`).
12
+ # `to_i`, etc. will try to convert regardless of input (``''.to_i => 0``).
13
13
  # As such, this cop is disabled by default because it's not necessarily
14
14
  # always correct to raise if a value is not numeric.
15
15
  #
16
16
  # NOTE: Some values cannot be converted properly using one of the `Kernel`
17
17
  # method (for instance, `Time` and `DateTime` values are allowed by this
18
18
  # cop by default). Similarly, Rails' duration methods do not work well
19
- # with `Integer()` and can be ignored with `IgnoredMethods`.
19
+ # with `Integer()` and can be allowed with `AllowedMethods`. By default,
20
+ # there are no methods to allowed.
20
21
  #
21
22
  # @safety
22
23
  # Autocorrection is unsafe because it is not guaranteed that the
@@ -45,7 +46,22 @@ module RuboCop
45
46
  # foo.try { |i| Float(i) }
46
47
  # bar.send { |i| Complex(i) }
47
48
  #
48
- # @example IgnoredMethods: [minutes]
49
+ # @example AllowedMethods: [] (default)
50
+ #
51
+ # # bad
52
+ # 10.minutes.to_i
53
+ #
54
+ # @example AllowedMethods: [minutes]
55
+ #
56
+ # # good
57
+ # 10.minutes.to_i
58
+ #
59
+ # @example AllowedPatterns: [] (default)
60
+ #
61
+ # # bad
62
+ # 10.minutes.to_i
63
+ #
64
+ # @example AllowedPatterns: ['min*']
49
65
  #
50
66
  # # good
51
67
  # 10.minutes.to_i
@@ -56,7 +72,9 @@ module RuboCop
56
72
  # Time.now.to_datetime.to_i
57
73
  class NumberConversion < Base
58
74
  extend AutoCorrector
59
- include IgnoredMethods
75
+ include AllowedMethods
76
+ include AllowedPattern
77
+ include IgnoredNode
60
78
 
61
79
  CONVERSION_METHOD_CLASS_MAPPING = {
62
80
  to_i: "#{Integer.name}(%<number_object>s, 10)",
@@ -64,34 +82,39 @@ module RuboCop
64
82
  to_c: "#{Complex.name}(%<number_object>s)",
65
83
  to_r: "#{Rational.name}(%<number_object>s)"
66
84
  }.freeze
67
- MSG = 'Replace unsafe number conversion with number '\
68
- 'class parsing, instead of using '\
69
- '`%<current>s`, use stricter '\
85
+ MSG = 'Replace unsafe number conversion with number ' \
86
+ 'class parsing, instead of using ' \
87
+ '`%<current>s`, use stricter ' \
70
88
  '`%<corrected_method>s`.'
71
89
  CONVERSION_METHODS = %i[Integer Float Complex Rational to_i to_f to_c to_r].freeze
72
90
  METHODS = CONVERSION_METHOD_CLASS_MAPPING.keys.map(&:inspect).join(' ')
73
91
 
74
92
  # @!method to_method(node)
75
93
  def_node_matcher :to_method, <<~PATTERN
76
- (send $_ ${#{METHODS}})
94
+ (call $_ ${#{METHODS}})
77
95
  PATTERN
78
96
 
79
97
  # @!method to_method_symbol(node)
80
98
  def_node_matcher :to_method_symbol, <<~PATTERN
81
- {(send _ $_ ${(sym ${#{METHODS}})} ...)
82
- (send _ $_ ${(block_pass (sym ${#{METHODS}}))} ...)}
99
+ (call _ $_ ${
100
+ {
101
+ (sym ${#{METHODS}})
102
+ (block_pass (sym ${#{METHODS}}))
103
+ }
104
+ } ...)
83
105
  PATTERN
84
106
 
85
107
  def on_send(node)
86
108
  handle_conversion_method(node)
87
109
  handle_as_symbol(node)
88
110
  end
111
+ alias on_csend on_send
89
112
 
90
113
  private
91
114
 
92
115
  def handle_conversion_method(node)
93
116
  to_method(node) do |receiver, to_method|
94
- next if receiver.nil? || ignore_receiver?(receiver)
117
+ next if receiver.nil? || allow_receiver?(receiver)
95
118
 
96
119
  message = format(
97
120
  MSG,
@@ -99,7 +122,11 @@ module RuboCop
99
122
  corrected_method: correct_method(node, receiver)
100
123
  )
101
124
  add_offense(node, message: message) do |corrector|
125
+ next if part_of_ignored_node?(node)
126
+
102
127
  corrector.replace(node, correct_method(node, node.receiver))
128
+
129
+ ignore_node(node)
103
130
  end
104
131
  end
105
132
  end
@@ -135,9 +162,10 @@ module RuboCop
135
162
  corrector.remove(node.loc.end)
136
163
  end
137
164
 
138
- def ignore_receiver?(receiver)
165
+ def allow_receiver?(receiver)
139
166
  if receiver.numeric_type? || (receiver.send_type? &&
140
- (conversion_method?(receiver.method_name) || ignored_method?(receiver.method_name)))
167
+ (conversion_method?(receiver.method_name) ||
168
+ allowed_method_name?(receiver.method_name)))
141
169
  true
142
170
  elsif (receiver = top_receiver(receiver))
143
171
  receiver.const_type? && ignored_class?(receiver.const_name)
@@ -146,6 +174,10 @@ module RuboCop
146
174
  end
147
175
  end
148
176
 
177
+ def allowed_method_name?(name)
178
+ allowed_method?(name) || matches_allowed_pattern?(name)
179
+ end
180
+
149
181
  def top_receiver(node)
150
182
  receiver = node
151
183
  receiver = receiver.receiver until receiver.receiver.nil?
@@ -6,13 +6,13 @@ module RuboCop
6
6
  # Checks for uses of numbered parameter assignment.
7
7
  # It emulates the following warning in Ruby 2.7:
8
8
  #
9
- # % ruby -ve '_1 = :value'
9
+ # $ ruby -ve '_1 = :value'
10
10
  # ruby 2.7.2p137 (2020-10-01 revision 5445e04352) [x86_64-darwin19]
11
11
  # -e:1: warning: `_1' is reserved for numbered parameter; consider another name
12
12
  #
13
13
  # Assigning to a numbered parameter (from `_1` to `_9`) causes an error in Ruby 3.0.
14
14
  #
15
- # % ruby -ve '_1 = :value'
15
+ # $ ruby -ve '_1 = :value'
16
16
  # ruby 3.0.0p0 (2020-12-25 revision 95aff21468) [x86_64-darwin19]
17
17
  # -e:1: _1 is reserved for numbered parameter
18
18
  #
@@ -31,6 +31,8 @@ module RuboCop
31
31
  return unless lhs&.casgn_type?
32
32
 
33
33
  add_offense(node.loc.operator) do |corrector|
34
+ next if node.each_ancestor(:def, :defs).any?
35
+
34
36
  corrector.replace(node.loc.operator, '=')
35
37
  end
36
38
  end
@@ -3,10 +3,12 @@
3
3
  module RuboCop
4
4
  module Cop
5
5
  module Lint
6
- #
7
6
  # Checks the proper ordering of magic comments and whether
8
7
  # a magic comment is not placed before a shebang.
9
8
  #
9
+ # @safety
10
+ # This cop's autocorrection is unsafe because file encoding may change.
11
+ #
10
12
  # @example
11
13
  # # bad
12
14
  #
@@ -61,7 +63,7 @@ module RuboCop
61
63
  def magic_comment_lines
62
64
  lines = [nil, nil]
63
65
 
64
- magic_comments.each.with_index do |comment, index|
66
+ leading_magic_comments.each.with_index do |comment, index|
65
67
  if comment.encoding_specified?
66
68
  lines[0] = index
67
69
  elsif comment.frozen_string_literal_specified?
@@ -73,10 +75,6 @@ module RuboCop
73
75
 
74
76
  lines
75
77
  end
76
-
77
- def magic_comments
78
- leading_comment_lines.map { |line| MagicComment.parse(line) }
79
- end
80
78
  end
81
79
  end
82
80
  end
@@ -3,7 +3,7 @@
3
3
  module RuboCop
4
4
  module Cop
5
5
  module Lint
6
- # This cops looks for references of Regexp captures that are out of range
6
+ # Looks for references of Regexp captures that are out of range
7
7
  # and thus always returns nil.
8
8
  #
9
9
  # @safety
@@ -65,7 +65,13 @@ module RuboCop
65
65
  def on_when(node)
66
66
  regexp_conditions = node.conditions.select(&:regexp_type?)
67
67
 
68
- @valid_ref = regexp_conditions.map { |condition| check_regexp(condition) }.compact.max
68
+ @valid_ref = regexp_conditions.filter_map { |condition| check_regexp(condition) }.max
69
+ end
70
+
71
+ def on_in_pattern(node)
72
+ regexp_patterns = regexp_patterns(node)
73
+
74
+ @valid_ref = regexp_patterns.filter_map { |pattern| check_regexp(pattern) }.max
69
75
  end
70
76
 
71
77
  def on_nth_ref(node)
@@ -84,6 +90,15 @@ module RuboCop
84
90
 
85
91
  private
86
92
 
93
+ def regexp_patterns(in_node)
94
+ pattern = in_node.pattern
95
+ if pattern.regexp_type?
96
+ [pattern]
97
+ else
98
+ pattern.each_descendant(:regexp).to_a
99
+ end
100
+ end
101
+
87
102
  def check_regexp(node)
88
103
  return if node.interpolation?
89
104
 
@@ -40,11 +40,16 @@ module RuboCop
40
40
  unless node.arguments.one? && first_argument_starts_with_left_parenthesis?(node)
41
41
  return true
42
42
  end
43
+ return true if first_argument_block_type?(node.first_argument)
43
44
 
44
45
  node.operator_method? || node.setter_method? || chained_calls?(node) ||
45
46
  valid_first_argument?(node.first_argument)
46
47
  end
47
48
 
49
+ def first_argument_block_type?(first_arg)
50
+ first_arg.block_type? || first_arg.numblock_type?
51
+ end
52
+
48
53
  def valid_first_argument?(first_arg)
49
54
  first_arg.operator_keyword? || first_arg.hash_type? || ternary_expression?(first_arg)
50
55
  end
@@ -55,7 +60,7 @@ module RuboCop
55
60
 
56
61
  def chained_calls?(node)
57
62
  first_argument = node.first_argument
58
- first_argument.send_type? && (node.children.last&.children&.count || 0) > 1
63
+ first_argument.call_type? && (node.children.last&.children&.count || 0) > 1
59
64
  end
60
65
 
61
66
  def ternary_expression?(node)
@@ -50,7 +50,7 @@ module RuboCop
50
50
 
51
51
  add_offense(node) do |corrector|
52
52
  node.each_value do |value|
53
- range = value.loc.expression
53
+ range = value.source_range
54
54
 
55
55
  match = range.source.match(TRAILING_QUOTE)
56
56
  corrector.remove_trailing(range, match[0].length) if match
@@ -41,7 +41,7 @@ module RuboCop
41
41
 
42
42
  def autocorrect(corrector, node)
43
43
  node.children.each do |child|
44
- range = child.loc.expression
44
+ range = child.source_range
45
45
 
46
46
  corrector.remove_trailing(range, 1) if range.source.end_with?(',')
47
47
  corrector.remove_leading(range, 1) if
@@ -71,16 +71,16 @@ module RuboCop
71
71
  processed_source.comment_config.comment_only_line?(directive_comment_range.line) &&
72
72
  directive_comment_range.begin_pos == line_comment_range.begin_pos
73
73
  # When the previous line is blank, it should be retained
74
- range_with_surrounding_space(range: directive_comment_range, side: :right)
74
+ range_with_surrounding_space(directive_comment_range, side: :right)
75
75
  else
76
76
  # Eat the entire comment, the preceding space, and the preceding
77
77
  # newline if there is one.
78
78
  original_begin = directive_comment_range.begin_pos
79
79
  range = range_with_surrounding_space(
80
- range: directive_comment_range, side: :left, newlines: true
80
+ directive_comment_range, side: :left, newlines: true
81
81
  )
82
82
 
83
- range_with_surrounding_space(range: range,
83
+ range_with_surrounding_space(range,
84
84
  side: :right,
85
85
  # Special for a comment that
86
86
  # begins the file: remove
@@ -94,13 +94,13 @@ module RuboCop
94
94
  # is NOT being removed?
95
95
  if ends_its_line?(ranges.last) && trailing_range?(ranges, range)
96
96
  # Eat the comma on the left.
97
- range = range_with_surrounding_space(range: range, side: :left)
97
+ range = range_with_surrounding_space(range, side: :left)
98
98
  range = range_with_surrounding_comma(range, :left)
99
99
  end
100
100
 
101
101
  range = range_with_surrounding_comma(range, :right)
102
102
  # Eat following spaces up to EOL, but not the newline itself.
103
- range_with_surrounding_space(range: range, side: :right, newlines: false)
103
+ range_with_surrounding_space(range, side: :right, newlines: false)
104
104
  end
105
105
 
106
106
  def each_redundant_disable(&block)
@@ -113,6 +113,7 @@ module RuboCop
113
113
  def each_line_range(cop, line_ranges)
114
114
  line_ranges.each_with_index do |line_range, line_range_index|
115
115
  next if ignore_offense?(line_range)
116
+ next if expected_final_disable?(cop, line_range)
116
117
 
117
118
  comment = processed_source.comment_at_line(line_range.begin)
118
119
  redundant = if all_disabled?(comment)
@@ -127,23 +128,35 @@ module RuboCop
127
128
  end
128
129
  end
129
130
 
131
+ # rubocop:disable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity
130
132
  def each_already_disabled(cop, line_ranges)
131
133
  line_ranges.each_cons(2) do |previous_range, range|
132
134
  next if ignore_offense?(range)
133
- next unless followed_ranges?(previous_range, range)
134
-
135
135
  # If a cop is disabled in a range that begins on the same line as
136
136
  # the end of the previous range, it means that the cop was
137
137
  # already disabled by an earlier comment. So it's redundant
138
138
  # whether there are offenses or not.
139
+ next unless followed_ranges?(previous_range, range)
140
+
139
141
  comment = processed_source.comment_at_line(range.begin)
140
142
 
143
+ next unless comment
141
144
  # Comments disabling all cops don't count since it's reasonable
142
145
  # to disable a few select cops first and then all cops further
143
146
  # down in the code.
144
- yield comment, cop if comment && !all_disabled?(comment)
147
+ next if all_disabled?(comment)
148
+
149
+ redundant =
150
+ if department_disabled?(cop, comment)
151
+ find_redundant_department(cop, range)
152
+ else
153
+ cop
154
+ end
155
+
156
+ yield comment, redundant if redundant
145
157
  end
146
158
  end
159
+ # rubocop:enable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity
147
160
 
148
161
  def find_redundant_cop(cop, range)
149
162
  cop_offenses = offenses_to_check.select { |offense| offense.cop_name == cop }
@@ -179,11 +192,21 @@ module RuboCop
179
192
  end
180
193
 
181
194
  def ignore_offense?(line_range)
195
+ return true if line_range.min == CommentConfig::CONFIG_DISABLED_LINE_RANGE_MIN
196
+
182
197
  disabled_ranges.any? do |range|
183
198
  range.cover?(line_range.min) && range.cover?(line_range.max)
184
199
  end
185
200
  end
186
201
 
202
+ def expected_final_disable?(cop, line_range)
203
+ # A cop which is disabled in the config is being re-disabled until end of file
204
+ cop_class = RuboCop::Cop::Registry.global.find_by_cop_name cop
205
+ cop_class &&
206
+ !processed_source.registry.enabled?(cop_class, config) &&
207
+ line_range.max == Float::INFINITY
208
+ end
209
+
187
210
  def department_disabled?(cop, comment)
188
211
  directive = DirectiveComment.new(comment)
189
212
  directive.in_directive_department?(cop) && !directive.overridden_by_department?(cop)
@@ -208,8 +231,13 @@ module RuboCop
208
231
  cop_names = cops.sort.map { |c| describe(c) }.join(', ')
209
232
 
210
233
  add_offense(location, message: message(cop_names)) do |corrector|
211
- range = comment_range_with_surrounding_space(location, comment.loc.expression)
212
- corrector.remove(range)
234
+ range = comment_range_with_surrounding_space(location, comment.source_range)
235
+
236
+ if leave_free_comment?(comment, range)
237
+ corrector.replace(range, ' # ')
238
+ else
239
+ corrector.remove(range)
240
+ end
213
241
  end
214
242
  end
215
243
 
@@ -227,10 +255,16 @@ module RuboCop
227
255
  end
228
256
  end
229
257
 
258
+ def leave_free_comment?(comment, range)
259
+ free_comment = comment.text.gsub(range.source.strip, '')
260
+
261
+ !free_comment.empty? && !free_comment.start_with?('#')
262
+ end
263
+
230
264
  def cop_range(comment, cop)
231
265
  cop = remove_department_marker(cop)
232
- matching_range(comment.loc.expression, cop) ||
233
- matching_range(comment.loc.expression, Badge.parse(cop).cop_name) ||
266
+ matching_range(comment.source_range, cop) ||
267
+ matching_range(comment.source_range, Badge.parse(cop).cop_name) ||
234
268
  raise("Couldn't find #{cop} in comment: #{comment.text}")
235
269
  end
236
270
 
@@ -247,15 +281,21 @@ module RuboCop
247
281
  .drop_while { |r| !r.equal?(range) }
248
282
  .each_cons(2)
249
283
  .map { |range1, range2| range1.end.join(range2.begin).source }
250
- .all? { |intervening| /\A\s*,\s*\Z/.match?(intervening) }
284
+ .all?(/\A\s*,\s*\z/)
285
+ end
286
+
287
+ SIMILAR_COP_NAMES_CACHE = Hash.new do |hash, cop_name|
288
+ hash[:all_cop_names] = Registry.global.names unless hash.key?(:all_cop_names)
289
+ hash[cop_name] = NameSimilarity.find_similar_name(cop_name, hash[:all_cop_names])
251
290
  end
291
+ private_constant :SIMILAR_COP_NAMES_CACHE
252
292
 
253
293
  def describe(cop)
254
294
  return 'all cops' if cop == 'all'
255
295
  return "`#{remove_department_marker(cop)}` department" if department_marker?(cop)
256
296
  return "`#{cop}`" if all_cop_names.include?(cop)
257
297
 
258
- similar = NameSimilarity.find_similar_name(cop, all_cop_names)
298
+ similar = SIMILAR_COP_NAMES_CACHE[cop]
259
299
  similar ? "`#{cop}` (did you mean `#{similar}`?)" : "`#{cop}` (unknown cop)"
260
300
  end
261
301