rubocop 1.69.2 → 1.75.1

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 (349) hide show
  1. checksums.yaml +4 -4
  2. data/LICENSE.txt +1 -1
  3. data/README.md +4 -4
  4. data/config/default.yml +154 -23
  5. data/config/internal_affairs.yml +20 -0
  6. data/config/obsoletion.yml +3 -1
  7. data/lib/rubocop/cli/command/execute_runner.rb +3 -3
  8. data/lib/rubocop/cli/command/show_cops.rb +24 -2
  9. data/lib/rubocop/cli/command/suggest_extensions.rb +7 -1
  10. data/lib/rubocop/cli.rb +1 -1
  11. data/lib/rubocop/comment_config.rb +2 -2
  12. data/lib/rubocop/config.rb +52 -10
  13. data/lib/rubocop/config_loader.rb +52 -9
  14. data/lib/rubocop/config_loader_resolver.rb +36 -10
  15. data/lib/rubocop/config_obsoletion/extracted_cop.rb +4 -3
  16. data/lib/rubocop/config_obsoletion/renamed_cop.rb +18 -3
  17. data/lib/rubocop/config_obsoletion.rb +46 -2
  18. data/lib/rubocop/config_validator.rb +20 -9
  19. data/lib/rubocop/cop/autocorrect_logic.rb +1 -1
  20. data/lib/rubocop/cop/base.rb +6 -0
  21. data/lib/rubocop/cop/bundler/duplicated_gem.rb +1 -1
  22. data/lib/rubocop/cop/bundler/gem_comment.rb +1 -1
  23. data/lib/rubocop/cop/internal_affairs/cop_enabled.rb +85 -0
  24. data/lib/rubocop/cop/internal_affairs/example_description.rb +7 -3
  25. data/lib/rubocop/cop/internal_affairs/location_exists.rb +116 -0
  26. data/lib/rubocop/cop/internal_affairs/location_expression.rb +2 -1
  27. data/lib/rubocop/cop/internal_affairs/node_first_or_last_argument.rb +3 -2
  28. data/lib/rubocop/cop/internal_affairs/node_matcher_directive.rb +1 -1
  29. data/lib/rubocop/cop/internal_affairs/node_pattern_groups/ast_processor.rb +63 -0
  30. data/lib/rubocop/cop/internal_affairs/node_pattern_groups/ast_walker.rb +131 -0
  31. data/lib/rubocop/cop/internal_affairs/node_pattern_groups.rb +229 -0
  32. data/lib/rubocop/cop/internal_affairs/node_type_group.rb +91 -0
  33. data/lib/rubocop/cop/internal_affairs/node_type_multiple_predicates.rb +126 -0
  34. data/lib/rubocop/cop/internal_affairs/node_type_predicate.rb +4 -3
  35. data/lib/rubocop/cop/internal_affairs/on_send_without_on_csend.rb +90 -0
  36. data/lib/rubocop/cop/internal_affairs/plugin.rb +33 -0
  37. data/lib/rubocop/cop/internal_affairs/redundant_described_class_as_subject.rb +6 -5
  38. data/lib/rubocop/cop/internal_affairs/redundant_source_range.rb +3 -1
  39. data/lib/rubocop/cop/internal_affairs/single_line_comparison.rb +5 -4
  40. data/lib/rubocop/cop/internal_affairs/undefined_config.rb +7 -1
  41. data/lib/rubocop/cop/internal_affairs.rb +6 -16
  42. data/lib/rubocop/cop/layout/access_modifier_indentation.rb +1 -1
  43. data/lib/rubocop/cop/layout/argument_alignment.rb +2 -8
  44. data/lib/rubocop/cop/layout/block_alignment.rb +4 -1
  45. data/lib/rubocop/cop/layout/block_end_newline.rb +1 -0
  46. data/lib/rubocop/cop/layout/class_structure.rb +9 -9
  47. data/lib/rubocop/cop/layout/closing_parenthesis_indentation.rb +4 -4
  48. data/lib/rubocop/cop/layout/dot_position.rb +1 -1
  49. data/lib/rubocop/cop/layout/else_alignment.rb +2 -2
  50. data/lib/rubocop/cop/layout/empty_line_after_guard_clause.rb +1 -1
  51. data/lib/rubocop/cop/layout/empty_line_between_defs.rb +7 -11
  52. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +28 -1
  53. data/lib/rubocop/cop/layout/empty_lines_around_block_body.rb +1 -0
  54. data/lib/rubocop/cop/layout/empty_lines_around_exception_handling_keywords.rb +1 -1
  55. data/lib/rubocop/cop/layout/empty_lines_around_method_body.rb +22 -2
  56. data/lib/rubocop/cop/layout/end_alignment.rb +1 -1
  57. data/lib/rubocop/cop/layout/extra_spacing.rb +1 -1
  58. data/lib/rubocop/cop/layout/first_argument_indentation.rb +3 -8
  59. data/lib/rubocop/cop/layout/first_array_element_indentation.rb +2 -7
  60. data/lib/rubocop/cop/layout/first_hash_element_indentation.rb +2 -7
  61. data/lib/rubocop/cop/layout/first_hash_element_line_break.rb +1 -1
  62. data/lib/rubocop/cop/layout/first_parameter_indentation.rb +2 -2
  63. data/lib/rubocop/cop/layout/hash_alignment.rb +6 -4
  64. data/lib/rubocop/cop/layout/heredoc_argument_closing_parenthesis.rb +2 -1
  65. data/lib/rubocop/cop/layout/indentation_width.rb +1 -0
  66. data/lib/rubocop/cop/layout/line_continuation_spacing.rb +7 -1
  67. data/lib/rubocop/cop/layout/line_end_string_concatenation_indentation.rb +2 -2
  68. data/lib/rubocop/cop/layout/line_length.rb +9 -4
  69. data/lib/rubocop/cop/layout/multiline_block_layout.rb +1 -0
  70. data/lib/rubocop/cop/layout/multiline_hash_key_line_breaks.rb +1 -1
  71. data/lib/rubocop/cop/layout/multiline_method_argument_line_breaks.rb +25 -0
  72. data/lib/rubocop/cop/layout/multiline_method_call_brace_layout.rb +1 -0
  73. data/lib/rubocop/cop/layout/multiline_method_call_indentation.rb +4 -4
  74. data/lib/rubocop/cop/layout/multiline_method_parameter_line_breaks.rb +1 -0
  75. data/lib/rubocop/cop/layout/redundant_line_break.rb +16 -11
  76. data/lib/rubocop/cop/layout/rescue_ensure_alignment.rb +2 -2
  77. data/lib/rubocop/cop/layout/single_line_block_chain.rb +1 -1
  78. data/lib/rubocop/cop/layout/space_after_colon.rb +2 -2
  79. data/lib/rubocop/cop/layout/space_after_comma.rb +1 -1
  80. data/lib/rubocop/cop/layout/space_after_method_name.rb +1 -1
  81. data/lib/rubocop/cop/layout/space_after_semicolon.rb +1 -1
  82. data/lib/rubocop/cop/layout/space_around_keyword.rb +1 -0
  83. data/lib/rubocop/cop/layout/space_around_method_call_operator.rb +1 -1
  84. data/lib/rubocop/cop/layout/space_around_operators.rb +7 -4
  85. data/lib/rubocop/cop/layout/space_before_block_braces.rb +1 -0
  86. data/lib/rubocop/cop/layout/space_before_comma.rb +1 -1
  87. data/lib/rubocop/cop/layout/space_before_semicolon.rb +1 -1
  88. data/lib/rubocop/cop/layout/space_inside_block_braces.rb +1 -0
  89. data/lib/rubocop/cop/layout/trailing_whitespace.rb +5 -3
  90. data/lib/rubocop/cop/lint/ambiguous_block_association.rb +1 -1
  91. data/lib/rubocop/cop/lint/array_literal_in_regexp.rb +119 -0
  92. data/lib/rubocop/cop/lint/assignment_in_condition.rb +1 -3
  93. data/lib/rubocop/cop/lint/binary_operator_with_identical_operands.rb +1 -1
  94. data/lib/rubocop/cop/lint/constant_definition_in_block.rb +3 -3
  95. data/lib/rubocop/cop/lint/constant_reassignment.rb +148 -0
  96. data/lib/rubocop/cop/lint/cop_directive_syntax.rb +84 -0
  97. data/lib/rubocop/cop/lint/debugger.rb +3 -3
  98. data/lib/rubocop/cop/lint/duplicate_match_pattern.rb +1 -1
  99. data/lib/rubocop/cop/lint/duplicate_methods.rb +0 -14
  100. data/lib/rubocop/cop/lint/duplicate_regexp_character_class_element.rb +1 -1
  101. data/lib/rubocop/cop/lint/duplicate_set_element.rb +20 -7
  102. data/lib/rubocop/cop/lint/empty_conditional_body.rb +14 -64
  103. data/lib/rubocop/cop/lint/empty_expression.rb +0 -2
  104. data/lib/rubocop/cop/lint/erb_new_arguments.rb +0 -6
  105. data/lib/rubocop/cop/lint/float_comparison.rb +6 -8
  106. data/lib/rubocop/cop/lint/float_out_of_range.rb +1 -1
  107. data/lib/rubocop/cop/lint/format_parameter_mismatch.rb +2 -2
  108. data/lib/rubocop/cop/lint/implicit_string_concatenation.rb +1 -1
  109. data/lib/rubocop/cop/lint/literal_as_condition.rb +103 -9
  110. data/lib/rubocop/cop/lint/literal_in_interpolation.rb +24 -6
  111. data/lib/rubocop/cop/lint/missing_super.rb +2 -2
  112. data/lib/rubocop/cop/lint/mixed_case_range.rb +3 -3
  113. data/lib/rubocop/cop/lint/mixed_regexp_capture_types.rb +1 -1
  114. data/lib/rubocop/cop/lint/nested_method_definition.rb +8 -4
  115. data/lib/rubocop/cop/lint/next_without_accumulator.rb +1 -1
  116. data/lib/rubocop/cop/lint/non_atomic_file_operation.rb +4 -3
  117. data/lib/rubocop/cop/lint/non_local_exit_from_iterator.rb +3 -3
  118. data/lib/rubocop/cop/lint/numeric_operation_with_constant_result.rb +18 -31
  119. data/lib/rubocop/cop/lint/out_of_range_regexp_ref.rb +2 -1
  120. data/lib/rubocop/cop/lint/parentheses_as_grouped_expression.rb +1 -5
  121. data/lib/rubocop/cop/lint/raise_exception.rb +29 -10
  122. data/lib/rubocop/cop/lint/redundant_regexp_quantifiers.rb +1 -1
  123. data/lib/rubocop/cop/lint/redundant_require_statement.rb +0 -21
  124. data/lib/rubocop/cop/lint/redundant_string_coercion.rb +2 -2
  125. data/lib/rubocop/cop/lint/redundant_type_conversion.rb +258 -0
  126. data/lib/rubocop/cop/lint/redundant_with_index.rb +3 -0
  127. data/lib/rubocop/cop/lint/redundant_with_object.rb +3 -0
  128. data/lib/rubocop/cop/lint/rescue_exception.rb +1 -1
  129. data/lib/rubocop/cop/lint/return_in_void_context.rb +4 -11
  130. data/lib/rubocop/cop/lint/safe_navigation_chain.rb +8 -1
  131. data/lib/rubocop/cop/lint/shadowing_outer_local_variable.rb +8 -1
  132. data/lib/rubocop/cop/lint/shared_mutable_default.rb +76 -0
  133. data/lib/rubocop/cop/lint/suppressed_exception.rb +1 -1
  134. data/lib/rubocop/cop/lint/suppressed_exception_in_number_conversion.rb +111 -0
  135. data/lib/rubocop/cop/lint/symbol_conversion.rb +1 -1
  136. data/lib/rubocop/cop/lint/syntax.rb +4 -1
  137. data/lib/rubocop/cop/lint/unescaped_bracket_in_regexp.rb +1 -4
  138. data/lib/rubocop/cop/lint/unexpected_block_arity.rb +3 -1
  139. data/lib/rubocop/cop/lint/unmodified_reduce_accumulator.rb +1 -1
  140. data/lib/rubocop/cop/lint/unreachable_code.rb +2 -1
  141. data/lib/rubocop/cop/lint/unreachable_loop.rb +6 -6
  142. data/lib/rubocop/cop/lint/useless_access_modifier.rb +5 -4
  143. data/lib/rubocop/cop/lint/useless_assignment.rb +1 -1
  144. data/lib/rubocop/cop/lint/useless_constant_scoping.rb +71 -0
  145. data/lib/rubocop/cop/lint/useless_method_definition.rb +1 -1
  146. data/lib/rubocop/cop/lint/useless_numeric_operation.rb +2 -1
  147. data/lib/rubocop/cop/lint/useless_ruby2_keywords.rb +2 -2
  148. data/lib/rubocop/cop/lint/void.rb +12 -9
  149. data/lib/rubocop/cop/metrics/block_length.rb +1 -0
  150. data/lib/rubocop/cop/metrics/block_nesting.rb +1 -1
  151. data/lib/rubocop/cop/metrics/collection_literal_length.rb +7 -0
  152. data/lib/rubocop/cop/metrics/cyclomatic_complexity.rb +1 -1
  153. data/lib/rubocop/cop/metrics/method_length.rb +9 -1
  154. data/lib/rubocop/cop/metrics/module_length.rb +1 -1
  155. data/lib/rubocop/cop/metrics/perceived_complexity.rb +1 -1
  156. data/lib/rubocop/cop/metrics/utils/code_length_calculator.rb +1 -1
  157. data/lib/rubocop/cop/metrics/utils/repeated_attribute_discount.rb +7 -7
  158. data/lib/rubocop/cop/mixin/alignment.rb +2 -2
  159. data/lib/rubocop/cop/mixin/allowed_pattern.rb +4 -4
  160. data/lib/rubocop/cop/mixin/check_line_breakable.rb +13 -13
  161. data/lib/rubocop/cop/mixin/check_single_line_suitability.rb +1 -1
  162. data/lib/rubocop/cop/mixin/comments_help.rb +4 -2
  163. data/lib/rubocop/cop/mixin/dig_help.rb +1 -1
  164. data/lib/rubocop/cop/mixin/forbidden_identifiers.rb +20 -0
  165. data/lib/rubocop/cop/mixin/forbidden_pattern.rb +16 -0
  166. data/lib/rubocop/cop/mixin/frozen_string_literal.rb +1 -1
  167. data/lib/rubocop/cop/mixin/hash_shorthand_syntax.rb +22 -22
  168. data/lib/rubocop/cop/mixin/hash_subset.rb +203 -0
  169. data/lib/rubocop/cop/mixin/hash_transform_method.rb +74 -74
  170. data/lib/rubocop/cop/mixin/method_complexity.rb +2 -1
  171. data/lib/rubocop/cop/mixin/percent_literal.rb +1 -1
  172. data/lib/rubocop/cop/mixin/preceding_following_alignment.rb +48 -24
  173. data/lib/rubocop/cop/mixin/range_help.rb +15 -3
  174. data/lib/rubocop/cop/mixin/space_before_punctuation.rb +1 -1
  175. data/lib/rubocop/cop/mixin/statement_modifier.rb +8 -3
  176. data/lib/rubocop/cop/mixin/string_help.rb +2 -2
  177. data/lib/rubocop/cop/mixin/string_literals_help.rb +1 -1
  178. data/lib/rubocop/cop/mixin/target_ruby_version.rb +1 -1
  179. data/lib/rubocop/cop/mixin/trailing_comma.rb +15 -3
  180. data/lib/rubocop/cop/naming/block_forwarding.rb +19 -15
  181. data/lib/rubocop/cop/naming/method_name.rb +64 -8
  182. data/lib/rubocop/cop/naming/predicate_name.rb +44 -0
  183. data/lib/rubocop/cop/naming/rescued_exceptions_variable_name.rb +3 -3
  184. data/lib/rubocop/cop/naming/variable_name.rb +51 -6
  185. data/lib/rubocop/cop/registry.rb +9 -6
  186. data/lib/rubocop/cop/security/compound_hash.rb +1 -0
  187. data/lib/rubocop/cop/style/access_modifier_declarations.rb +34 -5
  188. data/lib/rubocop/cop/style/accessor_grouping.rb +19 -5
  189. data/lib/rubocop/cop/style/and_or.rb +1 -1
  190. data/lib/rubocop/cop/style/arguments_forwarding.rb +39 -23
  191. data/lib/rubocop/cop/style/array_first_last.rb +18 -2
  192. data/lib/rubocop/cop/style/array_intersect.rb +39 -28
  193. data/lib/rubocop/cop/style/block_delimiters.rb +9 -21
  194. data/lib/rubocop/cop/style/class_and_module_children.rb +35 -10
  195. data/lib/rubocop/cop/style/collection_methods.rb +2 -1
  196. data/lib/rubocop/cop/style/combinable_defined.rb +1 -1
  197. data/lib/rubocop/cop/style/combinable_loops.rb +3 -2
  198. data/lib/rubocop/cop/style/commented_keyword.rb +10 -3
  199. data/lib/rubocop/cop/style/comparable_between.rb +75 -0
  200. data/lib/rubocop/cop/style/concat_array_literals.rb +1 -1
  201. data/lib/rubocop/cop/style/conditional_assignment.rb +6 -4
  202. data/lib/rubocop/cop/style/documentation.rb +1 -1
  203. data/lib/rubocop/cop/style/double_negation.rb +4 -4
  204. data/lib/rubocop/cop/style/each_for_simple_loop.rb +4 -7
  205. data/lib/rubocop/cop/style/each_with_object.rb +2 -3
  206. data/lib/rubocop/cop/style/empty_else.rb +4 -2
  207. data/lib/rubocop/cop/style/empty_literal.rb +1 -1
  208. data/lib/rubocop/cop/style/empty_method.rb +1 -1
  209. data/lib/rubocop/cop/style/endless_method.rb +163 -18
  210. data/lib/rubocop/cop/style/eval_with_location.rb +1 -1
  211. data/lib/rubocop/cop/style/exact_regexp_match.rb +3 -10
  212. data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -7
  213. data/lib/rubocop/cop/style/explicit_block_argument.rb +15 -2
  214. data/lib/rubocop/cop/style/exponential_notation.rb +3 -3
  215. data/lib/rubocop/cop/style/fetch_env_var.rb +1 -1
  216. data/lib/rubocop/cop/style/float_division.rb +8 -4
  217. data/lib/rubocop/cop/style/for.rb +1 -0
  218. data/lib/rubocop/cop/style/format_string_token.rb +38 -11
  219. data/lib/rubocop/cop/style/frozen_string_literal_comment.rb +1 -1
  220. data/lib/rubocop/cop/style/guard_clause.rb +2 -1
  221. data/lib/rubocop/cop/style/hash_each_methods.rb +6 -8
  222. data/lib/rubocop/cop/style/hash_except.rb +24 -148
  223. data/lib/rubocop/cop/style/hash_fetch_chain.rb +105 -0
  224. data/lib/rubocop/cop/style/hash_slice.rb +80 -0
  225. data/lib/rubocop/cop/style/hash_syntax.rb +6 -3
  226. data/lib/rubocop/cop/style/identical_conditional_branches.rb +22 -3
  227. data/lib/rubocop/cop/style/if_inside_else.rb +10 -13
  228. data/lib/rubocop/cop/style/if_unless_modifier.rb +5 -5
  229. data/lib/rubocop/cop/style/if_with_boolean_literal_branches.rb +1 -1
  230. data/lib/rubocop/cop/style/if_with_semicolon.rb +2 -2
  231. data/lib/rubocop/cop/style/infinite_loop.rb +1 -1
  232. data/lib/rubocop/cop/style/inverse_methods.rb +15 -11
  233. data/lib/rubocop/cop/style/invertible_unless_condition.rb +2 -2
  234. data/lib/rubocop/cop/style/ip_addresses.rb +2 -2
  235. data/lib/rubocop/cop/style/it_assignment.rb +36 -0
  236. data/lib/rubocop/cop/style/it_block_parameter.rb +100 -0
  237. data/lib/rubocop/cop/style/keyword_parameters_order.rb +14 -8
  238. data/lib/rubocop/cop/style/lambda.rb +1 -0
  239. data/lib/rubocop/cop/style/line_end_concatenation.rb +10 -4
  240. data/lib/rubocop/cop/style/map_into_array.rb +2 -1
  241. data/lib/rubocop/cop/style/map_to_hash.rb +1 -1
  242. data/lib/rubocop/cop/style/map_to_set.rb +3 -2
  243. data/lib/rubocop/cop/style/method_call_with_args_parentheses/omit_parentheses.rb +22 -15
  244. data/lib/rubocop/cop/style/method_call_with_args_parentheses.rb +2 -0
  245. data/lib/rubocop/cop/style/method_call_without_args_parentheses.rb +2 -1
  246. data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +3 -4
  247. data/lib/rubocop/cop/style/method_def_parentheses.rb +1 -1
  248. data/lib/rubocop/cop/style/missing_else.rb +2 -0
  249. data/lib/rubocop/cop/style/multiline_block_chain.rb +3 -2
  250. data/lib/rubocop/cop/style/multiline_method_signature.rb +1 -9
  251. data/lib/rubocop/cop/style/multiple_comparison.rb +26 -20
  252. data/lib/rubocop/cop/style/mutable_constant.rb +3 -3
  253. data/lib/rubocop/cop/style/negated_if_else_condition.rb +1 -1
  254. data/lib/rubocop/cop/style/nested_parenthesized_calls.rb +1 -1
  255. data/lib/rubocop/cop/style/next.rb +44 -0
  256. data/lib/rubocop/cop/style/object_then.rb +14 -15
  257. data/lib/rubocop/cop/style/open_struct_use.rb +5 -5
  258. data/lib/rubocop/cop/style/parallel_assignment.rb +1 -5
  259. data/lib/rubocop/cop/style/parentheses_around_condition.rb +2 -2
  260. data/lib/rubocop/cop/style/percent_literal_delimiters.rb +1 -1
  261. data/lib/rubocop/cop/style/proc.rb +2 -2
  262. data/lib/rubocop/cop/style/quoted_symbols.rb +1 -1
  263. data/lib/rubocop/cop/style/raise_args.rb +14 -12
  264. data/lib/rubocop/cop/style/random_with_offset.rb +3 -3
  265. data/lib/rubocop/cop/style/redundant_begin.rb +2 -1
  266. data/lib/rubocop/cop/style/redundant_condition.rb +47 -2
  267. data/lib/rubocop/cop/style/redundant_current_directory_in_path.rb +16 -5
  268. data/lib/rubocop/cop/style/redundant_double_splat_hash_braces.rb +6 -10
  269. data/lib/rubocop/cop/style/redundant_each.rb +1 -1
  270. data/lib/rubocop/cop/style/redundant_exception.rb +2 -2
  271. data/lib/rubocop/cop/style/redundant_format.rb +257 -0
  272. data/lib/rubocop/cop/style/redundant_freeze.rb +3 -3
  273. data/lib/rubocop/cop/style/redundant_initialize.rb +12 -3
  274. data/lib/rubocop/cop/style/redundant_line_continuation.rb +34 -13
  275. data/lib/rubocop/cop/style/redundant_parentheses.rb +30 -15
  276. data/lib/rubocop/cop/style/redundant_regexp_argument.rb +3 -0
  277. data/lib/rubocop/cop/style/redundant_regexp_character_class.rb +1 -1
  278. data/lib/rubocop/cop/style/redundant_regexp_escape.rb +1 -1
  279. data/lib/rubocop/cop/style/redundant_self.rb +1 -0
  280. data/lib/rubocop/cop/style/redundant_self_assignment.rb +14 -28
  281. data/lib/rubocop/cop/style/redundant_sort.rb +2 -2
  282. data/lib/rubocop/cop/style/redundant_sort_by.rb +17 -1
  283. data/lib/rubocop/cop/style/redundant_string_escape.rb +2 -2
  284. data/lib/rubocop/cop/style/rescue_modifier.rb +3 -0
  285. data/lib/rubocop/cop/style/return_nil.rb +1 -1
  286. data/lib/rubocop/cop/style/safe_navigation.rb +2 -2
  287. data/lib/rubocop/cop/style/select_by_regexp.rb +4 -1
  288. data/lib/rubocop/cop/style/semicolon.rb +1 -1
  289. data/lib/rubocop/cop/style/send_with_literal_method_name.rb +2 -1
  290. data/lib/rubocop/cop/style/single_line_block_params.rb +1 -1
  291. data/lib/rubocop/cop/style/single_line_do_end_block.rb +4 -3
  292. data/lib/rubocop/cop/style/single_line_methods.rb +6 -7
  293. data/lib/rubocop/cop/style/slicing_with_range.rb +40 -11
  294. data/lib/rubocop/cop/style/sole_nested_conditional.rb +41 -106
  295. data/lib/rubocop/cop/style/string_concatenation.rb +2 -2
  296. data/lib/rubocop/cop/style/string_literals.rb +1 -1
  297. data/lib/rubocop/cop/style/string_methods.rb +1 -1
  298. data/lib/rubocop/cop/style/super_arguments.rb +65 -17
  299. data/lib/rubocop/cop/style/symbol_proc.rb +2 -0
  300. data/lib/rubocop/cop/style/ternary_parentheses.rb +1 -1
  301. data/lib/rubocop/cop/style/top_level_method_definition.rb +2 -1
  302. data/lib/rubocop/cop/style/trailing_comma_in_arguments.rb +4 -1
  303. data/lib/rubocop/cop/style/trailing_comma_in_array_literal.rb +47 -6
  304. data/lib/rubocop/cop/style/trailing_comma_in_hash_literal.rb +48 -6
  305. data/lib/rubocop/cop/style/trivial_accessors.rb +1 -1
  306. data/lib/rubocop/cop/style/while_until_modifier.rb +0 -1
  307. data/lib/rubocop/cop/style/yoda_condition.rb +8 -4
  308. data/lib/rubocop/cop/style/yoda_expression.rb +2 -1
  309. data/lib/rubocop/cop/util.rb +12 -5
  310. data/lib/rubocop/cop/utils/format_string.rb +10 -5
  311. data/lib/rubocop/cop/variable_force/scope.rb +1 -1
  312. data/lib/rubocop/cop/variable_force/variable.rb +9 -2
  313. data/lib/rubocop/cop/variable_force/variable_table.rb +3 -3
  314. data/lib/rubocop/cop/variable_force.rb +1 -1
  315. data/lib/rubocop/cops_documentation_generator.rb +25 -14
  316. data/lib/rubocop/directive_comment.rb +45 -11
  317. data/lib/rubocop/ext/regexp_node.rb +0 -1
  318. data/lib/rubocop/formatter/formatter_set.rb +1 -1
  319. data/lib/rubocop/lsp/diagnostic.rb +189 -0
  320. data/lib/rubocop/lsp/logger.rb +2 -2
  321. data/lib/rubocop/lsp/routes.rb +7 -23
  322. data/lib/rubocop/lsp/runtime.rb +18 -50
  323. data/lib/rubocop/lsp/server.rb +0 -2
  324. data/lib/rubocop/lsp/stdin_runner.rb +85 -0
  325. data/lib/rubocop/options.rb +28 -12
  326. data/lib/rubocop/path_util.rb +15 -8
  327. data/lib/rubocop/plugin/configuration_integrator.rb +143 -0
  328. data/lib/rubocop/plugin/load_error.rb +26 -0
  329. data/lib/rubocop/plugin/loader.rb +100 -0
  330. data/lib/rubocop/plugin/not_supported_error.rb +29 -0
  331. data/lib/rubocop/plugin.rb +46 -0
  332. data/lib/rubocop/rake_task.rb +4 -1
  333. data/lib/rubocop/result_cache.rb +13 -13
  334. data/lib/rubocop/rspec/cop_helper.rb +13 -1
  335. data/lib/rubocop/rspec/expect_offense.rb +6 -2
  336. data/lib/rubocop/rspec/shared_contexts.rb +39 -1
  337. data/lib/rubocop/rspec/support.rb +4 -2
  338. data/lib/rubocop/runner.rb +10 -7
  339. data/lib/rubocop/server/cache.rb +35 -2
  340. data/lib/rubocop/server/cli.rb +2 -2
  341. data/lib/rubocop/target_finder.rb +1 -0
  342. data/lib/rubocop/target_ruby.rb +16 -1
  343. data/lib/rubocop/version.rb +30 -8
  344. data/lib/rubocop.rb +16 -1
  345. data/lib/ruby_lsp/rubocop/addon.rb +75 -0
  346. data/lib/ruby_lsp/rubocop/runtime_adapter.rb +65 -0
  347. metadata +59 -16
  348. data/lib/rubocop/cop/utils/regexp_ranges.rb +0 -113
  349. data/lib/rubocop/rspec/host_environment_simulation_helper.rb +0 -28
@@ -31,6 +31,20 @@ module RuboCop
31
31
  #
32
32
  # This cop handles not only method forwarding but also forwarding to `super`.
33
33
  #
34
+ # [NOTE]
35
+ # ====
36
+ # Because of a bug in Ruby 3.3.0, when a block is referenced inside of another block,
37
+ # no offense will be registered until Ruby 3.4:
38
+ #
39
+ # [source,ruby]
40
+ # ----
41
+ # def foo(&block)
42
+ # # Using an anonymous block would be a syntax error on Ruby 3.3.0
43
+ # block_method { bar(&block) }
44
+ # end
45
+ # ----
46
+ # ====
47
+ #
34
48
  # @example
35
49
  # # bad
36
50
  # def foo(*args, &block)
@@ -148,7 +162,7 @@ module RuboCop
148
162
 
149
163
  restarg, kwrestarg, blockarg = extract_forwardable_args(node.arguments)
150
164
  forwardable_args = redundant_forwardable_named_args(restarg, kwrestarg, blockarg)
151
- send_nodes = node.each_descendant(:send, :csend, :super, :yield).to_a
165
+ send_nodes = node.each_descendant(:call, :super, :yield).to_a
152
166
 
153
167
  send_classifications = classify_send_nodes(
154
168
  node, send_nodes, non_splat_or_block_pass_lvar_references(node.body), forwardable_args
@@ -191,9 +205,7 @@ module RuboCop
191
205
 
192
206
  send_classifications.each do |send_node, c, forward_rest, forward_kwrest, forward_block_arg| # rubocop:disable Layout/LineLength
193
207
  if !forward_rest && !forward_kwrest && c != :all_anonymous
194
- # Prevents `anonymous block parameter is also used within block (SyntaxError)` occurs
195
- # in Ruby 3.3.0.
196
- if outside_block?(forward_block_arg)
208
+ if allow_anonymous_forwarding_in_block?(forward_block_arg)
197
209
  register_forward_block_arg_offense(!forward_rest, node.arguments, block_arg)
198
210
  register_forward_block_arg_offense(!forward_rest, send_node, forward_block_arg)
199
211
  end
@@ -214,24 +226,22 @@ module RuboCop
214
226
  # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
215
227
  def add_post_ruby_32_offenses(def_node, send_classifications, forwardable_args)
216
228
  return unless use_anonymous_forwarding?
217
- return if send_inside_block?(send_classifications)
229
+ return unless all_forwarding_offenses_correctable?(send_classifications)
218
230
 
219
231
  rest_arg, kwrest_arg, block_arg = *forwardable_args
220
232
 
221
233
  send_classifications.each do |send_node, _c, forward_rest, forward_kwrest, forward_block_arg| # rubocop:disable Layout/LineLength
222
- if outside_block?(forward_rest)
234
+ if allow_anonymous_forwarding_in_block?(forward_rest)
223
235
  register_forward_args_offense(def_node.arguments, rest_arg)
224
236
  register_forward_args_offense(send_node, forward_rest)
225
237
  end
226
238
 
227
- if outside_block?(forward_kwrest)
239
+ if allow_anonymous_forwarding_in_block?(forward_kwrest)
228
240
  register_forward_kwargs_offense(!forward_rest, def_node.arguments, kwrest_arg)
229
241
  register_forward_kwargs_offense(!forward_rest, send_node, forward_kwrest)
230
242
  end
231
243
 
232
- # Prevents `anonymous block parameter is also used within block (SyntaxError)` occurs
233
- # in Ruby 3.3.0.
234
- if outside_block?(forward_block_arg)
244
+ if allow_anonymous_forwarding_in_block?(forward_block_arg)
235
245
  register_forward_block_arg_offense(!forward_rest, def_node.arguments, block_arg)
236
246
  register_forward_block_arg_offense(!forward_rest, send_node, forward_block_arg)
237
247
  end
@@ -293,10 +303,25 @@ module RuboCop
293
303
  redundant_arg_names.include?(arg.source) ? arg : nil
294
304
  end
295
305
 
296
- def outside_block?(node)
306
+ # Checks if forwarding is uses both in blocks and outside of blocks.
307
+ # On Ruby 3.3.0, anonymous block forwarding in blocks can be is a syntax
308
+ # error, so we only want to register an offense if we can change all occurrences.
309
+ def all_forwarding_offenses_correctable?(send_classifications)
310
+ return true if target_ruby_version >= 3.4
311
+
312
+ send_classifications.none? do |send_node, *|
313
+ send_node.each_ancestor(:any_block).any?
314
+ end
315
+ end
316
+
317
+ # Ruby 3.3.0 had a bug where accessing an anonymous block argument inside of a block
318
+ # was a syntax error in unambiguous cases: https://bugs.ruby-lang.org/issues/20090
319
+ # We disallow this also for earlier Ruby versions so that code is forwards compatible.
320
+ def allow_anonymous_forwarding_in_block?(node)
297
321
  return false unless node
322
+ return true if target_ruby_version >= 3.4
298
323
 
299
- node.each_ancestor(:block, :numblock).none?
324
+ node.each_ancestor(:any_block).none?
300
325
  end
301
326
 
302
327
  def register_forward_args_offense(def_arguments_or_send, rest_arg_or_splat)
@@ -357,12 +382,6 @@ module RuboCop
357
382
  cop_config.fetch('UseAnonymousForwarding', false)
358
383
  end
359
384
 
360
- def send_inside_block?(send_classifications)
361
- send_classifications.any? do |send_node, *|
362
- send_node.each_ancestor(:block, :numblock).any?
363
- end
364
- end
365
-
366
385
  def add_parens_if_missing(node, corrector)
367
386
  return if parentheses?(node)
368
387
  return if node.send_type? && node.method?(:[])
@@ -511,7 +530,7 @@ module RuboCop
511
530
  end
512
531
 
513
532
  def additional_kwargs?
514
- @def_node.arguments.any? { |a| a.kwarg_type? || a.kwoptarg_type? }
533
+ @def_node.arguments.any? { |a| a.type?(:kwarg, :kwoptarg) }
515
534
  end
516
535
 
517
536
  def forward_additional_kwargs?
@@ -540,10 +559,7 @@ module RuboCop
540
559
  end
541
560
 
542
561
  def explicit_block_name?
543
- block_forwarding_config = config.for_cop('Naming/BlockForwarding')
544
- return false unless block_forwarding_config['Enabled']
545
-
546
- block_forwarding_config['EnforcedStyle'] == 'explicit'
562
+ config.for_enabled_cop('Naming/BlockForwarding')['EnforcedStyle'] == 'explicit'
547
563
  end
548
564
  end
549
565
  end
@@ -42,14 +42,30 @@ module RuboCop
42
42
  return if node.parent && brace_method?(node.parent)
43
43
 
44
44
  preferred = (value.zero? ? 'first' : 'last')
45
- add_offense(node.loc.selector, message: format(MSG, preferred: preferred)) do |corrector|
46
- corrector.replace(node.loc.selector, ".#{preferred}")
45
+ offense_range = find_offense_range(node)
46
+
47
+ add_offense(offense_range, message: format(MSG, preferred: preferred)) do |corrector|
48
+ corrector.replace(offense_range, preferred_value(node, preferred))
47
49
  end
48
50
  end
49
51
  # rubocop:enable Metrics/AbcSize
52
+ alias on_csend on_send
50
53
 
51
54
  private
52
55
 
56
+ def preferred_value(node, value)
57
+ value = ".#{value}" unless node.loc.dot
58
+ value
59
+ end
60
+
61
+ def find_offense_range(node)
62
+ if node.loc.dot
63
+ node.loc.selector.join(node.source_range.end)
64
+ else
65
+ node.loc.selector
66
+ end
67
+ end
68
+
53
69
  def innermost_braces_node(node)
54
70
  node = node.receiver while node.receiver.send_type? && node.receiver.method?(:[])
55
71
  node
@@ -6,7 +6,8 @@ module RuboCop
6
6
  # In Ruby 3.1, `Array#intersect?` has been added.
7
7
  #
8
8
  # This cop identifies places where `(array1 & array2).any?`
9
- # can be replaced by `array1.intersect?(array2)`.
9
+ # or `(array1.intersection(array2)).any?` can be replaced by
10
+ # `array1.intersect?(array2)`.
10
11
  #
11
12
  # The `array1.intersect?(array2)` method is faster than
12
13
  # `(array1 & array2).any?` and is more readable.
@@ -20,6 +21,10 @@ module RuboCop
20
21
  # [1].intersect?([1,2]) { |x| false } # => true
21
22
  # ----
22
23
  #
24
+ # NOTE: Although `Array#intersection` can take zero or multiple arguments,
25
+ # only cases where exactly one argument is provided can be replaced with
26
+ # `Array#intersect?` and are handled by this cop.
27
+ #
23
28
  # @safety
24
29
  # This cop cannot guarantee that `array1` and `array2` are
25
30
  # actually arrays while method `intersect?` is for arrays only.
@@ -30,6 +35,11 @@ module RuboCop
30
35
  # (array1 & array2).empty?
31
36
  # (array1 & array2).none?
32
37
  #
38
+ # # bad
39
+ # array1.intersection(array2).any?
40
+ # array1.intersection(array2).empty?
41
+ # array1.intersection(array2).none?
42
+ #
33
43
  # # good
34
44
  # array1.intersect?(array2)
35
45
  # !array1.intersect?(array2)
@@ -53,65 +63,66 @@ module RuboCop
53
63
 
54
64
  minimum_target_ruby_version 3.1
55
65
 
56
- # @!method regular_bad_intersection_check?(node)
57
- def_node_matcher :regular_bad_intersection_check?, <<~PATTERN
58
- (send
59
- (begin
60
- (send $(...) :& $(...))
61
- ) ${:any? :empty? :none?}
62
- )
63
- PATTERN
66
+ PREDICATES = %i[any? empty? none?].to_set.freeze
67
+ ACTIVE_SUPPORT_PREDICATES = (PREDICATES + %i[present? blank?]).freeze
64
68
 
65
- # @!method active_support_bad_intersection_check?(node)
66
- def_node_matcher :active_support_bad_intersection_check?, <<~PATTERN
67
- (send
68
- (begin
69
- (send $(...) :& $(...))
70
- ) ${:present? :any? :blank? :empty? :none?}
69
+ # @!method bad_intersection_check?(node, predicates)
70
+ def_node_matcher :bad_intersection_check?, <<~PATTERN
71
+ (call
72
+ {
73
+ (begin (send $_ :& $_))
74
+ (call $_ :intersection $_)
75
+ }
76
+ $%1
71
77
  )
72
78
  PATTERN
73
79
 
74
- MSG = 'Use `%<negated>s%<receiver>s.intersect?(%<argument>s)` ' \
75
- 'instead of `(%<receiver>s & %<argument>s).%<method_name>s`.'
80
+ MSG = 'Use `%<negated>s%<receiver>s%<dot>sintersect?(%<argument>s)` ' \
81
+ 'instead of `%<existing>s`.'
76
82
  STRAIGHT_METHODS = %i[present? any?].freeze
77
83
  NEGATED_METHODS = %i[blank? empty? none?].freeze
78
84
  RESTRICT_ON_SEND = (STRAIGHT_METHODS + NEGATED_METHODS).freeze
79
85
 
80
86
  def on_send(node)
81
87
  return if node.block_literal?
82
- return unless (receiver, argument, method_name = bad_intersection_check?(node))
88
+ return unless (receiver, argument, method_name = bad_intersection?(node))
83
89
 
84
- message = message(receiver.source, argument.source, method_name)
90
+ dot = node.loc.dot.source
91
+ message = message(receiver.source, argument.source, method_name, dot, node.source)
85
92
 
86
93
  add_offense(node, message: message) do |corrector|
87
94
  bang = straight?(method_name) ? '' : '!'
88
95
 
89
- corrector.replace(node, "#{bang}#{receiver.source}.intersect?(#{argument.source})")
96
+ corrector.replace(node, "#{bang}#{receiver.source}#{dot}intersect?(#{argument.source})")
90
97
  end
91
98
  end
99
+ alias on_csend on_send
92
100
 
93
101
  private
94
102
 
95
- def bad_intersection_check?(node)
96
- if active_support_extensions_enabled?
97
- active_support_bad_intersection_check?(node)
98
- else
99
- regular_bad_intersection_check?(node)
100
- end
103
+ def bad_intersection?(node)
104
+ predicates = if active_support_extensions_enabled?
105
+ ACTIVE_SUPPORT_PREDICATES
106
+ else
107
+ PREDICATES
108
+ end
109
+
110
+ bad_intersection_check?(node, predicates)
101
111
  end
102
112
 
103
113
  def straight?(method_name)
104
114
  STRAIGHT_METHODS.include?(method_name.to_sym)
105
115
  end
106
116
 
107
- def message(receiver, argument, method_name)
117
+ def message(receiver, argument, method_name, dot, existing)
108
118
  negated = straight?(method_name) ? '' : '!'
109
119
  format(
110
120
  MSG,
111
121
  negated: negated,
112
122
  receiver: receiver,
113
123
  argument: argument,
114
- method_name: method_name
124
+ dot: dot,
125
+ existing: existing
115
126
  )
116
127
  end
117
128
  end
@@ -195,6 +195,7 @@ module RuboCop
195
195
  end
196
196
  end
197
197
  end
198
+ alias on_csend on_send
198
199
 
199
200
  def on_block(node)
200
201
  return if ignored_node?(node)
@@ -207,6 +208,7 @@ module RuboCop
207
208
  end
208
209
 
209
210
  alias on_numblock on_block
211
+ alias on_itblock on_block
210
212
 
211
213
  private
212
214
 
@@ -346,9 +348,9 @@ module RuboCop
346
348
  # rubocop:disable Metrics/CyclomaticComplexity
347
349
  def get_blocks(node, &block)
348
350
  case node.type
349
- when :block, :numblock
351
+ when :block, :numblock, :itblock
350
352
  yield node
351
- when :send
353
+ when :send, :csend
352
354
  # When a method has an argument which is another method with a block,
353
355
  # that block needs braces, otherwise a syntax error will be introduced
354
356
  # for subsequent arguments.
@@ -369,9 +371,8 @@ module RuboCop
369
371
  end
370
372
  # rubocop:enable Metrics/CyclomaticComplexity
371
373
 
372
- # rubocop:disable Metrics/CyclomaticComplexity
373
374
  def proper_block_style?(node)
374
- return true if require_braces?(node) || require_do_end?(node)
375
+ return true if require_do_end?(node)
375
376
  return special_method_proper_block_style?(node) if special_method?(node.method_name)
376
377
 
377
378
  case style
@@ -381,15 +382,6 @@ module RuboCop
381
382
  when :always_braces then braces_style?(node)
382
383
  end
383
384
  end
384
- # rubocop:enable Metrics/CyclomaticComplexity
385
-
386
- def require_braces?(node)
387
- return false unless node.braces?
388
-
389
- node.each_ancestor(:send).any? do |send|
390
- send.arithmetic_operation? && node.source_range.end_pos < send.loc.selector.begin_pos
391
- end
392
- end
393
385
 
394
386
  def require_do_end?(node)
395
387
  return false if node.braces? || node.multiline?
@@ -483,18 +475,14 @@ module RuboCop
483
475
  end
484
476
 
485
477
  def return_value_of_scope?(node)
486
- return false unless node.parent
487
-
488
- conditional?(node.parent) || array_or_range?(node.parent) ||
489
- node.parent.children.last == node
490
- end
478
+ return false unless (parent = node.parent)
491
479
 
492
- def conditional?(node)
493
- node.if_type? || node.operator_keyword?
480
+ parent.conditional? || parent.operator_keyword? || array_or_range?(parent) ||
481
+ parent.children.last == node
494
482
  end
495
483
 
496
484
  def array_or_range?(node)
497
- node.array_type? || node.range_type?
485
+ node.type?(:array, :range)
498
486
  end
499
487
 
500
488
  def begin_required?(block_node)
@@ -3,14 +3,26 @@
3
3
  module RuboCop
4
4
  module Cop
5
5
  module Style
6
- # Checks the style of children definitions at classes and
7
- # modules. Basically there are two different styles:
6
+ # Checks that namespaced classes and modules are defined with a consistent style.
7
+ #
8
+ # With `nested` style, classes and modules should be defined separately (one constant
9
+ # on each line, without `::`). With `compact` style, classes and modules should be
10
+ # defined with fully qualified names (using `::` for namespaces).
11
+ #
12
+ # NOTE: The style chosen will affect `Module.nesting` for the class or module. Using
13
+ # `nested` style will result in each level being added, whereas `compact` style will
14
+ # only include the fully qualified class or module name.
15
+ #
16
+ # By default, `EnforcedStyle` applies to both classes and modules. If desired, separate
17
+ # styles can be defined for classes and modules by using `EnforcedStyleForClasses` and
18
+ # `EnforcedStyleForModules` respectively. If not set, or set to nil, the `EnforcedStyle`
19
+ # value will be used.
8
20
  #
9
21
  # @safety
10
22
  # Autocorrection is unsafe.
11
23
  #
12
- # Moving from compact to nested children requires knowledge of whether the
13
- # outer parent is a module or a class. Moving from nested to compact requires
24
+ # Moving from `compact` to `nested` children requires knowledge of whether the
25
+ # outer parent is a module or a class. Moving from `nested` to `compact` requires
14
26
  # verification that the outer parent is defined elsewhere. RuboCop does not
15
27
  # have the knowledge to perform either operation safely and thus requires
16
28
  # manual oversight.
@@ -42,16 +54,18 @@ module RuboCop
42
54
  def on_class(node)
43
55
  return if node.parent_class && style != :nested
44
56
 
45
- check_style(node, node.body)
57
+ check_style(node, node.body, style_for_classes)
46
58
  end
47
59
 
48
60
  def on_module(node)
49
- check_style(node, node.body)
61
+ check_style(node, node.body, style_for_modules)
50
62
  end
51
63
 
52
64
  private
53
65
 
54
66
  def nest_or_compact(corrector, node)
67
+ style = node.class_type? ? style_for_classes : style_for_modules
68
+
55
69
  if style == :nested
56
70
  nest_definition(corrector, node)
57
71
  else
@@ -126,9 +140,12 @@ module RuboCop
126
140
  end
127
141
 
128
142
  def unindent(corrector, node)
129
- return if node.body.children.last.nil?
143
+ return unless node.body.children.last
144
+
145
+ last_child_leading_spaces = leading_spaces(node.body.children.last)
146
+ return if leading_spaces(node).size == last_child_leading_spaces.size
130
147
 
131
- column_delta = configured_indentation_width - leading_spaces(node.body.children.last).size
148
+ column_delta = configured_indentation_width - last_child_leading_spaces.size
132
149
  return if column_delta.zero?
133
150
 
134
151
  AlignmentCorrector.correct(corrector, processed_source, node, column_delta)
@@ -138,7 +155,7 @@ module RuboCop
138
155
  node.source_range.source_line[/\A\s*/]
139
156
  end
140
157
 
141
- def check_style(node, body)
158
+ def check_style(node, body, style)
142
159
  return if node.identifier.namespace&.cbase_type?
143
160
 
144
161
  if style == :nested
@@ -158,7 +175,7 @@ module RuboCop
158
175
 
159
176
  def check_compact_style(node, body)
160
177
  parent = node.parent
161
- return if parent&.class_type? || parent&.module_type?
178
+ return if parent&.type?(:class, :module)
162
179
 
163
180
  return unless needs_compacting?(body)
164
181
 
@@ -180,6 +197,14 @@ module RuboCop
180
197
  def compact_node_name?(node)
181
198
  node.identifier.source.include?('::')
182
199
  end
200
+
201
+ def style_for_classes
202
+ cop_config['EnforcedStyleForClasses'] || style
203
+ end
204
+
205
+ def style_for_modules
206
+ cop_config['EnforcedStyleForModules'] || style
207
+ end
183
208
  end
184
209
  end
185
210
  end
@@ -49,14 +49,15 @@ module RuboCop
49
49
  def on_block(node)
50
50
  check_method_node(node.send_node)
51
51
  end
52
-
53
52
  alias on_numblock on_block
53
+ alias on_itblock on_block
54
54
 
55
55
  def on_send(node)
56
56
  return unless implicit_block?(node)
57
57
 
58
58
  check_method_node(node)
59
59
  end
60
+ alias on_csend on_send
60
61
 
61
62
  private
62
63
 
@@ -55,7 +55,7 @@ module RuboCop
55
55
  def defined_calls(nodes)
56
56
  nodes.filter_map do |defined_node|
57
57
  subject = defined_node.first_argument
58
- subject if subject.const_type? || subject.call_type?
58
+ subject if subject.type?(:const, :call)
59
59
  end
60
60
  end
61
61
 
@@ -80,6 +80,7 @@ module RuboCop
80
80
  # rubocop:enable Metrics/CyclomaticComplexity
81
81
 
82
82
  alias on_numblock on_block
83
+ alias on_itblock on_block
83
84
 
84
85
  def on_for(node)
85
86
  return unless node.parent&.begin_type?
@@ -98,7 +99,7 @@ module RuboCop
98
99
  end
99
100
 
100
101
  def same_collection_looping_block?(node, sibling)
101
- return false if sibling.nil? || (!sibling.block_type? && !sibling.numblock_type?)
102
+ return false if sibling.nil? || !sibling.any_block_type?
102
103
 
103
104
  sibling.method?(node.method_name) &&
104
105
  sibling.receiver == node.receiver &&
@@ -118,7 +119,7 @@ module RuboCop
118
119
 
119
120
  def correct_end_of_block(corrector, node)
120
121
  return unless node.left_sibling.respond_to?(:braces?)
121
- return if node.right_sibling&.block_type? || node.right_sibling&.numblock_type?
122
+ return if node.right_sibling&.any_block_type?
122
123
 
123
124
  end_of_block = node.left_sibling.braces? ? '}' : ' end'
124
125
  corrector.remove(node.loc.end)
@@ -9,8 +9,8 @@ module RuboCop
9
9
  # These keywords are: `class`, `module`, `def`, `begin`, `end`.
10
10
  #
11
11
  # Note that some comments
12
- # (`:nodoc:`, `:yields:`, `rubocop:disable` and `rubocop:todo`)
13
- # and RBS::Inline annotation comments are allowed.
12
+ # (`:nodoc:`, `:yields:`, `rubocop:disable` and `rubocop:todo`),
13
+ # RBS::Inline annotation, and Steep annotation (`steep:ignore`) are allowed.
14
14
  #
15
15
  # Autocorrection removes comments from `end` keyword and keeps comments
16
16
  # for `class`, `module`, `def` and `begin` above the keyword.
@@ -57,9 +57,11 @@ module RuboCop
57
57
 
58
58
  REGEXP = /(?<keyword>\S+).*#/.freeze
59
59
 
60
- SUBCLASS_DEFINITION = /\A\s*class\s+\w+\s*<\s*\w+/.freeze
60
+ SUBCLASS_DEFINITION = /\A\s*class\s+(\w|::)+\s*<\s*(\w|::)+/.freeze
61
61
  METHOD_DEFINITION = /\A\s*def\s/.freeze
62
62
 
63
+ STEEP_REGEXP = /#\ssteep:ignore(\s|\z)/.freeze
64
+
63
65
  def on_new_investigation
64
66
  processed_source.comments.each do |comment|
65
67
  next unless offensive?(comment) && (match = source_line(comment).match(REGEXP))
@@ -86,6 +88,7 @@ module RuboCop
86
88
  def offensive?(comment)
87
89
  line = source_line(comment)
88
90
  return false if rbs_inline_annotation?(line, comment)
91
+ return false if steep_annotation?(comment)
89
92
 
90
93
  KEYWORD_REGEXES.any? { |r| r.match?(line) } &&
91
94
  ALLOWED_COMMENT_REGEXES.none? { |r| r.match?(line) }
@@ -105,6 +108,10 @@ module RuboCop
105
108
  false
106
109
  end
107
110
  end
111
+
112
+ def steep_annotation?(comment)
113
+ comment.text.match?(STEEP_REGEXP)
114
+ end
108
115
  end
109
116
  end
110
117
  end
@@ -0,0 +1,75 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Style
6
+ # Checks for logical comparison which can be replaced with `Comparable#between?`.
7
+ #
8
+ # NOTE: `Comparable#between?` is on average slightly slower than logical comparison,
9
+ # although the difference generally isn't observable. If you require maximum
10
+ # performance, consider using logical comparison.
11
+ #
12
+ # @example
13
+ #
14
+ # # bad
15
+ # x >= min && x <= max
16
+ #
17
+ # # bad
18
+ # x <= max && x >= min
19
+ #
20
+ # # good
21
+ # x.between?(min, max)
22
+ #
23
+ class ComparableBetween < Base
24
+ extend AutoCorrector
25
+
26
+ MSG = 'Prefer `%<prefer>s` over logical comparison.'
27
+
28
+ # @!method logical_comparison_between_by_min_first?(node)
29
+ def_node_matcher :logical_comparison_between_by_min_first?, <<~PATTERN
30
+ (and
31
+ (send
32
+ {$_value :>= $_min | $_min :<= $_value})
33
+ (send
34
+ {$_value :<= $_max | $_max :>= $_value}))
35
+ PATTERN
36
+
37
+ # @!method logical_comparison_between_by_max_first?(node)
38
+ def_node_matcher :logical_comparison_between_by_max_first?, <<~PATTERN
39
+ (and
40
+ (send
41
+ {$_value :<= $_max | $_max :>= $_value})
42
+ (send
43
+ {$_value :>= $_min | $_min :<= $_value}))
44
+ PATTERN
45
+
46
+ def on_and(node)
47
+ logical_comparison_between_by_min_first?(node) do |*args|
48
+ min_and_value, max_and_value = args.each_slice(2).to_a
49
+
50
+ register_offense(node, min_and_value, max_and_value)
51
+ end
52
+
53
+ logical_comparison_between_by_max_first?(node) do |*args|
54
+ max_and_value, min_and_value = args.each_slice(2).to_a
55
+
56
+ register_offense(node, min_and_value, max_and_value)
57
+ end
58
+ end
59
+
60
+ private
61
+
62
+ def register_offense(node, min_and_value, max_and_value)
63
+ value = (min_and_value & max_and_value).first
64
+ min = min_and_value.find { _1 != value }
65
+ max = max_and_value.find { _1 != value }
66
+
67
+ prefer = "#{value.source}.between?(#{min.source}, #{max.source})"
68
+ add_offense(node, message: format(MSG, prefer: prefer)) do |corrector|
69
+ corrector.replace(node, prefer)
70
+ end
71
+ end
72
+ end
73
+ end
74
+ end
75
+ end
@@ -86,7 +86,7 @@ module RuboCop
86
86
 
87
87
  def percent_literals_includes_only_basic_literals?(node)
88
88
  node.arguments.select(&:percent_literal?).all? do |arg|
89
- arg.children.all? { |child| child.str_type? || child.sym_type? }
89
+ arg.children.all? { |child| child.type?(:str, :sym) }
90
90
  end
91
91
  end
92
92
  end