rubocop 1.84.2 → 1.88.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 (313) hide show
  1. checksums.yaml +4 -4
  2. data/config/default.yml +180 -86
  3. data/config/obsoletion.yml +26 -1
  4. data/lib/rubocop/cache_config.rb +1 -1
  5. data/lib/rubocop/cli/command/auto_generate_config.rb +34 -2
  6. data/lib/rubocop/cli/command/list_enabled_cops_for.rb +40 -0
  7. data/lib/rubocop/cli/command/mcp.rb +19 -0
  8. data/lib/rubocop/cli/command/show_cops.rb +2 -2
  9. data/lib/rubocop/cli/command/show_docs_url.rb +4 -8
  10. data/lib/rubocop/cli/command/suggest_extensions.rb +1 -1
  11. data/lib/rubocop/cli.rb +9 -7
  12. data/lib/rubocop/comment_config.rb +12 -15
  13. data/lib/rubocop/config.rb +14 -10
  14. data/lib/rubocop/config_finder.rb +1 -1
  15. data/lib/rubocop/config_loader.rb +17 -2
  16. data/lib/rubocop/config_loader_resolver.rb +13 -4
  17. data/lib/rubocop/config_obsoletion/extracted_cop.rb +4 -2
  18. data/lib/rubocop/config_store.rb +2 -2
  19. data/lib/rubocop/config_validator.rb +1 -1
  20. data/lib/rubocop/cop/autocorrect_logic.rb +2 -1
  21. data/lib/rubocop/cop/base.rb +25 -4
  22. data/lib/rubocop/cop/bundler/gem_comment.rb +2 -2
  23. data/lib/rubocop/cop/correctors/condition_corrector.rb +1 -1
  24. data/lib/rubocop/cop/correctors/multiline_literal_brace_corrector.rb +1 -5
  25. data/lib/rubocop/cop/correctors/parentheses_corrector.rb +33 -2
  26. data/lib/rubocop/cop/correctors/percent_literal_corrector.rb +2 -2
  27. data/lib/rubocop/cop/correctors.rb +28 -0
  28. data/lib/rubocop/cop/documentation.rb +2 -3
  29. data/lib/rubocop/cop/exclude_limit.rb +31 -5
  30. data/lib/rubocop/cop/gemspec/duplicated_assignment.rb +2 -2
  31. data/lib/rubocop/cop/gemspec/require_mfa.rb +5 -5
  32. data/lib/rubocop/cop/gemspec/ruby_version_globals_usage.rb +3 -3
  33. data/lib/rubocop/cop/internal_affairs/itblock_handler.rb +69 -0
  34. data/lib/rubocop/cop/internal_affairs/location_line_equality_comparison.rb +1 -0
  35. data/lib/rubocop/cop/internal_affairs/redundant_let_rubocop_config_new.rb +5 -3
  36. data/lib/rubocop/cop/internal_affairs.rb +1 -0
  37. data/lib/rubocop/cop/layout/argument_alignment.rb +2 -2
  38. data/lib/rubocop/cop/layout/array_alignment.rb +1 -1
  39. data/lib/rubocop/cop/layout/begin_end_alignment.rb +1 -1
  40. data/lib/rubocop/cop/layout/block_alignment.rb +41 -4
  41. data/lib/rubocop/cop/layout/class_structure.rb +1 -1
  42. data/lib/rubocop/cop/layout/dot_position.rb +1 -1
  43. data/lib/rubocop/cop/layout/empty_line_after_guard_clause.rb +23 -7
  44. data/lib/rubocop/cop/layout/empty_line_between_defs.rb +1 -1
  45. data/lib/rubocop/cop/layout/empty_lines_around_attribute_accessor.rb +1 -0
  46. data/lib/rubocop/cop/layout/empty_lines_around_block_body.rb +12 -2
  47. data/lib/rubocop/cop/layout/empty_lines_around_class_body.rb +16 -2
  48. data/lib/rubocop/cop/layout/empty_lines_around_module_body.rb +16 -2
  49. data/lib/rubocop/cop/layout/end_alignment.rb +8 -5
  50. data/lib/rubocop/cop/layout/first_hash_element_indentation.rb +7 -1
  51. data/lib/rubocop/cop/layout/hash_alignment.rb +1 -1
  52. data/lib/rubocop/cop/layout/indentation_width.rb +12 -0
  53. data/lib/rubocop/cop/layout/line_length.rb +5 -3
  54. data/lib/rubocop/cop/layout/multiline_assignment_layout.rb +9 -2
  55. data/lib/rubocop/cop/layout/multiline_method_call_brace_layout.rb +1 -1
  56. data/lib/rubocop/cop/layout/multiline_method_call_indentation.rb +53 -3
  57. data/lib/rubocop/cop/layout/parameter_alignment.rb +1 -1
  58. data/lib/rubocop/cop/layout/redundant_line_break.rb +3 -1
  59. data/lib/rubocop/cop/layout/space_around_block_parameters.rb +1 -1
  60. data/lib/rubocop/cop/layout/space_around_keyword.rb +3 -1
  61. data/lib/rubocop/cop/layout/space_before_brackets.rb +1 -1
  62. data/lib/rubocop/cop/layout/space_in_lambda_literal.rb +1 -0
  63. data/lib/rubocop/cop/lint/ambiguous_assignment.rb +1 -11
  64. data/lib/rubocop/cop/lint/ambiguous_block_association.rb +1 -1
  65. data/lib/rubocop/cop/lint/ambiguous_operator_precedence.rb +1 -10
  66. data/lib/rubocop/cop/lint/circular_argument_reference.rb +1 -3
  67. data/lib/rubocop/cop/lint/constant_overwritten_in_rescue.rb +1 -1
  68. data/lib/rubocop/cop/lint/constant_reassignment.rb +93 -11
  69. data/lib/rubocop/cop/lint/constant_resolution.rb +6 -6
  70. data/lib/rubocop/cop/lint/data_define_override.rb +63 -0
  71. data/lib/rubocop/cop/lint/debugger.rb +0 -1
  72. data/lib/rubocop/cop/lint/deprecated_constants.rb +2 -8
  73. data/lib/rubocop/cop/lint/duplicate_methods.rb +55 -8
  74. data/lib/rubocop/cop/lint/empty_block.rb +4 -4
  75. data/lib/rubocop/cop/lint/empty_conditional_body.rb +6 -1
  76. data/lib/rubocop/cop/lint/empty_in_pattern.rb +8 -1
  77. data/lib/rubocop/cop/lint/empty_when.rb +8 -1
  78. data/lib/rubocop/cop/lint/ensure_return.rb +19 -1
  79. data/lib/rubocop/cop/lint/erb_new_arguments.rb +4 -2
  80. data/lib/rubocop/cop/lint/float_comparison.rb +1 -0
  81. data/lib/rubocop/cop/lint/incompatible_io_select_with_fiber_scheduler.rb +5 -1
  82. data/lib/rubocop/cop/lint/interpolation_check.rb +25 -5
  83. data/lib/rubocop/cop/lint/lambda_without_literal_block.rb +1 -1
  84. data/lib/rubocop/cop/lint/literal_assignment_in_condition.rb +11 -1
  85. data/lib/rubocop/cop/lint/literal_in_interpolation.rb +8 -11
  86. data/lib/rubocop/cop/lint/missing_cop_enable_directive.rb +5 -5
  87. data/lib/rubocop/cop/lint/multiple_comparison.rb +2 -2
  88. data/lib/rubocop/cop/lint/next_without_accumulator.rb +2 -0
  89. data/lib/rubocop/cop/lint/no_return_in_begin_end_blocks.rb +16 -0
  90. data/lib/rubocop/cop/lint/non_deterministic_require_order.rb +4 -2
  91. data/lib/rubocop/cop/lint/non_local_exit_from_iterator.rb +1 -1
  92. data/lib/rubocop/cop/lint/number_conversion.rb +19 -10
  93. data/lib/rubocop/cop/lint/numbered_parameter_assignment.rb +1 -1
  94. data/lib/rubocop/cop/lint/numeric_operation_with_constant_result.rb +3 -0
  95. data/lib/rubocop/cop/lint/ordered_magic_comments.rb +7 -7
  96. data/lib/rubocop/cop/lint/parentheses_as_grouped_expression.rb +3 -13
  97. data/lib/rubocop/cop/lint/raise_exception.rb +1 -1
  98. data/lib/rubocop/cop/lint/rand_one.rb +1 -1
  99. data/lib/rubocop/cop/lint/redundant_cop_disable_directive.rb +4 -1
  100. data/lib/rubocop/cop/lint/redundant_cop_enable_directive.rb +6 -12
  101. data/lib/rubocop/cop/lint/redundant_dir_glob_sort.rb +15 -4
  102. data/lib/rubocop/cop/lint/redundant_safe_navigation.rb +36 -12
  103. data/lib/rubocop/cop/lint/redundant_splat_expansion.rb +4 -0
  104. data/lib/rubocop/cop/lint/redundant_type_conversion.rb +10 -3
  105. data/lib/rubocop/cop/lint/redundant_with_index.rb +1 -1
  106. data/lib/rubocop/cop/lint/redundant_with_object.rb +5 -0
  107. data/lib/rubocop/cop/lint/refinement_import_methods.rb +8 -1
  108. data/lib/rubocop/cop/lint/regexp_as_condition.rb +9 -1
  109. data/lib/rubocop/cop/lint/require_parentheses.rb +13 -4
  110. data/lib/rubocop/cop/lint/require_range_parentheses.rb +2 -1
  111. data/lib/rubocop/cop/lint/require_relative_self_path.rb +7 -5
  112. data/lib/rubocop/cop/lint/rescue_type.rb +1 -1
  113. data/lib/rubocop/cop/lint/safe_navigation_chain.rb +18 -0
  114. data/lib/rubocop/cop/lint/safe_navigation_consistency.rb +7 -1
  115. data/lib/rubocop/cop/lint/safe_navigation_with_empty.rb +1 -1
  116. data/lib/rubocop/cop/lint/script_permission.rb +5 -1
  117. data/lib/rubocop/cop/lint/self_assignment.rb +24 -1
  118. data/lib/rubocop/cop/lint/send_with_mixin_argument.rb +1 -1
  119. data/lib/rubocop/cop/lint/shadowed_exception.rb +1 -1
  120. data/lib/rubocop/cop/lint/shadowing_outer_local_variable.rb +14 -0
  121. data/lib/rubocop/cop/lint/shared_mutable_default.rb +3 -1
  122. data/lib/rubocop/cop/lint/suppressed_exception_in_number_conversion.rb +12 -0
  123. data/lib/rubocop/cop/lint/symbol_conversion.rb +21 -4
  124. data/lib/rubocop/cop/lint/syntax.rb +25 -1
  125. data/lib/rubocop/cop/lint/to_enum_arguments.rb +28 -1
  126. data/lib/rubocop/cop/lint/top_level_return_with_argument.rb +1 -1
  127. data/lib/rubocop/cop/lint/trailing_comma_in_attribute_declaration.rb +5 -1
  128. data/lib/rubocop/cop/lint/unescaped_bracket_in_regexp.rb +4 -2
  129. data/lib/rubocop/cop/lint/unmodified_reduce_accumulator.rb +1 -0
  130. data/lib/rubocop/cop/lint/unreachable_code.rb +2 -2
  131. data/lib/rubocop/cop/lint/unreachable_pattern_branch.rb +113 -0
  132. data/lib/rubocop/cop/lint/unused_method_argument.rb +10 -0
  133. data/lib/rubocop/cop/lint/useless_assignment.rb +14 -14
  134. data/lib/rubocop/cop/lint/useless_constant_scoping.rb +4 -4
  135. data/lib/rubocop/cop/lint/useless_default_value_argument.rb +2 -0
  136. data/lib/rubocop/cop/lint/useless_ruby2_keywords.rb +8 -4
  137. data/lib/rubocop/cop/lint/useless_setter_call.rb +4 -1
  138. data/lib/rubocop/cop/lint/useless_times.rb +22 -1
  139. data/lib/rubocop/cop/lint/utils/nil_receiver_checker.rb +35 -9
  140. data/lib/rubocop/cop/lint/void.rb +32 -12
  141. data/lib/rubocop/cop/metrics/block_length.rb +1 -1
  142. data/lib/rubocop/cop/metrics/block_nesting.rb +23 -0
  143. data/lib/rubocop/cop/metrics/collection_literal_length.rb +1 -1
  144. data/lib/rubocop/cop/metrics/method_length.rb +1 -1
  145. data/lib/rubocop/cop/metrics/utils/iterating_block.rb +1 -1
  146. data/lib/rubocop/cop/migration/department_name.rb +12 -1
  147. data/lib/rubocop/cop/mixin/check_line_breakable.rb +1 -1
  148. data/lib/rubocop/cop/mixin/check_single_line_suitability.rb +2 -2
  149. data/lib/rubocop/cop/mixin/configurable_max.rb +6 -5
  150. data/lib/rubocop/cop/mixin/hash_transform_method/autocorrection.rb +63 -0
  151. data/lib/rubocop/cop/mixin/hash_transform_method.rb +10 -60
  152. data/lib/rubocop/cop/mixin/project_index_help.rb +48 -0
  153. data/lib/rubocop/cop/mixin.rb +86 -0
  154. data/lib/rubocop/cop/naming/binary_operator_parameter_name.rb +1 -1
  155. data/lib/rubocop/cop/naming/block_parameter_name.rb +1 -1
  156. data/lib/rubocop/cop/naming/memoized_instance_variable_name.rb +1 -1
  157. data/lib/rubocop/cop/naming/predicate_method.rb +2 -2
  158. data/lib/rubocop/cop/naming/predicate_prefix.rb +1 -1
  159. data/lib/rubocop/cop/naming/rescued_exceptions_variable_name.rb +1 -1
  160. data/lib/rubocop/cop/offense.rb +8 -0
  161. data/lib/rubocop/cop/registry.rb +62 -38
  162. data/lib/rubocop/cop/security/eval.rb +15 -2
  163. data/lib/rubocop/cop/security/io_methods.rb +1 -1
  164. data/lib/rubocop/cop/style/access_modifier_declarations.rb +14 -2
  165. data/lib/rubocop/cop/style/accessor_grouping.rb +4 -2
  166. data/lib/rubocop/cop/style/alias.rb +15 -3
  167. data/lib/rubocop/cop/style/and_or.rb +2 -1
  168. data/lib/rubocop/cop/style/arguments_forwarding.rb +25 -7
  169. data/lib/rubocop/cop/style/array_first_last.rb +12 -1
  170. data/lib/rubocop/cop/style/array_intersect.rb +4 -0
  171. data/lib/rubocop/cop/style/array_intersect_with_single_element.rb +3 -0
  172. data/lib/rubocop/cop/style/array_join.rb +4 -2
  173. data/lib/rubocop/cop/style/ascii_comments.rb +6 -3
  174. data/lib/rubocop/cop/style/attr.rb +5 -2
  175. data/lib/rubocop/cop/style/bare_percent_literals.rb +3 -1
  176. data/lib/rubocop/cop/style/begin_block.rb +3 -1
  177. data/lib/rubocop/cop/style/block_delimiters.rb +37 -31
  178. data/lib/rubocop/cop/style/case_equality.rb +18 -2
  179. data/lib/rubocop/cop/style/character_literal.rb +2 -2
  180. data/lib/rubocop/cop/style/class_and_module_children.rb +18 -2
  181. data/lib/rubocop/cop/style/class_equality_comparison.rb +21 -13
  182. data/lib/rubocop/cop/style/class_methods_definitions.rb +11 -5
  183. data/lib/rubocop/cop/style/collection_compact.rb +36 -16
  184. data/lib/rubocop/cop/style/colon_method_call.rb +16 -7
  185. data/lib/rubocop/cop/style/combinable_loops.rb +5 -0
  186. data/lib/rubocop/cop/style/comparable_clamp.rb +12 -1
  187. data/lib/rubocop/cop/style/concat_array_literals.rb +7 -1
  188. data/lib/rubocop/cop/style/conditional_assignment.rb +6 -5
  189. data/lib/rubocop/cop/style/constant_visibility.rb +4 -1
  190. data/lib/rubocop/cop/style/copyright.rb +22 -11
  191. data/lib/rubocop/cop/style/date_time.rb +4 -4
  192. data/lib/rubocop/cop/style/dig_chain.rb +5 -0
  193. data/lib/rubocop/cop/style/disable_cops_within_source_code_directive.rb +1 -1
  194. data/lib/rubocop/cop/style/document_dynamic_eval_definition.rb +6 -1
  195. data/lib/rubocop/cop/style/each_for_simple_loop.rb +1 -1
  196. data/lib/rubocop/cop/style/each_with_object.rb +2 -0
  197. data/lib/rubocop/cop/style/empty_block_parameter.rb +1 -1
  198. data/lib/rubocop/cop/style/empty_class_definition.rb +43 -20
  199. data/lib/rubocop/cop/style/empty_lambda_parameter.rb +1 -1
  200. data/lib/rubocop/cop/style/encoding.rb +7 -1
  201. data/lib/rubocop/cop/style/end_block.rb +3 -1
  202. data/lib/rubocop/cop/style/endless_method.rb +8 -3
  203. data/lib/rubocop/cop/style/fetch_env_var.rb +1 -1
  204. data/lib/rubocop/cop/style/file_open.rb +84 -0
  205. data/lib/rubocop/cop/style/file_write.rb +21 -16
  206. data/lib/rubocop/cop/style/for.rb +3 -0
  207. data/lib/rubocop/cop/style/format_string.rb +4 -3
  208. data/lib/rubocop/cop/style/format_string_token.rb +29 -2
  209. data/lib/rubocop/cop/style/global_vars.rb +5 -2
  210. data/lib/rubocop/cop/style/guard_clause.rb +9 -6
  211. data/lib/rubocop/cop/style/hash_as_last_array_item.rb +21 -5
  212. data/lib/rubocop/cop/style/hash_conversion.rb +1 -1
  213. data/lib/rubocop/cop/style/hash_lookup_method.rb +19 -7
  214. data/lib/rubocop/cop/style/hash_slice.rb +16 -0
  215. data/lib/rubocop/cop/style/hash_transform_keys.rb +17 -7
  216. data/lib/rubocop/cop/style/hash_transform_values.rb +17 -7
  217. data/lib/rubocop/cop/style/if_inside_else.rb +16 -7
  218. data/lib/rubocop/cop/style/if_unless_modifier.rb +15 -4
  219. data/lib/rubocop/cop/style/if_with_semicolon.rb +7 -5
  220. data/lib/rubocop/cop/style/inline_comment.rb +4 -1
  221. data/lib/rubocop/cop/style/ip_addresses.rb +1 -2
  222. data/lib/rubocop/cop/style/magic_comment_format.rb +3 -3
  223. data/lib/rubocop/cop/style/map_join.rb +123 -0
  224. data/lib/rubocop/cop/style/method_call_with_args_parentheses/require_parentheses.rb +5 -3
  225. data/lib/rubocop/cop/style/min_max_comparison.rb +1 -1
  226. data/lib/rubocop/cop/style/module_member_existence_check.rb +7 -14
  227. data/lib/rubocop/cop/style/multiline_if_then.rb +3 -1
  228. data/lib/rubocop/cop/style/mutable_constant.rb +106 -12
  229. data/lib/rubocop/cop/style/nil_comparison.rb +2 -3
  230. data/lib/rubocop/cop/style/nil_lambda.rb +1 -1
  231. data/lib/rubocop/cop/style/non_nil_check.rb +5 -11
  232. data/lib/rubocop/cop/style/not.rb +2 -0
  233. data/lib/rubocop/cop/style/numeric_literals.rb +3 -2
  234. data/lib/rubocop/cop/style/one_class_per_file.rb +115 -0
  235. data/lib/rubocop/cop/style/one_line_conditional.rb +4 -3
  236. data/lib/rubocop/cop/style/parallel_assignment.rb +12 -1
  237. data/lib/rubocop/cop/style/partition_instead_of_double_select.rb +270 -0
  238. data/lib/rubocop/cop/style/percent_literal_delimiters.rb +2 -0
  239. data/lib/rubocop/cop/style/predicate_with_kind.rb +84 -0
  240. data/lib/rubocop/cop/style/proc.rb +3 -2
  241. data/lib/rubocop/cop/style/raise_args.rb +1 -1
  242. data/lib/rubocop/cop/style/reduce_to_hash.rb +200 -0
  243. data/lib/rubocop/cop/style/redundant_array_constructor.rb +2 -2
  244. data/lib/rubocop/cop/style/redundant_begin.rb +3 -3
  245. data/lib/rubocop/cop/style/redundant_constant_base.rb +5 -5
  246. data/lib/rubocop/cop/style/redundant_each.rb +3 -3
  247. data/lib/rubocop/cop/style/redundant_fetch_block.rb +1 -1
  248. data/lib/rubocop/cop/style/redundant_format.rb +1 -0
  249. data/lib/rubocop/cop/style/redundant_interpolation_unfreeze.rb +26 -10
  250. data/lib/rubocop/cop/style/redundant_line_continuation.rb +16 -0
  251. data/lib/rubocop/cop/style/redundant_min_max_by.rb +93 -0
  252. data/lib/rubocop/cop/style/redundant_parentheses.rb +25 -22
  253. data/lib/rubocop/cop/style/redundant_percent_q.rb +4 -1
  254. data/lib/rubocop/cop/style/redundant_regexp_constructor.rb +2 -2
  255. data/lib/rubocop/cop/style/redundant_return.rb +3 -1
  256. data/lib/rubocop/cop/style/redundant_self.rb +2 -2
  257. data/lib/rubocop/cop/style/redundant_self_assignment_branch.rb +0 -5
  258. data/lib/rubocop/cop/style/redundant_struct_keyword_init.rb +114 -0
  259. data/lib/rubocop/cop/style/regexp_literal.rb +31 -2
  260. data/lib/rubocop/cop/style/rescue_modifier.rb +3 -3
  261. data/lib/rubocop/cop/style/safe_navigation.rb +7 -7
  262. data/lib/rubocop/cop/style/select_by_kind.rb +158 -0
  263. data/lib/rubocop/cop/style/select_by_range.rb +197 -0
  264. data/lib/rubocop/cop/style/select_by_regexp.rb +51 -21
  265. data/lib/rubocop/cop/style/self_assignment.rb +1 -1
  266. data/lib/rubocop/cop/style/semicolon.rb +18 -1
  267. data/lib/rubocop/cop/style/single_line_block_params.rb +2 -2
  268. data/lib/rubocop/cop/style/single_line_do_end_block.rb +1 -1
  269. data/lib/rubocop/cop/style/single_line_methods.rb +3 -1
  270. data/lib/rubocop/cop/style/sole_nested_conditional.rb +4 -2
  271. data/lib/rubocop/cop/style/special_global_vars.rb +6 -1
  272. data/lib/rubocop/cop/style/struct_inheritance.rb +13 -0
  273. data/lib/rubocop/cop/style/symbol_proc.rb +7 -6
  274. data/lib/rubocop/cop/style/tally_method.rb +181 -0
  275. data/lib/rubocop/cop/style/top_level_method_definition.rb +2 -2
  276. data/lib/rubocop/cop/style/trailing_comma_in_block_args.rb +1 -1
  277. data/lib/rubocop/cop/style/trailing_method_end_statement.rb +1 -0
  278. data/lib/rubocop/cop/style/unless_logical_operators.rb +3 -3
  279. data/lib/rubocop/cop/style/while_until_do.rb +7 -0
  280. data/lib/rubocop/cop/style/while_until_modifier.rb +16 -0
  281. data/lib/rubocop/cop/style/word_array.rb +1 -0
  282. data/lib/rubocop/cop/style/yoda_condition.rb +1 -1
  283. data/lib/rubocop/cop/style/yoda_expression.rb +1 -1
  284. data/lib/rubocop/cop/style/zero_length_predicate.rb +6 -3
  285. data/lib/rubocop/cop/team.rb +86 -35
  286. data/lib/rubocop/cop/variable_force/branch.rb +2 -2
  287. data/lib/rubocop/directive_comment.rb +2 -1
  288. data/lib/rubocop/file_patterns.rb +9 -1
  289. data/lib/rubocop/formatter/disabled_config_formatter.rb +19 -9
  290. data/lib/rubocop/formatter/formatter_set.rb +1 -1
  291. data/lib/rubocop/formatter/junit_formatter.rb +1 -1
  292. data/lib/rubocop/formatter/simple_text_formatter.rb +0 -2
  293. data/lib/rubocop/formatter/worst_offenders_formatter.rb +1 -1
  294. data/lib/rubocop/formatter.rb +22 -21
  295. data/lib/rubocop/lsp/diagnostic.rb +1 -0
  296. data/lib/rubocop/lsp/routes.rb +10 -3
  297. data/lib/rubocop/lsp/runtime.rb +1 -2
  298. data/lib/rubocop/mcp/server.rb +200 -0
  299. data/lib/rubocop/options.rb +35 -4
  300. data/lib/rubocop/path_util.rb +14 -2
  301. data/lib/rubocop/plugin/loader.rb +1 -1
  302. data/lib/rubocop/project_index_loader.rb +66 -0
  303. data/lib/rubocop/result_cache.rb +22 -10
  304. data/lib/rubocop/rspec/cop_helper.rb +8 -0
  305. data/lib/rubocop/rspec/shared_contexts.rb +32 -2
  306. data/lib/rubocop/runner.rb +124 -53
  307. data/lib/rubocop/server/cache.rb +5 -7
  308. data/lib/rubocop/server/core.rb +8 -0
  309. data/lib/rubocop/target_finder.rb +14 -7
  310. data/lib/rubocop/target_ruby.rb +18 -12
  311. data/lib/rubocop/version.rb +21 -3
  312. data/lib/rubocop.rb +22 -96
  313. metadata +27 -5
@@ -6,6 +6,14 @@ module RuboCop
6
6
  # Checks whether some constant value isn't a
7
7
  # mutable literal (e.g. array or hash).
8
8
  #
9
+ # When the `Recursive` option is enabled, mutable literals nested inside
10
+ # arrays and hashes are also frozen, so an offense on the outermost
11
+ # unfrozen literal will autocorrect every nested mutable literal as well.
12
+ # When the outer literal already has `.freeze` appended, the cop descends
13
+ # into it and reports each outermost unfrozen literal underneath. The
14
+ # option is disabled by default to preserve existing behavior; opt in to
15
+ # get strict nested freezing.
16
+ #
9
17
  # Strict mode can be used to freeze all constants, rather than
10
18
  # just literals.
11
19
  # Strict mode is considered an experimental feature. It has not been
@@ -49,6 +57,17 @@ module RuboCop
49
57
  # CONST = Something.new
50
58
  #
51
59
  #
60
+ # @example Recursive: false (default)
61
+ # # good - only the outer container needs to be frozen
62
+ # CONST = [{ a: [], b: 'foo' }].freeze
63
+ #
64
+ # @example Recursive: true
65
+ # # bad - nested mutable literals must be frozen too
66
+ # CONST = [{ a: [], b: 'foo' }].freeze
67
+ #
68
+ # # good
69
+ # CONST = [{ a: [].freeze, b: 'foo'.freeze }.freeze].freeze
70
+ #
52
71
  # @example EnforcedStyle: strict
53
72
  # # bad
54
73
  # CONST = Something.new
@@ -125,7 +144,7 @@ module RuboCop
125
144
  MSG = 'Freeze mutable objects assigned to constants.'
126
145
 
127
146
  def on_casgn(node)
128
- if node.expression.nil? # This is only the case for `CONST += ...` or similarg66
147
+ if node.expression.nil? # This is only the case for `CONST += ...` or similar
129
148
  parent = node.parent
130
149
  return unless parent.or_asgn_type? # We only care about `CONST ||= ...`
131
150
 
@@ -138,10 +157,30 @@ module RuboCop
138
157
  private
139
158
 
140
159
  def on_assignment(value)
141
- if style == :strict
142
- strict_check(value)
160
+ nodes = mutable_nodes(value) do |node|
161
+ if style == :strict
162
+ strict_check(node)
163
+ else
164
+ literal_check(node)
165
+ end
166
+ end
167
+
168
+ nodes.each do |node|
169
+ add_offense(node) { |corrector| autocorrect(corrector, node) }
170
+ end
171
+ end
172
+
173
+ def mutable_nodes(value, &block)
174
+ if recursive? && explicitly_frozen_literal?(value)
175
+ literal_children(value.receiver).flat_map { |c| mutable_nodes(c, &block) }
143
176
  else
144
- check(value)
177
+ node_offending = yield(value)
178
+
179
+ if node_offending
180
+ [value]
181
+ else
182
+ []
183
+ end
145
184
  end
146
185
  end
147
186
 
@@ -151,18 +190,20 @@ module RuboCop
151
190
  return if frozen_string_literal?(value)
152
191
  return if shareable_constant_value?(value)
153
192
 
154
- add_offense(value) { |corrector| autocorrect(corrector, value) }
193
+ true
155
194
  end
156
195
 
157
- def check(value)
158
- range_enclosed_in_parentheses = range_enclosed_in_parentheses?(value)
159
- return unless mutable_literal?(value) ||
160
- (target_ruby_version <= 2.7 && range_enclosed_in_parentheses)
161
-
196
+ def literal_check(value)
197
+ return unless mutable_or_unfrozen_range?(value)
162
198
  return if frozen_string_literal?(value)
163
199
  return if shareable_constant_value?(value)
164
200
 
165
- add_offense(value) { |corrector| autocorrect(corrector, value) }
201
+ true
202
+ end
203
+
204
+ def mutable_or_unfrozen_range?(value)
205
+ mutable_literal?(value) ||
206
+ (target_ruby_version <= 2.7 && range_enclosed_in_parentheses?(value))
166
207
  end
167
208
 
168
209
  def autocorrect(corrector, node)
@@ -171,13 +212,66 @@ module RuboCop
171
212
  splat_value = splat_value(node)
172
213
  if splat_value
173
214
  correct_splat_expansion(corrector, expr, splat_value)
174
- elsif node.array_type? && !node.bracketed?
215
+ corrector.insert_after(expr, '.freeze')
216
+ return
217
+ end
218
+
219
+ if node.array_type? && !node.bracketed?
175
220
  corrector.wrap(expr, '[', ']')
176
221
  elsif requires_parentheses?(node)
177
222
  corrector.wrap(expr, '(', ')')
178
223
  end
179
224
 
180
225
  corrector.insert_after(expr, '.freeze')
226
+
227
+ freeze_nested_literals(corrector, node) if recursive?
228
+ end
229
+
230
+ # Recursively freezes every nested mutable literal inside an array or
231
+ # hash literal. Already-frozen subtrees are not re-frozen, but their
232
+ # children are still inspected for unfrozen literals deeper down.
233
+ def freeze_nested_literals(corrector, node)
234
+ literal_children(node).each do |child|
235
+ if explicitly_frozen_literal?(child)
236
+ freeze_nested_literals(corrector, child.receiver)
237
+ elsif freezable_nested_literal?(child)
238
+ autocorrect(corrector, child)
239
+ end
240
+ end
241
+ end
242
+
243
+ def freezable_nested_literal?(node)
244
+ return false if frozen_string_literal?(node)
245
+ return false if shareable_constant_value?(node)
246
+
247
+ mutable_literal?(node)
248
+ end
249
+
250
+ # Returns the child literals of an array or hash node that may
251
+ # themselves need freezing. For hashes, both keys and values are
252
+ # included. Percent-literal arrays (e.g. `%w(a b)`) are skipped because
253
+ # `.freeze` cannot be appended to their contents.
254
+ def literal_children(node)
255
+ case node.type
256
+ when :array
257
+ return [] if node.percent_literal?
258
+
259
+ node.children
260
+ when :hash
261
+ node.children.flat_map { |child| child.pair_type? ? child.children : [] }
262
+ else
263
+ []
264
+ end
265
+ end
266
+
267
+ def explicitly_frozen_literal?(node)
268
+ return false unless node.send_type? && node.method?(:freeze)
269
+
270
+ node.receiver && mutable_literal?(node.receiver)
271
+ end
272
+
273
+ def recursive?
274
+ cop_config.fetch('Recursive', false)
181
275
  end
182
276
 
183
277
  def mutable_literal?(value)
@@ -4,9 +4,8 @@ module RuboCop
4
4
  module Cop
5
5
  module Style
6
6
  # Checks for comparison of something with nil using `==` and
7
- # `nil?`.
8
- #
9
- # Supported styles are: predicate, comparison.
7
+ # `nil?`. Enforcing a consistent style (either the `nil?`
8
+ # predicate or `==` comparison) improves readability.
10
9
  #
11
10
  # @example EnforcedStyle: predicate (default)
12
11
  #
@@ -43,7 +43,7 @@ module RuboCop
43
43
  { ({return next break} nil) (nil) }
44
44
  PATTERN
45
45
 
46
- def on_block(node) # rubocop:disable InternalAffairs/NumblockHandler
46
+ def on_block(node) # rubocop:disable InternalAffairs/NumblockHandler, InternalAffairs/ItblockHandler
47
47
  return unless node.lambda_or_proc?
48
48
  return unless nil_return?(node.body)
49
49
 
@@ -121,17 +121,11 @@ module RuboCop
121
121
  end
122
122
 
123
123
  def autocorrect_comparison(corrector, node)
124
- expr = node.source
125
-
126
- new_code = if include_semantic_changes?
127
- expr.sub(/\s*!=\s*nil/, '')
128
- else
129
- expr.sub(/^(\S*)\s*!=\s*nil/, '!\1.nil?')
130
- end
131
-
132
- return if expr == new_code
133
-
134
- corrector.replace(node, new_code)
124
+ if include_semantic_changes?
125
+ corrector.replace(node, node.receiver.source)
126
+ else
127
+ corrector.replace(node, "!#{node.receiver.source}.nil?")
128
+ end
135
129
  end
136
130
 
137
131
  def autocorrect_non_nil(corrector, node, inner_node)
@@ -4,6 +4,8 @@ module RuboCop
4
4
  module Cop
5
5
  module Style
6
6
  # Checks for uses of the keyword `not` instead of `!`.
7
+ # The `not` keyword has lower precedence than `!`, which can
8
+ # lead to surprising behavior and often requires parentheses.
7
9
  #
8
10
  # @example
9
11
  #
@@ -4,7 +4,8 @@ module RuboCop
4
4
  module Cop
5
5
  module Style
6
6
  # Checks for big numeric literals without `_` between groups
7
- # of digits in them.
7
+ # of digits in them. Underscores make large numbers easier to
8
+ # read by visually separating groups of digits.
8
9
  #
9
10
  # Additional allowed patterns can be added by adding regexps to
10
11
  # the `AllowedPatterns` configuration. All regexps are treated
@@ -116,7 +117,7 @@ module RuboCop
116
117
  end
117
118
 
118
119
  def allowed_numbers
119
- cop_config.fetch('AllowedNumbers', []).map(&:to_s)
120
+ @allowed_numbers ||= cop_config.fetch('AllowedNumbers', []).map(&:to_s).freeze
120
121
  end
121
122
 
122
123
  def allowed_patterns
@@ -0,0 +1,115 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Style
6
+ # Checks that each source file defines at most one top-level class or module.
7
+ #
8
+ # Keeping one class or module per file makes it easier to find and navigate
9
+ # code, and follows the convention used by most Ruby projects.
10
+ #
11
+ # Classes and modules listed in `AllowedClasses` are not counted toward the
12
+ # limit. This is useful for small ancillary classes like custom exception
13
+ # classes that logically belong with the main class.
14
+ #
15
+ # @example
16
+ # # bad - Multiple top-level classes
17
+ # class Foo
18
+ # end
19
+ #
20
+ # class Bar
21
+ # end
22
+ #
23
+ # # bad - Multiple top-level modules
24
+ # module Foo
25
+ # end
26
+ #
27
+ # module Bar
28
+ # end
29
+ #
30
+ # # bad - A top-level class and a top-level module
31
+ # class Foo
32
+ # end
33
+ #
34
+ # module Bar
35
+ # end
36
+ #
37
+ # # good - A single top-level class
38
+ # class Foo
39
+ # end
40
+ #
41
+ # # good - A single top-level module
42
+ # module Foo
43
+ # end
44
+ #
45
+ # # good - Nested classes within a single top-level class
46
+ # class Foo
47
+ # class Bar
48
+ # end
49
+ # end
50
+ #
51
+ # # good - Multiple classes within a single top-level module
52
+ # module Foo
53
+ # class Bar
54
+ # end
55
+ #
56
+ # class Baz
57
+ # end
58
+ # end
59
+ #
60
+ # @example AllowedClasses: ['AllowedClass']
61
+ # # good
62
+ # class Foo
63
+ # end
64
+ #
65
+ # class AllowedClass
66
+ # end
67
+ #
68
+ class OneClassPerFile < Base
69
+ include RangeHelp
70
+
71
+ MSG = 'Do not define multiple classes/modules at the top level in a single file.'
72
+
73
+ def on_new_investigation
74
+ @top_level_definitions = []
75
+ end
76
+
77
+ def on_class(node)
78
+ check_top_level(node)
79
+ end
80
+
81
+ def on_module(node)
82
+ check_top_level(node)
83
+ end
84
+
85
+ private
86
+
87
+ def check_top_level(node)
88
+ return unless top_level_definition?(node)
89
+ return if allowed_class?(node)
90
+
91
+ @top_level_definitions << node
92
+ return unless @top_level_definitions.length > 1
93
+
94
+ add_offense(range_between(node.source_range.begin_pos, node.loc.name.end_pos))
95
+ end
96
+
97
+ def top_level_definition?(node)
98
+ if node.parent&.begin_type?
99
+ node.parent.root?
100
+ else
101
+ node.root?
102
+ end
103
+ end
104
+
105
+ def allowed_class?(node)
106
+ allowed_classes.include?(node.identifier.short_name)
107
+ end
108
+
109
+ def allowed_classes
110
+ @allowed_classes ||= cop_config.fetch('AllowedClasses', []).map(&:intern)
111
+ end
112
+ end
113
+ end
114
+ end
115
+ end
@@ -3,9 +3,10 @@
3
3
  module RuboCop
4
4
  module Cop
5
5
  module Style
6
- # Checks for uses of if/then/else/end constructs on a single line.
7
- # `AlwaysCorrectToMultiline` config option can be set to true to autocorrect all offenses to
8
- # multi-line constructs. When `AlwaysCorrectToMultiline` is false (default case) the
6
+ # Checks for uses of `if/then/else/end` constructs on a single line.
7
+ # A ternary operator (`?:`) or multi-line `if` is more readable.
8
+ # `AlwaysCorrectToMultiline` config option can be set to `true` to autocorrect all offenses to
9
+ # multi-line constructs. When `AlwaysCorrectToMultiline` is `false` (default case) the
9
10
  # autocorrect will first try converting them to ternary operators.
10
11
  #
11
12
  # @example
@@ -4,6 +4,10 @@ module RuboCop
4
4
  module Cop
5
5
  module Style
6
6
  # Checks for simple usages of parallel assignment.
7
+ # Parallel assignment is less readable than individual
8
+ # assignments and makes it harder to follow what each
9
+ # variable is being set to.
10
+ #
7
11
  # This will only complain when the number of variables
8
12
  # being assigned matched the number of assigning variables.
9
13
  #
@@ -34,7 +38,7 @@ module RuboCop
34
38
  rhs_elements = Array(rhs).compact # edge case for one constant
35
39
 
36
40
  return if allowed_lhs?(node.assignments) || allowed_rhs?(rhs) ||
37
- allowed_masign?(node.assignments, rhs_elements)
41
+ allowed_masign?(node.assignments, rhs_elements) || contains_heredoc?(rhs)
38
42
 
39
43
  range = node.source_range.begin.join(rhs.source_range.end)
40
44
 
@@ -73,6 +77,13 @@ module RuboCop
73
77
  !node.array_type? || elements.any?(&:splat_type?)
74
78
  end
75
79
 
80
+ # Autocorrection splits the assignment into single assignments on
81
+ # consecutive lines, which would put following assignments into the
82
+ # heredoc body unless the heredoc bodies were moved along.
83
+ def contains_heredoc?(node)
84
+ node.each_descendant(:any_str).any?(&:heredoc?)
85
+ end
86
+
76
87
  def assignment_corrector(node, rhs, order)
77
88
  if node.parent&.rescue_type?
78
89
  _assignment, modifier = *node.parent
@@ -0,0 +1,270 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Style
6
+ # Checks for consecutive calls to `select`/`filter`/`find_all` and `reject`
7
+ # on the same receiver with the same block body, where `partition` could be
8
+ # used instead. Also detects two `select` or two `reject` calls where one
9
+ # block negates the other with `!`. Using `partition` reduces two collection
10
+ # traversals to one.
11
+ #
12
+ # @safety
13
+ # This cop is unsafe because:
14
+ #
15
+ # * `Hash#select` and `Hash#reject` return hashes, but `Hash#partition`
16
+ # returns nested arrays.
17
+ # * When the receiver has side effects, calling it once (with `partition`)
18
+ # versus twice (with `select` + `reject`) may produce different results.
19
+ # * Custom classes may override `select`/`reject` without providing a
20
+ # compatible `partition` method.
21
+ #
22
+ # @example
23
+ # # bad
24
+ # positives = array.select { |x| x > 0 }
25
+ # negatives = array.reject { |x| x > 0 }
26
+ #
27
+ # # bad
28
+ # positives = array.filter { |x| x > 0 }
29
+ # negatives = array.reject { |x| x > 0 }
30
+ #
31
+ # # bad
32
+ # negatives = array.reject { |x| x > 0 }
33
+ # positives = array.select { |x| x > 0 }
34
+ #
35
+ # # bad
36
+ # positives = array.select(&:positive?)
37
+ # negatives = array.reject(&:positive?)
38
+ #
39
+ # # bad
40
+ # positives = array.select(&:positive?)
41
+ # negatives = array.reject { |x| x.positive? }
42
+ #
43
+ # # bad
44
+ # positives = array.select { |x| x.positive? }
45
+ # non_positives = array.select { |x| !x.positive? }
46
+ #
47
+ # # good
48
+ # positives, negatives = array.partition { |x| x > 0 }
49
+ #
50
+ # # good
51
+ # positives, non_positives = array.partition { |x| x.positive? }
52
+ #
53
+ # # good
54
+ # positives, negatives = array.partition(&:positive?)
55
+ #
56
+ class PartitionInsteadOfDoubleSelect < Base
57
+ include RangeHelp
58
+ extend AutoCorrector
59
+
60
+ MSG = 'Use `partition` instead of consecutive `%<first>s` and `%<second>s` calls.'
61
+
62
+ SELECT_METHODS = %i[select filter find_all].freeze
63
+ CANDIDATE_METHODS = (SELECT_METHODS + %i[reject]).to_set.freeze
64
+ RESTRICT_ON_SEND = (SELECT_METHODS + %i[reject]).freeze
65
+
66
+ # @!method symbol_proc_method?(node)
67
+ def_node_matcher :symbol_proc_method?, <<~PATTERN
68
+ (block _ (args (arg _name)) (send (lvar _name) $_method_name))
69
+ PATTERN
70
+
71
+ def on_block(node)
72
+ return unless CANDIDATE_METHODS.include?(node.method_name)
73
+
74
+ find_and_register_offense(node)
75
+ end
76
+ alias on_numblock on_block
77
+ alias on_itblock on_block
78
+
79
+ def on_send(node)
80
+ return unless node.last_argument&.block_pass_type?
81
+
82
+ find_and_register_offense(node)
83
+ end
84
+ alias on_csend on_send
85
+
86
+ private
87
+
88
+ def find_and_register_offense(node)
89
+ container = node_container(node)
90
+ return unless container
91
+
92
+ sibling_container = container.left_sibling
93
+ sibling = find_matching_candidate(node, sibling_container)
94
+ return unless sibling
95
+
96
+ register_offense(node, sibling, container, sibling_container)
97
+ end
98
+
99
+ def node_container(node)
100
+ parent = node.parent
101
+ if parent&.begin_type?
102
+ node
103
+ elsif parent&.assignment? && parent.parent&.begin_type?
104
+ parent
105
+ end
106
+ end
107
+
108
+ def find_matching_candidate(node, sibling_container)
109
+ return unless sibling_container
110
+
111
+ sibling = extract_candidate(sibling_container)
112
+ return unless sibling
113
+ return unless node.receiver == sibling.receiver
114
+ return unless matching_pair?(node, sibling)
115
+
116
+ sibling
117
+ end
118
+
119
+ def matching_pair?(node, sibling)
120
+ (complementary_pair?(node, sibling) && equivalent_predicate?(node, sibling)) ||
121
+ (node.method?(sibling.method_name) && negated_predicate?(node, sibling))
122
+ end
123
+
124
+ def extract_candidate(container)
125
+ extract_block(container) || extract_block_pass_send(container)
126
+ end
127
+
128
+ def extract_block(container)
129
+ if container.any_block_type?
130
+ container
131
+ elsif container.assignment?
132
+ rhs = container.children.last
133
+ rhs if rhs&.any_block_type?
134
+ end
135
+ end
136
+
137
+ def extract_block_pass_send(container)
138
+ node = container.assignment? ? container.children.last : container
139
+ return unless node&.type?(:call)
140
+ return unless node.last_argument&.block_pass_type?
141
+
142
+ node
143
+ end
144
+
145
+ def complementary_pair?(node1, node2)
146
+ m1 = node1.method_name
147
+ m2 = node2.method_name
148
+ (SELECT_METHODS.include?(m1) && m2 == :reject) ||
149
+ (m1 == :reject && SELECT_METHODS.include?(m2))
150
+ end
151
+
152
+ def equivalent_predicate?(node1, node2)
153
+ if node1.any_block_type? && node2.any_block_type?
154
+ same_block_contents?(node1, node2)
155
+ elsif node1.any_block_type?
156
+ block_matches_block_pass?(node1, node2)
157
+ elsif node2.any_block_type?
158
+ block_matches_block_pass?(node2, node1)
159
+ else
160
+ node1.last_argument == node2.last_argument
161
+ end
162
+ end
163
+
164
+ def same_block_contents?(block1, block2)
165
+ return false unless block1.type == block2.type
166
+
167
+ if block1.block_type?
168
+ block1.arguments == block2.arguments &&
169
+ block1.body == block2.body
170
+ else
171
+ block1.body == block2.body
172
+ end
173
+ end
174
+
175
+ def block_matches_block_pass?(block_node, send_node)
176
+ method_name = symbol_proc_method?(block_node)
177
+ return false unless method_name
178
+
179
+ sym_node = send_node.last_argument.children.first
180
+ sym_node.sym_type? && sym_node.children.first == method_name
181
+ end
182
+
183
+ def negated_predicate?(node1, node2)
184
+ return false unless node1.any_block_type? && node2.any_block_type?
185
+ return false unless node1.type == node2.type
186
+ return false if node1.block_type? && node1.arguments != node2.arguments
187
+
188
+ negated_body?(node1.body, node2.body) || negated_body?(node2.body, node1.body)
189
+ end
190
+
191
+ def negated_body?(body1, body2)
192
+ body1&.send_type? && body1.method?(:!) && body1.receiver == body2
193
+ end
194
+
195
+ def register_offense(node, sibling, container, sibling_container)
196
+ message = format(MSG, first: sibling.method_name, second: node.method_name)
197
+
198
+ add_offense(container, message: message) do |corrector|
199
+ next unless both_lvasgn?(container, sibling_container)
200
+
201
+ autocorrect(corrector, node, sibling, container, sibling_container)
202
+ end
203
+ end
204
+
205
+ def both_lvasgn?(container, sibling_container)
206
+ container.lvasgn_type? && sibling_container.lvasgn_type?
207
+ end
208
+
209
+ def autocorrect(corrector, node, sibling, container, sibling_container)
210
+ if complementary_pair?(node, sibling)
211
+ select_var, reject_var =
212
+ complementary_variable_order(sibling, container, sibling_container)
213
+ partition_node = select_node_for(sibling, container)
214
+ else
215
+ select_var, reject_var, partition_node =
216
+ negation_partition_args(node, sibling, container, sibling_container)
217
+ end
218
+
219
+ partition_call = build_partition_call(partition_node)
220
+ replacement = "#{select_var}, #{reject_var} = #{partition_call}"
221
+
222
+ corrector.replace(sibling_container, replacement)
223
+ range = range_by_whole_lines(container.source_range, include_final_newline: true)
224
+ corrector.remove(range)
225
+ end
226
+
227
+ def complementary_variable_order(sibling, container, sibling_container)
228
+ if SELECT_METHODS.include?(sibling.method_name)
229
+ [sibling_container.children.first, container.children.first]
230
+ else
231
+ [container.children.first, sibling_container.children.first]
232
+ end
233
+ end
234
+
235
+ def negation_partition_args(node, sibling, container, sibling_container)
236
+ node_is_negated = negated_body?(node.body, sibling.body)
237
+ is_select = SELECT_METHODS.include?(node.method_name)
238
+ # For select: non-negated is truthy (first). For reject: negated is truthy (first).
239
+ node_is_truthy = is_select != node_is_negated
240
+ partition_node = node_is_negated ? sibling : node
241
+
242
+ if node_is_truthy
243
+ [container.children.first, sibling_container.children.first, partition_node]
244
+ else
245
+ [sibling_container.children.first, container.children.first, partition_node]
246
+ end
247
+ end
248
+
249
+ def select_node_for(sibling, container)
250
+ if SELECT_METHODS.include?(sibling.method_name)
251
+ sibling
252
+ else
253
+ container.children.last
254
+ end
255
+ end
256
+
257
+ def build_partition_call(node)
258
+ source = node.source
259
+ send_node = node.any_block_type? ? node.send_node : node
260
+ selector = send_node.loc.selector
261
+ offset = node.source_range.begin_pos
262
+ method_start = selector.begin_pos - offset
263
+ method_end = selector.end_pos - offset
264
+
265
+ "#{source[0...method_start]}partition#{source[method_end..]}"
266
+ end
267
+ end
268
+ end
269
+ end
270
+ end
@@ -4,6 +4,8 @@ module RuboCop
4
4
  module Cop
5
5
  module Style
6
6
  # Enforces the consistent usage of `%`-literal delimiters.
7
+ # Using consistent delimiters across the codebase reduces
8
+ # cognitive load when reading `%`-literals.
7
9
  #
8
10
  # Specify the 'default' key to set all preferred delimiters at once. You
9
11
  # can continue to specify individual preferred delimiters to override the