rubocop 1.39.0 → 1.57.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (448) hide show
  1. checksums.yaml +4 -4
  2. data/LICENSE.txt +1 -1
  3. data/README.md +6 -4
  4. data/config/default.yml +306 -54
  5. data/config/obsoletion.yml +5 -0
  6. data/exe/rubocop +1 -1
  7. data/lib/rubocop/cli/command/auto_generate_config.rb +7 -0
  8. data/lib/rubocop/cli/command/execute_runner.rb +7 -2
  9. data/lib/rubocop/cli/command/lsp.rb +19 -0
  10. data/lib/rubocop/cli.rb +59 -10
  11. data/lib/rubocop/comment_config.rb +24 -0
  12. data/lib/rubocop/config.rb +46 -18
  13. data/lib/rubocop/config_finder.rb +2 -2
  14. data/lib/rubocop/config_loader.rb +34 -28
  15. data/lib/rubocop/config_loader_resolver.rb +6 -2
  16. data/lib/rubocop/config_obsoletion/parameter_rule.rb +9 -1
  17. data/lib/rubocop/config_obsoletion.rb +2 -2
  18. data/lib/rubocop/config_validator.rb +1 -1
  19. data/lib/rubocop/cop/autocorrect_logic.rb +31 -13
  20. data/lib/rubocop/cop/badge.rb +9 -4
  21. data/lib/rubocop/cop/base.rb +116 -84
  22. data/lib/rubocop/cop/bundler/duplicated_gem.rb +1 -0
  23. data/lib/rubocop/cop/bundler/duplicated_group.rb +127 -0
  24. data/lib/rubocop/cop/bundler/gem_comment.rb +2 -2
  25. data/lib/rubocop/cop/bundler/gem_version.rb +2 -2
  26. data/lib/rubocop/cop/bundler/ordered_gems.rb +9 -1
  27. data/lib/rubocop/cop/commissioner.rb +16 -5
  28. data/lib/rubocop/cop/cop.rb +53 -33
  29. data/lib/rubocop/cop/corrector.rb +31 -11
  30. data/lib/rubocop/cop/correctors/alignment_corrector.rb +3 -3
  31. data/lib/rubocop/cop/correctors/each_to_for_corrector.rb +3 -3
  32. data/lib/rubocop/cop/correctors/for_to_each_corrector.rb +3 -3
  33. data/lib/rubocop/cop/correctors/lambda_literal_to_method_corrector.rb +7 -4
  34. data/lib/rubocop/cop/correctors/line_break_corrector.rb +1 -1
  35. data/lib/rubocop/cop/correctors/multiline_literal_brace_corrector.rb +23 -7
  36. data/lib/rubocop/cop/correctors/ordered_gem_corrector.rb +2 -7
  37. data/lib/rubocop/cop/correctors/parentheses_corrector.rb +1 -1
  38. data/lib/rubocop/cop/correctors/percent_literal_corrector.rb +2 -2
  39. data/lib/rubocop/cop/gemspec/dependency_version.rb +19 -21
  40. data/lib/rubocop/cop/gemspec/deprecated_attribute_assignment.rb +1 -1
  41. data/lib/rubocop/cop/gemspec/development_dependencies.rb +107 -0
  42. data/lib/rubocop/cop/gemspec/ordered_dependencies.rb +9 -1
  43. data/lib/rubocop/cop/generator/require_file_injector.rb +1 -1
  44. data/lib/rubocop/cop/internal_affairs/cop_description.rb +38 -12
  45. data/lib/rubocop/cop/internal_affairs/example_description.rb +42 -21
  46. data/lib/rubocop/cop/internal_affairs/example_heredoc_delimiter.rb +3 -3
  47. data/lib/rubocop/cop/internal_affairs/inherit_deprecated_cop_class.rb +1 -1
  48. data/lib/rubocop/cop/internal_affairs/lambda_or_proc.rb +46 -0
  49. data/lib/rubocop/cop/internal_affairs/location_expression.rb +37 -0
  50. data/lib/rubocop/cop/internal_affairs/location_line_equality_comparison.rb +3 -1
  51. data/lib/rubocop/cop/internal_affairs/node_matcher_directive.rb +6 -6
  52. data/lib/rubocop/cop/internal_affairs/node_type_predicate.rb +1 -1
  53. data/lib/rubocop/cop/internal_affairs/processed_source_buffer_name.rb +42 -0
  54. data/lib/rubocop/cop/internal_affairs/redundant_let_rubocop_config_new.rb +11 -3
  55. data/lib/rubocop/cop/internal_affairs/redundant_location_argument.rb +1 -1
  56. data/lib/rubocop/cop/internal_affairs/redundant_message_argument.rb +1 -1
  57. data/lib/rubocop/cop/internal_affairs/redundant_method_dispatch_node.rb +11 -2
  58. data/lib/rubocop/cop/internal_affairs/redundant_source_range.rb +66 -0
  59. data/lib/rubocop/cop/internal_affairs/useless_message_assertion.rb +2 -0
  60. data/lib/rubocop/cop/internal_affairs.rb +4 -0
  61. data/lib/rubocop/cop/layout/array_alignment.rb +1 -1
  62. data/lib/rubocop/cop/layout/block_end_newline.rb +7 -15
  63. data/lib/rubocop/cop/layout/class_structure.rb +44 -27
  64. data/lib/rubocop/cop/layout/closing_heredoc_indentation.rb +2 -3
  65. data/lib/rubocop/cop/layout/closing_parenthesis_indentation.rb +2 -6
  66. data/lib/rubocop/cop/layout/comment_indentation.rb +3 -1
  67. data/lib/rubocop/cop/layout/dot_position.rb +1 -5
  68. data/lib/rubocop/cop/layout/empty_comment.rb +3 -3
  69. data/lib/rubocop/cop/layout/empty_line_after_guard_clause.rb +42 -9
  70. data/lib/rubocop/cop/layout/empty_line_between_defs.rb +28 -5
  71. data/lib/rubocop/cop/layout/empty_lines.rb +3 -1
  72. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +2 -0
  73. data/lib/rubocop/cop/layout/empty_lines_around_exception_handling_keywords.rb +2 -0
  74. data/lib/rubocop/cop/layout/end_alignment.rb +9 -1
  75. data/lib/rubocop/cop/layout/extra_spacing.rb +15 -6
  76. data/lib/rubocop/cop/layout/first_argument_indentation.rb +8 -3
  77. data/lib/rubocop/cop/layout/first_array_element_line_break.rb +35 -8
  78. data/lib/rubocop/cop/layout/first_hash_element_line_break.rb +36 -1
  79. data/lib/rubocop/cop/layout/first_method_argument_line_break.rb +57 -8
  80. data/lib/rubocop/cop/layout/first_method_parameter_line_break.rb +52 -19
  81. data/lib/rubocop/cop/layout/heredoc_argument_closing_parenthesis.rb +10 -4
  82. data/lib/rubocop/cop/layout/heredoc_indentation.rb +11 -11
  83. data/lib/rubocop/cop/layout/indentation_style.rb +8 -3
  84. data/lib/rubocop/cop/layout/indentation_width.rb +3 -3
  85. data/lib/rubocop/cop/layout/initial_indentation.rb +1 -1
  86. data/lib/rubocop/cop/layout/leading_comment_space.rb +2 -2
  87. data/lib/rubocop/cop/layout/line_continuation_leading_space.rb +21 -10
  88. data/lib/rubocop/cop/layout/line_continuation_spacing.rb +19 -9
  89. data/lib/rubocop/cop/layout/line_end_string_concatenation_indentation.rb +2 -0
  90. data/lib/rubocop/cop/layout/line_length.rb +2 -0
  91. data/lib/rubocop/cop/layout/multiline_array_line_breaks.rb +31 -1
  92. data/lib/rubocop/cop/layout/multiline_block_layout.rb +1 -1
  93. data/lib/rubocop/cop/layout/multiline_hash_key_line_breaks.rb +29 -1
  94. data/lib/rubocop/cop/layout/multiline_method_argument_line_breaks.rb +35 -1
  95. data/lib/rubocop/cop/layout/multiline_method_call_indentation.rb +18 -3
  96. data/lib/rubocop/cop/layout/multiline_method_parameter_line_breaks.rb +33 -1
  97. data/lib/rubocop/cop/layout/redundant_line_break.rb +22 -13
  98. data/lib/rubocop/cop/layout/rescue_ensure_alignment.rb +2 -2
  99. data/lib/rubocop/cop/layout/space_after_comma.rb +9 -1
  100. data/lib/rubocop/cop/layout/space_after_not.rb +1 -1
  101. data/lib/rubocop/cop/layout/space_around_keyword.rb +2 -2
  102. data/lib/rubocop/cop/layout/space_around_method_call_operator.rb +2 -2
  103. data/lib/rubocop/cop/layout/space_around_operators.rb +4 -2
  104. data/lib/rubocop/cop/layout/space_before_first_arg.rb +1 -1
  105. data/lib/rubocop/cop/layout/space_in_lambda_literal.rb +2 -2
  106. data/lib/rubocop/cop/layout/space_inside_array_literal_brackets.rb +11 -13
  107. data/lib/rubocop/cop/layout/space_inside_block_braces.rb +3 -1
  108. data/lib/rubocop/cop/layout/space_inside_parens.rb +3 -3
  109. data/lib/rubocop/cop/layout/space_inside_percent_literal_delimiters.rb +1 -1
  110. data/lib/rubocop/cop/layout/space_inside_range_literal.rb +1 -1
  111. data/lib/rubocop/cop/layout/space_inside_reference_brackets.rb +4 -4
  112. data/lib/rubocop/cop/layout/space_inside_string_interpolation.rb +5 -4
  113. data/lib/rubocop/cop/layout/trailing_empty_lines.rb +6 -1
  114. data/lib/rubocop/cop/layout/trailing_whitespace.rb +12 -5
  115. data/lib/rubocop/cop/lint/ambiguous_block_association.rb +14 -2
  116. data/lib/rubocop/cop/lint/ambiguous_operator.rb +4 -0
  117. data/lib/rubocop/cop/lint/assignment_in_condition.rb +11 -1
  118. data/lib/rubocop/cop/lint/constant_resolution.rb +5 -1
  119. data/lib/rubocop/cop/lint/debugger.rb +23 -24
  120. data/lib/rubocop/cop/lint/deprecated_class_methods.rb +62 -112
  121. data/lib/rubocop/cop/lint/deprecated_constants.rb +8 -1
  122. data/lib/rubocop/cop/lint/deprecated_open_ssl_constant.rb +1 -1
  123. data/lib/rubocop/cop/lint/duplicate_branch.rb +0 -2
  124. data/lib/rubocop/cop/lint/duplicate_hash_key.rb +2 -1
  125. data/lib/rubocop/cop/lint/duplicate_match_pattern.rb +122 -0
  126. data/lib/rubocop/cop/lint/duplicate_methods.rb +20 -9
  127. data/lib/rubocop/cop/lint/duplicate_regexp_character_class_element.rb +47 -22
  128. data/lib/rubocop/cop/lint/else_layout.rb +3 -7
  129. data/lib/rubocop/cop/lint/empty_block.rb +3 -7
  130. data/lib/rubocop/cop/lint/empty_conditional_body.rb +5 -3
  131. data/lib/rubocop/cop/lint/empty_interpolation.rb +1 -1
  132. data/lib/rubocop/cop/lint/erb_new_arguments.rb +3 -4
  133. data/lib/rubocop/cop/lint/format_parameter_mismatch.rb +14 -7
  134. data/lib/rubocop/cop/lint/heredoc_method_call_position.rb +16 -18
  135. data/lib/rubocop/cop/lint/identity_comparison.rb +0 -1
  136. data/lib/rubocop/cop/lint/implicit_string_concatenation.rb +1 -1
  137. data/lib/rubocop/cop/lint/incompatible_io_select_with_fiber_scheduler.rb +5 -3
  138. data/lib/rubocop/cop/lint/ineffective_access_modifier.rb +1 -1
  139. data/lib/rubocop/cop/lint/inherit_exception.rb +9 -0
  140. data/lib/rubocop/cop/lint/interpolation_check.rb +4 -3
  141. data/lib/rubocop/cop/lint/lambda_without_literal_block.rb +1 -1
  142. data/lib/rubocop/cop/lint/literal_in_interpolation.rb +47 -5
  143. data/lib/rubocop/cop/lint/missing_super.rb +63 -5
  144. data/lib/rubocop/cop/lint/mixed_case_range.rb +111 -0
  145. data/lib/rubocop/cop/lint/mixed_regexp_capture_types.rb +1 -0
  146. data/lib/rubocop/cop/lint/nested_method_definition.rb +4 -9
  147. data/lib/rubocop/cop/lint/non_atomic_file_operation.rb +20 -12
  148. data/lib/rubocop/cop/lint/number_conversion.rb +5 -0
  149. data/lib/rubocop/cop/lint/numbered_parameter_assignment.rb +2 -2
  150. data/lib/rubocop/cop/lint/or_assignment_to_constant.rb +2 -0
  151. data/lib/rubocop/cop/lint/ordered_magic_comments.rb +0 -1
  152. data/lib/rubocop/cop/lint/out_of_range_regexp_ref.rb +16 -1
  153. data/lib/rubocop/cop/lint/parentheses_as_grouped_expression.rb +5 -0
  154. data/lib/rubocop/cop/lint/percent_string_array.rb +1 -1
  155. data/lib/rubocop/cop/lint/percent_symbol_array.rb +1 -1
  156. data/lib/rubocop/cop/lint/redundant_cop_disable_directive.rb +26 -8
  157. data/lib/rubocop/cop/lint/redundant_cop_enable_directive.rb +6 -6
  158. data/lib/rubocop/cop/lint/redundant_regexp_quantifiers.rb +130 -0
  159. data/lib/rubocop/cop/lint/redundant_require_statement.rb +21 -2
  160. data/lib/rubocop/cop/lint/redundant_safe_navigation.rb +20 -4
  161. data/lib/rubocop/cop/lint/redundant_splat_expansion.rb +1 -1
  162. data/lib/rubocop/cop/lint/redundant_string_coercion.rb +35 -15
  163. data/lib/rubocop/cop/lint/redundant_with_index.rb +1 -1
  164. data/lib/rubocop/cop/lint/redundant_with_object.rb +1 -1
  165. data/lib/rubocop/cop/lint/refinement_import_methods.rb +2 -1
  166. data/lib/rubocop/cop/lint/regexp_as_condition.rb +6 -0
  167. data/lib/rubocop/cop/lint/require_parentheses.rb +3 -1
  168. data/lib/rubocop/cop/lint/rescue_type.rb +3 -3
  169. data/lib/rubocop/cop/lint/safe_navigation_chain.rb +21 -16
  170. data/lib/rubocop/cop/lint/safe_navigation_consistency.rb +1 -1
  171. data/lib/rubocop/cop/lint/script_permission.rb +1 -1
  172. data/lib/rubocop/cop/lint/send_with_mixin_argument.rb +6 -6
  173. data/lib/rubocop/cop/lint/shadowed_exception.rb +6 -12
  174. data/lib/rubocop/cop/lint/shadowing_outer_local_variable.rb +11 -4
  175. data/lib/rubocop/cop/lint/struct_new_override.rb +12 -12
  176. data/lib/rubocop/cop/lint/suppressed_exception.rb +2 -2
  177. data/lib/rubocop/cop/lint/symbol_conversion.rb +1 -1
  178. data/lib/rubocop/cop/lint/syntax.rb +4 -0
  179. data/lib/rubocop/cop/lint/to_enum_arguments.rb +18 -6
  180. data/lib/rubocop/cop/lint/top_level_return_with_argument.rb +23 -9
  181. data/lib/rubocop/cop/lint/unreachable_loop.rb +3 -3
  182. data/lib/rubocop/cop/lint/unused_method_argument.rb +2 -1
  183. data/lib/rubocop/cop/lint/useless_access_modifier.rb +10 -7
  184. data/lib/rubocop/cop/lint/useless_assignment.rb +94 -10
  185. data/lib/rubocop/cop/lint/useless_method_definition.rb +12 -4
  186. data/lib/rubocop/cop/lint/useless_rescue.rb +89 -0
  187. data/lib/rubocop/cop/lint/useless_ruby2_keywords.rb +14 -4
  188. data/lib/rubocop/cop/lint/useless_times.rb +1 -1
  189. data/lib/rubocop/cop/lint/void.rb +84 -21
  190. data/lib/rubocop/cop/metrics/block_length.rb +11 -6
  191. data/lib/rubocop/cop/metrics/block_nesting.rb +2 -2
  192. data/lib/rubocop/cop/metrics/class_length.rb +12 -6
  193. data/lib/rubocop/cop/metrics/collection_literal_length.rb +76 -0
  194. data/lib/rubocop/cop/metrics/cyclomatic_complexity.rb +1 -1
  195. data/lib/rubocop/cop/metrics/method_length.rb +10 -5
  196. data/lib/rubocop/cop/metrics/module_length.rb +10 -5
  197. data/lib/rubocop/cop/metrics/parameter_lists.rb +27 -0
  198. data/lib/rubocop/cop/metrics/perceived_complexity.rb +1 -1
  199. data/lib/rubocop/cop/metrics/utils/abc_size_calculator.rb +4 -8
  200. data/lib/rubocop/cop/metrics/utils/code_length_calculator.rb +39 -8
  201. data/lib/rubocop/cop/migration/department_name.rb +3 -3
  202. data/lib/rubocop/cop/mixin/alignment.rb +2 -2
  203. data/lib/rubocop/cop/mixin/allowed_identifiers.rb +2 -2
  204. data/lib/rubocop/cop/mixin/allowed_methods.rb +3 -1
  205. data/lib/rubocop/cop/mixin/allowed_receivers.rb +34 -0
  206. data/lib/rubocop/cop/mixin/annotation_comment.rb +14 -7
  207. data/lib/rubocop/cop/mixin/code_length.rb +1 -1
  208. data/lib/rubocop/cop/mixin/comments_help.rb +13 -7
  209. data/lib/rubocop/cop/mixin/configurable_enforced_style.rb +21 -9
  210. data/lib/rubocop/cop/mixin/def_node.rb +1 -1
  211. data/lib/rubocop/cop/mixin/documentation_comment.rb +1 -1
  212. data/lib/rubocop/cop/mixin/end_keyword_alignment.rb +1 -1
  213. data/lib/rubocop/cop/mixin/first_element_line_break.rb +11 -7
  214. data/lib/rubocop/cop/mixin/hash_alignment_styles.rb +1 -1
  215. data/lib/rubocop/cop/mixin/hash_shorthand_syntax.rb +75 -13
  216. data/lib/rubocop/cop/mixin/hash_transform_method.rb +3 -3
  217. data/lib/rubocop/cop/mixin/heredoc.rb +6 -2
  218. data/lib/rubocop/cop/mixin/line_length_help.rb +11 -2
  219. data/lib/rubocop/cop/mixin/method_complexity.rb +5 -3
  220. data/lib/rubocop/cop/mixin/min_branches_count.rb +40 -0
  221. data/lib/rubocop/cop/mixin/multiline_element_line_breaks.rb +5 -6
  222. data/lib/rubocop/cop/mixin/multiline_expression_indentation.rb +3 -2
  223. data/lib/rubocop/cop/mixin/ordered_gem_node.rb +1 -1
  224. data/lib/rubocop/cop/mixin/percent_array.rb +3 -5
  225. data/lib/rubocop/cop/mixin/percent_literal.rb +1 -1
  226. data/lib/rubocop/cop/mixin/preceding_following_alignment.rb +6 -8
  227. data/lib/rubocop/cop/mixin/range_help.rb +1 -6
  228. data/lib/rubocop/cop/mixin/require_library.rb +2 -0
  229. data/lib/rubocop/cop/mixin/rescue_node.rb +3 -3
  230. data/lib/rubocop/cop/mixin/space_after_punctuation.rb +1 -1
  231. data/lib/rubocop/cop/mixin/statement_modifier.rb +18 -3
  232. data/lib/rubocop/cop/mixin/string_help.rb +4 -2
  233. data/lib/rubocop/cop/mixin/surrounding_space.rb +3 -3
  234. data/lib/rubocop/cop/mixin/trailing_comma.rb +3 -3
  235. data/lib/rubocop/cop/naming/ascii_identifiers.rb +1 -1
  236. data/lib/rubocop/cop/naming/block_forwarding.rb +6 -2
  237. data/lib/rubocop/cop/naming/class_and_module_camel_case.rb +3 -1
  238. data/lib/rubocop/cop/naming/constant_name.rb +1 -1
  239. data/lib/rubocop/cop/naming/file_name.rb +1 -1
  240. data/lib/rubocop/cop/naming/heredoc_delimiter_case.rb +1 -1
  241. data/lib/rubocop/cop/naming/heredoc_delimiter_naming.rb +3 -1
  242. data/lib/rubocop/cop/naming/inclusive_language.rb +27 -5
  243. data/lib/rubocop/cop/naming/memoized_instance_variable_name.rb +25 -10
  244. data/lib/rubocop/cop/naming/method_name.rb +3 -3
  245. data/lib/rubocop/cop/naming/predicate_name.rb +1 -1
  246. data/lib/rubocop/cop/naming/rescued_exceptions_variable_name.rb +12 -4
  247. data/lib/rubocop/cop/naming/variable_name.rb +6 -1
  248. data/lib/rubocop/cop/registry.rb +66 -44
  249. data/lib/rubocop/cop/security/compound_hash.rb +2 -1
  250. data/lib/rubocop/cop/style/access_modifier_declarations.rb +26 -11
  251. data/lib/rubocop/cop/style/accessor_grouping.rb +43 -17
  252. data/lib/rubocop/cop/style/alias.rb +18 -9
  253. data/lib/rubocop/cop/style/arguments_forwarding.rb +280 -62
  254. data/lib/rubocop/cop/style/array_intersect.rb +119 -0
  255. data/lib/rubocop/cop/style/ascii_comments.rb +1 -1
  256. data/lib/rubocop/cop/style/attr.rb +11 -1
  257. data/lib/rubocop/cop/style/begin_block.rb +1 -2
  258. data/lib/rubocop/cop/style/bisected_attr_accessor.rb +1 -1
  259. data/lib/rubocop/cop/style/block_comments.rb +3 -3
  260. data/lib/rubocop/cop/style/block_delimiters.rb +22 -6
  261. data/lib/rubocop/cop/style/case_like_if.rb +20 -3
  262. data/lib/rubocop/cop/style/class_and_module_children.rb +5 -12
  263. data/lib/rubocop/cop/style/class_equality_comparison.rb +58 -40
  264. data/lib/rubocop/cop/style/collection_compact.rb +20 -7
  265. data/lib/rubocop/cop/style/collection_methods.rb +2 -0
  266. data/lib/rubocop/cop/style/colon_method_call.rb +2 -2
  267. data/lib/rubocop/cop/style/combinable_loops.rb +30 -8
  268. data/lib/rubocop/cop/style/command_literal.rb +1 -1
  269. data/lib/rubocop/cop/style/comment_annotation.rb +1 -1
  270. data/lib/rubocop/cop/style/commented_keyword.rb +2 -2
  271. data/lib/rubocop/cop/style/comparable_clamp.rb +125 -0
  272. data/lib/rubocop/cop/style/concat_array_literals.rb +94 -0
  273. data/lib/rubocop/cop/style/conditional_assignment.rb +11 -15
  274. data/lib/rubocop/cop/style/copyright.rb +6 -3
  275. data/lib/rubocop/cop/style/data_inheritance.rb +75 -0
  276. data/lib/rubocop/cop/style/dir.rb +1 -1
  277. data/lib/rubocop/cop/style/dir_empty.rb +54 -0
  278. data/lib/rubocop/cop/style/disable_cops_within_source_code_directive.rb +2 -2
  279. data/lib/rubocop/cop/style/document_dynamic_eval_definition.rb +3 -3
  280. data/lib/rubocop/cop/style/documentation.rb +22 -10
  281. data/lib/rubocop/cop/style/documentation_method.rb +10 -4
  282. data/lib/rubocop/cop/style/double_negation.rb +2 -2
  283. data/lib/rubocop/cop/style/each_with_object.rb +1 -1
  284. data/lib/rubocop/cop/style/empty_block_parameter.rb +1 -1
  285. data/lib/rubocop/cop/style/empty_case_condition.rb +6 -1
  286. data/lib/rubocop/cop/style/empty_lambda_parameter.rb +1 -1
  287. data/lib/rubocop/cop/style/eval_with_location.rb +8 -8
  288. data/lib/rubocop/cop/style/exact_regexp_match.rb +68 -0
  289. data/lib/rubocop/cop/style/explicit_block_argument.rb +1 -1
  290. data/lib/rubocop/cop/style/file_empty.rb +71 -0
  291. data/lib/rubocop/cop/style/file_read.rb +3 -3
  292. data/lib/rubocop/cop/style/file_write.rb +1 -1
  293. data/lib/rubocop/cop/style/for.rb +1 -1
  294. data/lib/rubocop/cop/style/format_string.rb +24 -3
  295. data/lib/rubocop/cop/style/frozen_string_literal_comment.rb +4 -2
  296. data/lib/rubocop/cop/style/guard_clause.rb +73 -10
  297. data/lib/rubocop/cop/style/hash_conversion.rb +10 -0
  298. data/lib/rubocop/cop/style/hash_each_methods.rb +1 -10
  299. data/lib/rubocop/cop/style/hash_except.rb +23 -12
  300. data/lib/rubocop/cop/style/hash_like_case.rb +3 -9
  301. data/lib/rubocop/cop/style/hash_syntax.rb +16 -9
  302. data/lib/rubocop/cop/style/hash_transform_keys.rb +2 -2
  303. data/lib/rubocop/cop/style/hash_transform_values.rb +2 -2
  304. data/lib/rubocop/cop/style/identical_conditional_branches.rb +36 -3
  305. data/lib/rubocop/cop/style/if_inside_else.rb +6 -0
  306. data/lib/rubocop/cop/style/if_unless_modifier.rb +111 -15
  307. data/lib/rubocop/cop/style/if_with_boolean_literal_branches.rb +2 -0
  308. data/lib/rubocop/cop/style/if_with_semicolon.rb +6 -6
  309. data/lib/rubocop/cop/style/infinite_loop.rb +2 -5
  310. data/lib/rubocop/cop/style/inverse_methods.rb +7 -5
  311. data/lib/rubocop/cop/style/invertible_unless_condition.rb +118 -0
  312. data/lib/rubocop/cop/style/lambda.rb +3 -3
  313. data/lib/rubocop/cop/style/lambda_call.rb +5 -0
  314. data/lib/rubocop/cop/style/line_end_concatenation.rb +4 -1
  315. data/lib/rubocop/cop/style/map_compact_with_conditional_block.rb +2 -2
  316. data/lib/rubocop/cop/style/map_to_hash.rb +4 -1
  317. data/lib/rubocop/cop/style/map_to_set.rb +64 -0
  318. data/lib/rubocop/cop/style/method_call_with_args_parentheses/omit_parentheses.rb +35 -24
  319. data/lib/rubocop/cop/style/method_call_with_args_parentheses.rb +44 -37
  320. data/lib/rubocop/cop/style/method_call_without_args_parentheses.rb +2 -0
  321. data/lib/rubocop/cop/style/method_def_parentheses.rb +11 -4
  322. data/lib/rubocop/cop/style/min_max.rb +3 -3
  323. data/lib/rubocop/cop/style/min_max_comparison.rb +83 -0
  324. data/lib/rubocop/cop/style/missing_else.rb +13 -1
  325. data/lib/rubocop/cop/style/mixin_grouping.rb +5 -5
  326. data/lib/rubocop/cop/style/multiline_block_chain.rb +1 -1
  327. data/lib/rubocop/cop/style/multiline_if_modifier.rb +0 -4
  328. data/lib/rubocop/cop/style/multiline_memoization.rb +2 -2
  329. data/lib/rubocop/cop/style/multiline_method_signature.rb +7 -4
  330. data/lib/rubocop/cop/style/multiline_ternary_operator.rb +19 -4
  331. data/lib/rubocop/cop/style/multiple_comparison.rb +14 -0
  332. data/lib/rubocop/cop/style/negated_if_else_condition.rb +13 -12
  333. data/lib/rubocop/cop/style/nested_ternary_operator.rb +3 -11
  334. data/lib/rubocop/cop/style/nil_lambda.rb +3 -3
  335. data/lib/rubocop/cop/style/numbered_parameters_limit.rb +11 -3
  336. data/lib/rubocop/cop/style/numeric_literals.rb +1 -1
  337. data/lib/rubocop/cop/style/one_line_conditional.rb +3 -6
  338. data/lib/rubocop/cop/style/open_struct_use.rb +1 -1
  339. data/lib/rubocop/cop/style/operator_method_call.rb +22 -2
  340. data/lib/rubocop/cop/style/parallel_assignment.rb +29 -19
  341. data/lib/rubocop/cop/style/percent_literal_delimiters.rb +2 -3
  342. data/lib/rubocop/cop/style/percent_q_literals.rb +1 -1
  343. data/lib/rubocop/cop/style/preferred_hash_methods.rb +1 -1
  344. data/lib/rubocop/cop/style/redundant_argument.rb +9 -1
  345. data/lib/rubocop/cop/style/redundant_array_constructor.rb +77 -0
  346. data/lib/rubocop/cop/style/redundant_begin.rb +10 -2
  347. data/lib/rubocop/cop/style/redundant_condition.rb +18 -3
  348. data/lib/rubocop/cop/style/redundant_conditional.rb +2 -14
  349. data/lib/rubocop/cop/style/redundant_constant_base.rb +85 -0
  350. data/lib/rubocop/cop/style/redundant_current_directory_in_path.rb +38 -0
  351. data/lib/rubocop/cop/style/redundant_double_splat_hash_braces.rb +126 -0
  352. data/lib/rubocop/cop/style/redundant_exception.rb +32 -12
  353. data/lib/rubocop/cop/style/redundant_fetch_block.rb +6 -4
  354. data/lib/rubocop/cop/style/redundant_filter_chain.rb +117 -0
  355. data/lib/rubocop/cop/style/redundant_heredoc_delimiter_quotes.rb +58 -0
  356. data/lib/rubocop/cop/style/redundant_interpolation.rb +2 -2
  357. data/lib/rubocop/cop/style/redundant_line_continuation.rb +183 -0
  358. data/lib/rubocop/cop/style/redundant_parentheses.rb +26 -8
  359. data/lib/rubocop/cop/style/redundant_percent_q.rb +1 -1
  360. data/lib/rubocop/cop/style/redundant_regexp_argument.rb +100 -0
  361. data/lib/rubocop/cop/style/redundant_regexp_character_class.rb +7 -8
  362. data/lib/rubocop/cop/style/redundant_regexp_constructor.rb +46 -0
  363. data/lib/rubocop/cop/style/redundant_regexp_escape.rb +13 -4
  364. data/lib/rubocop/cop/style/redundant_return.rb +14 -2
  365. data/lib/rubocop/cop/style/redundant_self_assignment_branch.rb +8 -1
  366. data/lib/rubocop/cop/style/redundant_sort.rb +5 -5
  367. data/lib/rubocop/cop/style/redundant_string_escape.rb +11 -7
  368. data/lib/rubocop/cop/style/regexp_literal.rb +11 -2
  369. data/lib/rubocop/cop/style/require_order.rb +139 -0
  370. data/lib/rubocop/cop/style/rescue_modifier.rb +1 -3
  371. data/lib/rubocop/cop/style/rescue_standard_error.rb +2 -2
  372. data/lib/rubocop/cop/style/return_nil.rb +6 -2
  373. data/lib/rubocop/cop/style/return_nil_in_predicate_method_definition.rb +95 -0
  374. data/lib/rubocop/cop/style/safe_navigation.rb +37 -8
  375. data/lib/rubocop/cop/style/select_by_regexp.rb +23 -5
  376. data/lib/rubocop/cop/style/self_assignment.rb +2 -2
  377. data/lib/rubocop/cop/style/semicolon.rb +37 -6
  378. data/lib/rubocop/cop/style/signal_exception.rb +9 -7
  379. data/lib/rubocop/cop/style/single_line_do_end_block.rb +65 -0
  380. data/lib/rubocop/cop/style/single_line_methods.rb +1 -1
  381. data/lib/rubocop/cop/style/slicing_with_range.rb +1 -1
  382. data/lib/rubocop/cop/style/sole_nested_conditional.rb +9 -5
  383. data/lib/rubocop/cop/style/special_global_vars.rb +3 -4
  384. data/lib/rubocop/cop/style/stderr_puts.rb +1 -1
  385. data/lib/rubocop/cop/style/string_hash_keys.rb +4 -1
  386. data/lib/rubocop/cop/style/string_literals.rb +1 -5
  387. data/lib/rubocop/cop/style/string_literals_in_interpolation.rb +30 -5
  388. data/lib/rubocop/cop/style/struct_inheritance.rb +1 -1
  389. data/lib/rubocop/cop/style/symbol_array.rb +35 -15
  390. data/lib/rubocop/cop/style/symbol_proc.rb +2 -4
  391. data/lib/rubocop/cop/style/trailing_body_on_class.rb +1 -0
  392. data/lib/rubocop/cop/style/trailing_comma_in_arguments.rb +4 -4
  393. data/lib/rubocop/cop/style/trailing_underscore_variable.rb +1 -1
  394. data/lib/rubocop/cop/style/trivial_accessors.rb +1 -1
  395. data/lib/rubocop/cop/style/unless_logical_operators.rb +1 -0
  396. data/lib/rubocop/cop/style/unpack_first.rb +3 -3
  397. data/lib/rubocop/cop/style/word_array.rb +54 -1
  398. data/lib/rubocop/cop/style/yaml_file_read.rb +66 -0
  399. data/lib/rubocop/cop/style/yoda_condition.rb +17 -8
  400. data/lib/rubocop/cop/style/yoda_expression.rb +91 -0
  401. data/lib/rubocop/cop/style/zero_length_predicate.rb +40 -19
  402. data/lib/rubocop/cop/team.rb +61 -53
  403. data/lib/rubocop/cop/util.rb +43 -7
  404. data/lib/rubocop/cop/utils/regexp_ranges.rb +113 -0
  405. data/lib/rubocop/cop/variable_force/assignment.rb +45 -4
  406. data/lib/rubocop/cop/variable_force/scope.rb +3 -3
  407. data/lib/rubocop/cop/variable_force/variable.rb +5 -3
  408. data/lib/rubocop/cop/variable_force/variable_table.rb +5 -3
  409. data/lib/rubocop/cop/variable_force.rb +19 -30
  410. data/lib/rubocop/cops_documentation_generator.rb +44 -15
  411. data/lib/rubocop/directive_comment.rb +4 -4
  412. data/lib/rubocop/ext/comment.rb +18 -0
  413. data/lib/rubocop/ext/regexp_node.rb +1 -1
  414. data/lib/rubocop/ext/regexp_parser.rb +5 -2
  415. data/lib/rubocop/file_finder.rb +4 -7
  416. data/lib/rubocop/file_patterns.rb +43 -0
  417. data/lib/rubocop/formatter/disabled_config_formatter.rb +17 -6
  418. data/lib/rubocop/formatter/html_formatter.rb +1 -1
  419. data/lib/rubocop/formatter/junit_formatter.rb +4 -1
  420. data/lib/rubocop/formatter/simple_text_formatter.rb +1 -1
  421. data/lib/rubocop/formatter.rb +4 -1
  422. data/lib/rubocop/lsp/logger.rb +22 -0
  423. data/lib/rubocop/lsp/routes.rb +246 -0
  424. data/lib/rubocop/lsp/runtime.rb +99 -0
  425. data/lib/rubocop/lsp/server.rb +68 -0
  426. data/lib/rubocop/lsp/severity.rb +27 -0
  427. data/lib/rubocop/magic_comment.rb +12 -10
  428. data/lib/rubocop/options.rb +45 -3
  429. data/lib/rubocop/path_util.rb +50 -22
  430. data/lib/rubocop/result_cache.rb +8 -4
  431. data/lib/rubocop/rspec/cop_helper.rb +6 -3
  432. data/lib/rubocop/rspec/expect_offense.rb +6 -4
  433. data/lib/rubocop/rspec/shared_contexts.rb +6 -3
  434. data/lib/rubocop/rspec/support.rb +3 -2
  435. data/lib/rubocop/runner.rb +55 -10
  436. data/lib/rubocop/server/cache.rb +12 -4
  437. data/lib/rubocop/server/cli.rb +37 -18
  438. data/lib/rubocop/server/client_command/exec.rb +4 -3
  439. data/lib/rubocop/server/client_command/start.rb +6 -1
  440. data/lib/rubocop/server/core.rb +25 -10
  441. data/lib/rubocop/server/helper.rb +1 -1
  442. data/lib/rubocop/server/server_command/exec.rb +1 -1
  443. data/lib/rubocop/string_interpreter.rb +3 -3
  444. data/lib/rubocop/target_finder.rb +8 -4
  445. data/lib/rubocop/target_ruby.rb +14 -10
  446. data/lib/rubocop/version.rb +10 -6
  447. data/lib/rubocop.rb +47 -6
  448. metadata +91 -37
@@ -3,41 +3,59 @@
3
3
  module RuboCop
4
4
  # Common methods and behaviors for dealing with paths.
5
5
  module PathUtil
6
+ class << self
7
+ attr_accessor :relative_paths_cache
8
+ end
9
+ self.relative_paths_cache = Hash.new { |hash, key| hash[key] = {} }
10
+
6
11
  module_function
7
12
 
8
13
  def relative_path(path, base_dir = Dir.pwd)
9
- # Optimization for the common case where path begins with the base
10
- # dir. Just cut off the first part.
11
- if path.start_with?(base_dir)
12
- base_dir_length = base_dir.length
13
- result_length = path.length - base_dir_length - 1
14
- return path[base_dir_length + 1, result_length]
15
- end
16
-
17
- path_name = Pathname.new(File.expand_path(path))
18
- begin
19
- path_name.relative_path_from(Pathname.new(base_dir)).to_s
20
- rescue ArgumentError
21
- path
22
- end
14
+ PathUtil.relative_paths_cache[base_dir][path] ||=
15
+ # Optimization for the common case where path begins with the base
16
+ # dir. Just cut off the first part.
17
+ if path.start_with?(base_dir)
18
+ base_dir_length = base_dir.length
19
+ result_length = path.length - base_dir_length - 1
20
+ path[base_dir_length + 1, result_length]
21
+ else
22
+ path_name = Pathname.new(File.expand_path(path))
23
+ begin
24
+ path_name.relative_path_from(Pathname.new(base_dir)).to_s
25
+ rescue ArgumentError
26
+ path
27
+ end
28
+ end
23
29
  end
24
30
 
31
+ SMART_PATH_CACHE = {} # rubocop:disable Style/MutableConstant
32
+ private_constant :SMART_PATH_CACHE
33
+
25
34
  def smart_path(path)
26
- # Ideally, we calculate this relative to the project root.
27
- base_dir = Dir.pwd
35
+ SMART_PATH_CACHE[path] ||= begin
36
+ # Ideally, we calculate this relative to the project root.
37
+ base_dir = Dir.pwd
28
38
 
29
- if path.start_with? base_dir
30
- relative_path(path, base_dir)
31
- else
32
- path
39
+ if path.start_with? base_dir
40
+ relative_path(path, base_dir)
41
+ else
42
+ path
43
+ end
33
44
  end
34
45
  end
35
46
 
47
+ # rubocop:disable Metrics/MethodLength, Metrics/CyclomaticComplexity
36
48
  def match_path?(pattern, path)
37
49
  case pattern
38
50
  when String
39
- File.fnmatch?(pattern, path, File::FNM_PATHNAME | File::FNM_EXTGLOB) ||
40
- hidden_file_in_not_hidden_dir?(pattern, path)
51
+ matches =
52
+ if pattern == path
53
+ true
54
+ elsif glob?(pattern)
55
+ File.fnmatch?(pattern, path, File::FNM_PATHNAME | File::FNM_EXTGLOB)
56
+ end
57
+
58
+ matches || hidden_file_in_not_hidden_dir?(pattern, path)
41
59
  when Regexp
42
60
  begin
43
61
  pattern.match?(path)
@@ -48,12 +66,18 @@ module RuboCop
48
66
  end
49
67
  end
50
68
  end
69
+ # rubocop:enable Metrics/MethodLength, Metrics/CyclomaticComplexity
51
70
 
52
71
  # Returns true for an absolute Unix or Windows path.
53
72
  def absolute?(path)
54
73
  %r{\A([A-Z]:)?/}i.match?(path)
55
74
  end
56
75
 
76
+ # Returns true for a glob
77
+ def glob?(path)
78
+ path.match?(/[*{\[?]/)
79
+ end
80
+
57
81
  def hidden_file_in_not_hidden_dir?(pattern, path)
58
82
  hidden_file?(path) &&
59
83
  File.fnmatch?(
@@ -67,8 +91,12 @@ module RuboCop
67
91
  maybe_hidden_file?(path) && File.basename(path).start_with?('.')
68
92
  end
69
93
 
94
+ HIDDEN_FILE_PATTERN = "#{File::SEPARATOR}."
95
+
70
96
  # Loose check to reduce memory allocations
71
97
  def maybe_hidden_file?(path)
98
+ return false unless path.include?(HIDDEN_FILE_PATTERN)
99
+
72
100
  separator_index = path.rindex(File::SEPARATOR)
73
101
  return false unless separator_index
74
102
 
@@ -1,8 +1,8 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require 'digest/sha1'
4
- require 'find'
5
4
  require 'etc'
5
+ require 'find'
6
6
  require 'zlib'
7
7
  require_relative 'cache_config'
8
8
 
@@ -53,7 +53,7 @@ module RuboCop
53
53
  def remove_oldest_files(files, dirs, cache_root, verbose)
54
54
  # Add 1 to half the number of files, so that we remove the file if
55
55
  # there's only 1 left.
56
- remove_count = 1 + (files.length / 2)
56
+ remove_count = (files.length / 2) + 1
57
57
  puts "Removing the #{remove_count} oldest files from #{cache_root}" if verbose
58
58
  sorted = files.sort_by { |path| File.mtime(path) }
59
59
  remove_files(sorted, dirs, remove_count)
@@ -83,10 +83,10 @@ module RuboCop
83
83
  config_store.for_pwd.for_all_cops['AllowSymlinksInCacheRootDirectory']
84
84
  end
85
85
 
86
- attr :path
86
+ attr_reader :path
87
87
 
88
88
  def initialize(file, team, options, config_store, cache_root = nil)
89
- cache_root ||= options[:cache_root]
89
+ cache_root ||= File.join(options[:cache_root], 'rubocop_cache') if options[:cache_root]
90
90
  cache_root ||= ResultCache.cache_root(config_store)
91
91
  @allow_symlinks_in_cache_location =
92
92
  ResultCache.allow_symlinks_in_cache_location?(config_store)
@@ -202,6 +202,10 @@ module RuboCop
202
202
  lib_root = File.join(File.dirname(__FILE__), '..')
203
203
  exe_root = File.join(lib_root, '..', 'exe')
204
204
 
205
+ # Make sure to use an absolute path to prevent errors on Windows
206
+ # when traversing the relative paths with symlinks.
207
+ exe_root = File.absolute_path(exe_root)
208
+
205
209
  # These are all the files we have `require`d plus everything in the
206
210
  # exe directory. A change to any of them could affect the cop output
207
211
  # so we include them in the cache hash.
@@ -6,7 +6,7 @@ require 'tempfile'
6
6
  module CopHelper
7
7
  extend RSpec::SharedContext
8
8
 
9
- let(:ruby_version) { 2.6 }
9
+ let(:ruby_version) { RuboCop::TargetRuby::DEFAULT_VERSION }
10
10
  let(:rails_version) { false }
11
11
 
12
12
  def inspect_source(source, file = nil)
@@ -44,7 +44,10 @@ module CopHelper
44
44
 
45
45
  def registry
46
46
  @registry ||= begin
47
- cops = configuration.keys.map { |cop| RuboCop::Cop::Registry.global.find_by_cop_name(cop) }
47
+ keys = configuration.keys
48
+ cops =
49
+ keys.map { |directive| RuboCop::Cop::Registry.global.find_cops_by_directive(directive) }
50
+ .flatten
48
51
  cops << cop_class if defined?(cop_class) && !cops.include?(cop_class)
49
52
  cops.compact!
50
53
  RuboCop::Cop::Registry.new(cops)
@@ -66,7 +69,7 @@ module CopHelper
66
69
  end
67
70
 
68
71
  def _investigate(cop, processed_source)
69
- team = RuboCop::Cop::Team.new([cop], nil, raise_error: true)
72
+ team = RuboCop::Cop::Team.new([cop], configuration, raise_error: true)
70
73
  report = team.investigate(processed_source)
71
74
  @last_corrector = report.correctors.first || RuboCop::Cop::Corrector.new(processed_source)
72
75
  report.offenses.reject(&:disabled?)
@@ -6,7 +6,7 @@ module RuboCop
6
6
  #
7
7
  # This mixin makes it easier to specify strict offense expectations
8
8
  # in a declarative and visual fashion. Just type out the code that
9
- # should generate a offense, annotate code by writing '^'s
9
+ # should generate an offense, annotate code by writing '^'s
10
10
  # underneath each character that should be highlighted, and follow
11
11
  # the carets with a string (separated by a space) that is the
12
12
  # message of the offense. You can include multiple offenses in
@@ -126,7 +126,7 @@ module RuboCop
126
126
  @offenses
127
127
  end
128
128
 
129
- # rubocop:disable Metrics/AbcSize, Metrics/MethodLength, Metrics/CyclomaticComplexity
129
+ # rubocop:disable Metrics/AbcSize, Metrics/MethodLength, Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity
130
130
  def expect_correction(correction, loop: true, source: nil)
131
131
  if source
132
132
  expected_annotations = parse_annotations(source, raise_error: false)
@@ -138,6 +138,8 @@ module RuboCop
138
138
 
139
139
  source = @processed_source.raw_source
140
140
 
141
+ raise 'Use `expect_no_corrections` if the code will not change' if correction == source
142
+
141
143
  iteration = 0
142
144
  new_source = loop do
143
145
  iteration += 1
@@ -157,11 +159,11 @@ module RuboCop
157
159
  _investigate(cop, @processed_source)
158
160
  end
159
161
 
160
- raise 'Use `expect_no_corrections` if the code will not change' if new_source == source
162
+ raise 'Expected correction but no corrections were made' if new_source == source
161
163
 
162
164
  expect(new_source).to eq(correction)
163
165
  end
164
- # rubocop:enable Metrics/AbcSize, Metrics/MethodLength, Metrics/CyclomaticComplexity
166
+ # rubocop:enable Metrics/AbcSize, Metrics/MethodLength, Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity
165
167
 
166
168
  def expect_no_corrections
167
169
  raise '`expect_no_corrections` must follow `expect_offense`' unless @processed_source
@@ -11,6 +11,8 @@ RSpec.shared_context 'isolated environment' do # rubocop:disable Metrics/BlockLe
11
11
  # Make sure to expand all symlinks in the path first. Otherwise we may
12
12
  # get mismatched pathnames when loading config files later on.
13
13
  tmpdir = File.realpath(tmpdir)
14
+ # Make upwards search for .rubocop.yml files stop at this directory.
15
+ RuboCop::FileFinder.root_level = tmpdir
14
16
 
15
17
  virtual_home = File.expand_path(File.join(tmpdir, 'home'))
16
18
  Dir.mkdir(virtual_home)
@@ -21,9 +23,6 @@ RSpec.shared_context 'isolated environment' do # rubocop:disable Metrics/BlockLe
21
23
  root = example.metadata[:root]
22
24
  working_dir = root ? File.join(base_dir, 'work', root) : File.join(base_dir, 'work')
23
25
 
24
- # Make upwards search for .rubocop.yml files stop at this directory.
25
- RuboCop::FileFinder.root_level = working_dir
26
-
27
26
  begin
28
27
  FileUtils.mkdir_p(working_dir)
29
28
 
@@ -172,3 +171,7 @@ end
172
171
  RSpec.shared_context 'ruby 3.2' do
173
172
  let(:ruby_version) { 3.2 }
174
173
  end
174
+
175
+ RSpec.shared_context 'ruby 3.3' do
176
+ let(:ruby_version) { 3.3 }
177
+ end
@@ -3,10 +3,10 @@
3
3
  # Require this file to load code that supports testing using RSpec.
4
4
 
5
5
  require_relative 'cop_helper'
6
- require_relative 'host_environment_simulation_helper'
7
- require_relative 'shared_contexts'
8
6
  require_relative 'expect_offense'
7
+ require_relative 'host_environment_simulation_helper'
9
8
  require_relative 'parallel_formatter'
9
+ require_relative 'shared_contexts'
10
10
 
11
11
  RSpec.configure do |config|
12
12
  config.include CopHelper
@@ -25,4 +25,5 @@ RSpec.configure do |config|
25
25
  config.include_context 'ruby 3.0', :ruby30
26
26
  config.include_context 'ruby 3.1', :ruby31
27
27
  config.include_context 'ruby 3.2', :ruby32
28
+ config.include_context 'ruby 3.3', :ruby33
28
29
  end
@@ -24,9 +24,35 @@ module RuboCop
24
24
  end
25
25
  end
26
26
 
27
+ class << self
28
+ # @return [Array<#call>]
29
+ def ruby_extractors
30
+ @ruby_extractors ||= [default_ruby_extractor]
31
+ end
32
+
33
+ private
34
+
35
+ # @return [#call]
36
+ def default_ruby_extractor
37
+ lambda do |processed_source|
38
+ [
39
+ {
40
+ offset: 0,
41
+ processed_source: processed_source
42
+ }
43
+ ]
44
+ end
45
+ end
46
+ end
47
+
27
48
  # @api private
28
49
  MAX_ITERATIONS = 200
29
50
 
51
+ # @api private
52
+ REDUNDANT_COP_DISABLE_DIRECTIVE_RULES = %w[
53
+ Lint/RedundantCopDisableDirective RedundantCopDisableDirective Lint
54
+ ].freeze
55
+
30
56
  attr_reader :errors, :warnings
31
57
  attr_writer :aborting
32
58
 
@@ -194,7 +220,9 @@ module RuboCop
194
220
  end
195
221
 
196
222
  def check_for_redundant_disables?(source)
197
- !source.disabled_line_ranges.empty? && !filtered_run?
223
+ return false if source.disabled_line_ranges.empty? || except_redundant_cop_disable_directive?
224
+
225
+ !@options[:only]
198
226
  end
199
227
 
200
228
  def redundant_cop_disable_directive(file)
@@ -205,8 +233,8 @@ module RuboCop
205
233
  yield cop if cop.relevant_file?(file)
206
234
  end
207
235
 
208
- def filtered_run?
209
- @options[:except] || @options[:only]
236
+ def except_redundant_cop_disable_directive?
237
+ @options[:except] && (@options[:except] & REDUNDANT_COP_DISABLE_DIRECTIVE_RULES).any?
210
238
  end
211
239
 
212
240
  def file_started(file)
@@ -312,10 +340,25 @@ module RuboCop
312
340
  end
313
341
 
314
342
  def inspect_file(processed_source, team = mobilize_team(processed_source))
315
- report = team.investigate(processed_source)
316
- @errors.concat(team.errors)
317
- @warnings.concat(team.warnings)
318
- [report.offenses, team.updated_source_file?]
343
+ extracted_ruby_sources = extract_ruby_sources(processed_source)
344
+ offenses = extracted_ruby_sources.flat_map do |extracted_ruby_source|
345
+ report = team.investigate(
346
+ extracted_ruby_source[:processed_source],
347
+ offset: extracted_ruby_source[:offset],
348
+ original: processed_source
349
+ )
350
+ @errors.concat(team.errors)
351
+ @warnings.concat(team.warnings)
352
+ report.offenses
353
+ end
354
+ [offenses, team.updated_source_file?]
355
+ end
356
+
357
+ def extract_ruby_sources(processed_source)
358
+ self.class.ruby_extractors.find do |ruby_extractor|
359
+ result = ruby_extractor.call(processed_source)
360
+ break result if result
361
+ end
319
362
  end
320
363
 
321
364
  def mobilize_team(processed_source)
@@ -378,10 +421,10 @@ module RuboCop
378
421
  end
379
422
 
380
423
  def considered_failure?(offense)
381
- # For :autocorrect level, any offense - corrected or not - is a failure.
382
424
  return false if offense.disabled?
383
425
 
384
- return true if @options[:fail_level] == :autocorrect
426
+ # For :autocorrect level, any correctable offense is a failure, regardless of severity
427
+ return true if @options[:fail_level] == :autocorrect && offense.correctable?
385
428
 
386
429
  !offense.corrected? && offense.severity >= minimum_severity_to_fail
387
430
  end
@@ -418,7 +461,9 @@ module RuboCop
418
461
  @minimum_severity_to_fail ||= begin
419
462
  # Unless given explicitly as `fail_level`, `:info` severity offenses do not fail
420
463
  name = @options[:fail_level] || :refactor
421
- RuboCop::Cop::Severity.new(name)
464
+
465
+ # autocorrect is a fake level - use the default
466
+ RuboCop::Cop::Severity.new(name == :autocorrect ? :refactor : name)
422
467
  end
423
468
  end
424
469
 
@@ -1,5 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ require 'fileutils'
3
4
  require 'pathname'
4
5
  require_relative '../cache_config'
5
6
  require_relative '../config_finder'
@@ -57,18 +58,24 @@ module RuboCop
57
58
  File.expand_path(File.join(cache_root_dir, 'server'))
58
59
  end
59
60
 
61
+ # rubocop:disable Metrics/MethodLength
60
62
  def cache_root_dir_from_config
61
63
  CacheConfig.root_dir do
62
64
  # `RuboCop::ConfigStore` has heavy dependencies, this is a lightweight implementation
63
65
  # so that only the necessary `CacheRootDirectory` can be obtained.
64
- require 'yaml'
65
66
  config_path = ConfigFinder.find_config_path(Dir.pwd)
67
+ file_contents = File.read(config_path)
68
+
69
+ # Returns early if `CacheRootDirectory` is not used before requiring `erb` or `yaml`.
70
+ next unless file_contents.include?('CacheRootDirectory')
66
71
 
67
72
  require 'erb'
68
- file_contents = File.read(config_path)
69
73
  yaml_code = ERB.new(file_contents).result
70
74
 
71
- config_yaml = YAML.safe_load(yaml_code, permitted_classes: [Regexp, Symbol])
75
+ require 'yaml'
76
+ config_yaml = YAML.safe_load(
77
+ yaml_code, permitted_classes: [Regexp, Symbol], aliases: true
78
+ )
72
79
 
73
80
  # For compatibility with Ruby 3.0 or lower.
74
81
  if Gem::Version.new(Psych::VERSION) < Gem::Version.new('4.0.0')
@@ -78,6 +85,7 @@ module RuboCop
78
85
  config_yaml&.dig('AllCops', 'CacheRootDirectory')
79
86
  end
80
87
  end
88
+ # rubocop:enable Metrics/MethodLength
81
89
 
82
90
  def port_path
83
91
  dir.join('port')
@@ -109,7 +117,7 @@ module RuboCop
109
117
 
110
118
  def pid_running?
111
119
  Process.kill(0, pid_path.read.to_i) == 1
112
- rescue Errno::ESRCH, Errno::ENOENT
120
+ rescue Errno::ESRCH, Errno::ENOENT, Errno::EACCES
113
121
  false
114
122
  end
115
123
 
@@ -1,6 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'rainbow'
4
3
  require_relative '../arguments_env'
5
4
  require_relative '../arguments_file'
6
5
 
@@ -23,15 +22,21 @@ module RuboCop
23
22
  STATUS_ERROR = 2
24
23
 
25
24
  SERVER_OPTIONS = %w[
26
- --server --no-server --server-status --restart-server --start-server --stop-server
25
+ --server
26
+ --no-server
27
+ --server-status
28
+ --restart-server
29
+ --start-server
30
+ --stop-server
31
+ --no-detach
27
32
  ].freeze
28
33
  EXCLUSIVE_OPTIONS = (SERVER_OPTIONS - %w[--server --no-server]).freeze
34
+ NO_DETACH_OPTIONS = %w[--server --start-server --restart-server].freeze
29
35
 
30
36
  def initialize
31
37
  @exit = false
32
38
  end
33
39
 
34
- # rubocop:disable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/MethodLength
35
40
  def run(argv = ARGV)
36
41
  unless Server.support_server?
37
42
  return error('RuboCop server is not supported by this Ruby.') if use_server_option?(argv)
@@ -40,16 +45,34 @@ module RuboCop
40
45
  end
41
46
 
42
47
  Cache.cache_root_path = fetch_cache_root_path_from(argv)
43
- deleted_server_arguments = delete_server_argument_from(argv)
44
48
 
45
- if deleted_server_arguments.size >= 2
46
- return error("#{deleted_server_arguments.join(', ')} cannot be specified together.")
49
+ process_arguments(argv)
50
+ end
51
+
52
+ def exit?
53
+ @exit
54
+ end
55
+
56
+ private
57
+
58
+ # rubocop:disable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/MethodLength
59
+ def process_arguments(argv)
60
+ server_arguments = delete_server_argument_from(argv)
61
+
62
+ detach = !server_arguments.delete('--no-detach')
63
+
64
+ if server_arguments.size >= 2
65
+ return error("#{server_arguments.join(', ')} cannot be specified together.")
47
66
  end
48
67
 
49
- server_command = deleted_server_arguments.first
68
+ server_command = server_arguments.first
69
+
70
+ unless detach || NO_DETACH_OPTIONS.include?(server_command)
71
+ return error("#{server_command} cannot be combined with --no-detach.")
72
+ end
50
73
 
51
74
  if EXCLUSIVE_OPTIONS.include?(server_command) && argv.count > allowed_option_count
52
- return error("#{server_command} cannot be combined with other options.")
75
+ return error("#{server_command} cannot be combined with #{argv[0]}.")
53
76
  end
54
77
 
55
78
  if server_command.nil?
@@ -57,23 +80,17 @@ module RuboCop
57
80
  ArgumentsFile.read_as_arguments.delete('--server')
58
81
  end
59
82
 
60
- run_command(server_command)
83
+ run_command(server_command, detach: detach)
61
84
 
62
85
  STATUS_SUCCESS
63
86
  end
64
87
  # rubocop:enable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/MethodLength
65
88
 
66
- def exit?
67
- @exit
68
- end
69
-
70
- private
71
-
72
89
  # rubocop:disable Metrics/CyclomaticComplexity, Metrics/MethodLength:
73
- def run_command(server_command)
90
+ def run_command(server_command, detach:)
74
91
  case server_command
75
92
  when '--server'
76
- Server::ClientCommand::Start.new.run unless Server.running?
93
+ Server::ClientCommand::Start.new(detach: detach).run unless Server.running?
77
94
  when '--no-server'
78
95
  Server::ClientCommand::Stop.new.run if Server.running?
79
96
  when '--restart-server'
@@ -81,7 +98,7 @@ module RuboCop
81
98
  Server::ClientCommand::Restart.new.run
82
99
  when '--start-server'
83
100
  @exit = true
84
- Server::ClientCommand::Start.new.run
101
+ Server::ClientCommand::Start.new(detach: detach).run
85
102
  when '--stop-server'
86
103
  @exit = true
87
104
  Server::ClientCommand::Stop.new.run
@@ -118,6 +135,8 @@ module RuboCop
118
135
  end
119
136
 
120
137
  def error(message)
138
+ require 'rainbow'
139
+
121
140
  @exit = true
122
141
  warn Rainbow(message).red
123
142
 
@@ -18,10 +18,11 @@ module RuboCop
18
18
  def run
19
19
  ensure_server!
20
20
  Cache.status_path.delete if Cache.status_path.file?
21
+ read_stdin = ARGV.include?('-s') || ARGV.include?('--stdin')
21
22
  send_request(
22
23
  command: 'exec',
23
24
  args: ARGV.dup,
24
- body: $stdin.tty? ? '' : $stdin.read
25
+ body: read_stdin ? $stdin.read : ''
25
26
  )
26
27
  warn stderr unless stderr.empty?
27
28
  status
@@ -31,7 +32,7 @@ module RuboCop
31
32
 
32
33
  def ensure_server!
33
34
  if incompatible_version?
34
- puts 'RuboCop version incompatibility found, RuboCop server restarting...'
35
+ warn 'RuboCop version incompatibility found, RuboCop server restarting...'
35
36
  ClientCommand::Stop.new.run
36
37
  elsif check_running_server
37
38
  return
@@ -41,7 +42,7 @@ module RuboCop
41
42
  end
42
43
 
43
44
  def incompatible_version?
44
- RuboCop::Version::STRING != Cache.version_path.read
45
+ Cache.version_path.read != RuboCop::Version::STRING
45
46
  end
46
47
 
47
48
  def stderr
@@ -15,6 +15,11 @@ module RuboCop
15
15
  # This class is a client command to start server process.
16
16
  # @api private
17
17
  class Start < Base
18
+ def initialize(detach: true)
19
+ @detach = detach
20
+ super()
21
+ end
22
+
18
23
  def run
19
24
  if Server.running?
20
25
  warn "RuboCop server (#{Cache.pid_path.read}) is already running."
@@ -34,7 +39,7 @@ module RuboCop
34
39
  host = ENV.fetch('RUBOCOP_SERVER_HOST', '127.0.0.1')
35
40
  port = ENV.fetch('RUBOCOP_SERVER_PORT', 0)
36
41
 
37
- Server::Core.new.start(host, port)
42
+ Server::Core.new.start(host, port, detach: @detach)
38
43
  end
39
44
  end
40
45
  end
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'socket'
4
3
  require 'securerandom'
4
+ require 'socket'
5
5
 
6
6
  #
7
7
  # This code is based on https://github.com/fohte/rubocop-daemon.
@@ -27,31 +27,46 @@ module RuboCop
27
27
  self.class.token
28
28
  end
29
29
 
30
- def start(host, port)
30
+ def start(host, port, detach: true)
31
31
  $PROGRAM_NAME = "rubocop --server #{Cache.project_dir}"
32
32
 
33
- require 'rubocop'
33
+ require_relative '../../rubocop'
34
34
  start_server(host, port)
35
35
 
36
- demonize if server_mode?
36
+ return unless server_mode?
37
+
38
+ detach ? detach_server : run_server
37
39
  end
38
40
 
39
41
  private
40
42
 
41
- def demonize
42
- Cache.write_port_and_token_files(port: @server.addr[1], token: token)
43
+ def detach_server
44
+ write_port_and_token_files
43
45
 
44
46
  pid = fork do
45
47
  Process.daemon(true)
46
48
  $stderr.reopen(Cache.stderr_path, 'w')
47
- Cache.write_pid_file do
48
- read_socket(@server.accept) until @server.closed?
49
- end
49
+ process_input
50
50
  end
51
51
 
52
52
  Process.waitpid(pid)
53
53
  end
54
54
 
55
+ def write_port_and_token_files
56
+ Cache.write_port_and_token_files(port: @server.addr[1], token: token)
57
+ end
58
+
59
+ def process_input
60
+ Cache.write_pid_file do
61
+ read_socket(@server.accept) until @server.closed?
62
+ end
63
+ end
64
+
65
+ def run_server
66
+ write_port_and_token_files
67
+ process_input
68
+ end
69
+
55
70
  def server_mode?
56
71
  true
57
72
  end
@@ -85,7 +100,7 @@ module RuboCop
85
100
 
86
101
  def use_json_format?
87
102
  return true if ARGV.include?('--format=json') || ARGV.include?('--format=j')
88
- return false unless (index = ARGV.index('--format'))
103
+ return false unless (index = ARGV.index('--format') || ARGV.index('-f'))
89
104
 
90
105
  format = ARGV[index + 1]
91
106
 
@@ -11,7 +11,7 @@
11
11
  #
12
12
  module RuboCop
13
13
  module Server
14
- # This module has a helper memthod for `RuboCop::Server::SocketReader`.
14
+ # This module has a helper method for `RuboCop::Server::SocketReader`.
15
15
  # @api private
16
16
  module Helper
17
17
  def self.redirect(stdin: $stdin, stdout: $stdout, stderr: $stderr, &_block)