rubocop 0.89.1 → 0.93.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 (365) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +2 -2
  3. data/config/default.yml +160 -14
  4. data/lib/rubocop.rb +33 -5
  5. data/lib/rubocop/cached_data.rb +3 -1
  6. data/lib/rubocop/cli/command.rb +1 -0
  7. data/lib/rubocop/cli/command/auto_genenerate_config.rb +1 -0
  8. data/lib/rubocop/cli/command/base.rb +1 -0
  9. data/lib/rubocop/cli/command/execute_runner.rb +9 -0
  10. data/lib/rubocop/cli/command/init_dotfile.rb +1 -0
  11. data/lib/rubocop/cli/command/show_cops.rb +1 -0
  12. data/lib/rubocop/cli/command/version.rb +1 -0
  13. data/lib/rubocop/cli/environment.rb +1 -0
  14. data/lib/rubocop/comment_config.rb +14 -5
  15. data/lib/rubocop/config_loader.rb +20 -9
  16. data/lib/rubocop/config_loader_resolver.rb +1 -0
  17. data/lib/rubocop/config_obsoletion.rb +1 -0
  18. data/lib/rubocop/config_regeneration.rb +33 -0
  19. data/lib/rubocop/config_store.rb +3 -3
  20. data/lib/rubocop/config_validator.rb +3 -0
  21. data/lib/rubocop/cop/base.rb +23 -0
  22. data/lib/rubocop/cop/bundler/duplicated_gem.rb +5 -1
  23. data/lib/rubocop/cop/bundler/gem_comment.rb +8 -3
  24. data/lib/rubocop/cop/bundler/insecure_protocol_source.rb +2 -0
  25. data/lib/rubocop/cop/commissioner.rb +47 -7
  26. data/lib/rubocop/cop/correctors/alignment_corrector.rb +4 -4
  27. data/lib/rubocop/cop/correctors/condition_corrector.rb +3 -5
  28. data/lib/rubocop/cop/correctors/empty_line_corrector.rb +9 -10
  29. data/lib/rubocop/cop/correctors/line_break_corrector.rb +4 -4
  30. data/lib/rubocop/cop/correctors/multiline_literal_brace_corrector.rb +8 -3
  31. data/lib/rubocop/cop/correctors/parentheses_corrector.rb +5 -8
  32. data/lib/rubocop/cop/correctors/percent_literal_corrector.rb +4 -9
  33. data/lib/rubocop/cop/correctors/punctuation_corrector.rb +8 -10
  34. data/lib/rubocop/cop/documentation.rb +22 -0
  35. data/lib/rubocop/cop/gemspec/required_ruby_version.rb +10 -10
  36. data/lib/rubocop/cop/generator.rb +2 -1
  37. data/lib/rubocop/cop/internal_affairs/method_name_equal.rb +1 -0
  38. data/lib/rubocop/cop/internal_affairs/node_type_predicate.rb +1 -0
  39. data/lib/rubocop/cop/internal_affairs/offense_location_keyword.rb +1 -0
  40. data/lib/rubocop/cop/internal_affairs/redundant_location_argument.rb +1 -0
  41. data/lib/rubocop/cop/internal_affairs/redundant_message_argument.rb +1 -0
  42. data/lib/rubocop/cop/layout/array_alignment.rb +1 -0
  43. data/lib/rubocop/cop/layout/begin_end_alignment.rb +77 -0
  44. data/lib/rubocop/cop/layout/block_alignment.rb +23 -19
  45. data/lib/rubocop/cop/layout/case_indentation.rb +4 -7
  46. data/lib/rubocop/cop/layout/class_structure.rb +11 -10
  47. data/lib/rubocop/cop/layout/closing_heredoc_indentation.rb +4 -6
  48. data/lib/rubocop/cop/layout/condition_position.rb +13 -15
  49. data/lib/rubocop/cop/layout/def_end_alignment.rb +8 -5
  50. data/lib/rubocop/cop/layout/dot_position.rb +21 -20
  51. data/lib/rubocop/cop/layout/empty_comment.rb +30 -23
  52. data/lib/rubocop/cop/layout/empty_line_after_guard_clause.rb +21 -18
  53. data/lib/rubocop/cop/layout/empty_line_after_magic_comment.rb +13 -13
  54. data/lib/rubocop/cop/layout/empty_line_after_multiline_condition.rb +136 -0
  55. data/lib/rubocop/cop/layout/empty_line_between_defs.rb +21 -23
  56. data/lib/rubocop/cop/layout/empty_lines.rb +6 -7
  57. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +17 -14
  58. data/lib/rubocop/cop/layout/empty_lines_around_arguments.rb +7 -8
  59. data/lib/rubocop/cop/layout/empty_lines_around_attribute_accessor.rb +5 -8
  60. data/lib/rubocop/cop/layout/empty_lines_around_begin_body.rb +2 -5
  61. data/lib/rubocop/cop/layout/empty_lines_around_block_body.rb +2 -5
  62. data/lib/rubocop/cop/layout/empty_lines_around_class_body.rb +2 -5
  63. data/lib/rubocop/cop/layout/empty_lines_around_exception_handling_keywords.rb +3 -7
  64. data/lib/rubocop/cop/layout/empty_lines_around_method_body.rb +2 -5
  65. data/lib/rubocop/cop/layout/empty_lines_around_module_body.rb +2 -5
  66. data/lib/rubocop/cop/layout/end_alignment.rb +11 -17
  67. data/lib/rubocop/cop/layout/first_array_element_line_break.rb +2 -5
  68. data/lib/rubocop/cop/layout/first_hash_element_line_break.rb +2 -5
  69. data/lib/rubocop/cop/layout/first_method_argument_line_break.rb +4 -8
  70. data/lib/rubocop/cop/layout/first_method_parameter_line_break.rb +2 -5
  71. data/lib/rubocop/cop/layout/hash_alignment.rb +17 -20
  72. data/lib/rubocop/cop/layout/heredoc_argument_closing_parenthesis.rb +15 -14
  73. data/lib/rubocop/cop/layout/heredoc_indentation.rb +14 -11
  74. data/lib/rubocop/cop/layout/initial_indentation.rb +6 -7
  75. data/lib/rubocop/cop/layout/leading_comment_space.rb +11 -9
  76. data/lib/rubocop/cop/layout/leading_empty_lines.rb +6 -11
  77. data/lib/rubocop/cop/layout/multiline_array_brace_layout.rb +2 -5
  78. data/lib/rubocop/cop/layout/multiline_array_line_breaks.rb +2 -5
  79. data/lib/rubocop/cop/layout/multiline_assignment_layout.rb +10 -14
  80. data/lib/rubocop/cop/layout/multiline_block_layout.rb +21 -19
  81. data/lib/rubocop/cop/layout/multiline_hash_brace_layout.rb +2 -5
  82. data/lib/rubocop/cop/layout/multiline_hash_key_line_breaks.rb +2 -5
  83. data/lib/rubocop/cop/layout/multiline_method_argument_line_breaks.rb +5 -9
  84. data/lib/rubocop/cop/layout/multiline_method_call_brace_layout.rb +2 -5
  85. data/lib/rubocop/cop/layout/multiline_method_definition_brace_layout.rb +2 -5
  86. data/lib/rubocop/cop/layout/rescue_ensure_alignment.rb +41 -22
  87. data/lib/rubocop/cop/layout/space_after_colon.rb +11 -7
  88. data/lib/rubocop/cop/layout/space_after_comma.rb +2 -5
  89. data/lib/rubocop/cop/layout/space_after_method_name.rb +5 -6
  90. data/lib/rubocop/cop/layout/space_after_not.rb +9 -11
  91. data/lib/rubocop/cop/layout/space_after_semicolon.rb +2 -5
  92. data/lib/rubocop/cop/layout/space_around_equals_in_parameter_default.rb +17 -21
  93. data/lib/rubocop/cop/layout/space_around_keyword.rb +17 -18
  94. data/lib/rubocop/cop/layout/space_around_operators.rb +17 -16
  95. data/lib/rubocop/cop/layout/space_before_block_braces.rb +23 -22
  96. data/lib/rubocop/cop/layout/space_before_comma.rb +3 -5
  97. data/lib/rubocop/cop/layout/space_before_comment.rb +10 -7
  98. data/lib/rubocop/cop/layout/space_before_first_arg.rb +7 -7
  99. data/lib/rubocop/cop/layout/space_before_semicolon.rb +2 -5
  100. data/lib/rubocop/cop/layout/space_in_lambda_literal.rb +9 -17
  101. data/lib/rubocop/cop/layout/space_inside_array_literal_brackets.rb +20 -23
  102. data/lib/rubocop/cop/layout/space_inside_array_percent_literal.rb +3 -8
  103. data/lib/rubocop/cop/layout/space_inside_block_braces.rb +13 -16
  104. data/lib/rubocop/cop/layout/space_inside_hash_literal_braces.rb +19 -37
  105. data/lib/rubocop/cop/layout/space_inside_parens.rb +9 -14
  106. data/lib/rubocop/cop/layout/space_inside_percent_literal_delimiters.rb +5 -10
  107. data/lib/rubocop/cop/layout/space_inside_range_literal.rb +8 -17
  108. data/lib/rubocop/cop/layout/space_inside_reference_brackets.rb +16 -24
  109. data/lib/rubocop/cop/layout/space_inside_string_interpolation.rb +12 -13
  110. data/lib/rubocop/cop/layout/trailing_empty_lines.rb +10 -15
  111. data/lib/rubocop/cop/layout/trailing_whitespace.rb +11 -11
  112. data/lib/rubocop/cop/lint/ambiguous_block_association.rb +2 -0
  113. data/lib/rubocop/cop/lint/ambiguous_operator.rb +2 -0
  114. data/lib/rubocop/cop/lint/ambiguous_regexp_literal.rb +15 -1
  115. data/lib/rubocop/cop/lint/big_decimal_new.rb +1 -2
  116. data/lib/rubocop/cop/lint/binary_operator_with_identical_operands.rb +1 -1
  117. data/lib/rubocop/cop/lint/boolean_symbol.rb +3 -0
  118. data/lib/rubocop/cop/lint/constant_definition_in_block.rb +74 -0
  119. data/lib/rubocop/cop/lint/constant_resolution.rb +1 -1
  120. data/lib/rubocop/cop/lint/debugger.rb +2 -3
  121. data/lib/rubocop/cop/lint/deprecated_class_methods.rb +1 -3
  122. data/lib/rubocop/cop/lint/duplicate_methods.rb +2 -4
  123. data/lib/rubocop/cop/lint/duplicate_require.rb +46 -0
  124. data/lib/rubocop/cop/lint/duplicate_rescue_exception.rb +2 -15
  125. data/lib/rubocop/cop/lint/each_with_object_argument.rb +1 -0
  126. data/lib/rubocop/cop/lint/empty_file.rb +50 -0
  127. data/lib/rubocop/cop/lint/erb_new_arguments.rb +2 -0
  128. data/lib/rubocop/cop/lint/float_comparison.rb +2 -2
  129. data/lib/rubocop/cop/lint/format_parameter_mismatch.rb +2 -2
  130. data/lib/rubocop/cop/lint/hash_compare_by_identity.rb +37 -0
  131. data/lib/rubocop/cop/lint/identity_comparison.rb +51 -0
  132. data/lib/rubocop/cop/lint/ineffective_access_modifier.rb +2 -5
  133. data/lib/rubocop/cop/lint/inherit_exception.rb +2 -2
  134. data/lib/rubocop/cop/lint/missing_super.rb +2 -2
  135. data/lib/rubocop/cop/lint/mixed_regexp_capture_types.rb +3 -35
  136. data/lib/rubocop/cop/lint/multiple_comparison.rb +3 -1
  137. data/lib/rubocop/cop/lint/number_conversion.rb +1 -0
  138. data/lib/rubocop/cop/lint/out_of_range_regexp_ref.rb +9 -20
  139. data/lib/rubocop/cop/lint/parentheses_as_grouped_expression.rb +1 -1
  140. data/lib/rubocop/cop/lint/percent_string_array.rb +8 -12
  141. data/lib/rubocop/cop/lint/raise_exception.rb +1 -0
  142. data/lib/rubocop/cop/lint/rand_one.rb +2 -1
  143. data/lib/rubocop/cop/lint/redundant_cop_disable_directive.rb +22 -12
  144. data/lib/rubocop/cop/lint/redundant_cop_enable_directive.rb +14 -4
  145. data/lib/rubocop/cop/lint/redundant_require_statement.rb +1 -0
  146. data/lib/rubocop/cop/lint/redundant_safe_navigation.rb +45 -0
  147. data/lib/rubocop/cop/lint/rescue_type.rb +0 -1
  148. data/lib/rubocop/cop/lint/send_with_mixin_argument.rb +3 -1
  149. data/lib/rubocop/cop/lint/shadowed_exception.rb +6 -6
  150. data/lib/rubocop/cop/lint/shadowing_outer_local_variable.rb +4 -4
  151. data/lib/rubocop/cop/lint/struct_new_override.rb +1 -0
  152. data/lib/rubocop/cop/lint/to_json.rb +16 -5
  153. data/lib/rubocop/cop/lint/top_level_return_with_argument.rb +1 -1
  154. data/lib/rubocop/cop/lint/trailing_comma_in_attribute_declaration.rb +57 -0
  155. data/lib/rubocop/cop/lint/unreachable_loop.rb +3 -6
  156. data/lib/rubocop/cop/lint/uri_escape_unescape.rb +3 -1
  157. data/lib/rubocop/cop/lint/uri_regexp.rb +2 -1
  158. data/lib/rubocop/cop/lint/useless_access_modifier.rb +3 -9
  159. data/lib/rubocop/cop/lint/useless_method_definition.rb +70 -0
  160. data/lib/rubocop/cop/lint/useless_times.rb +106 -0
  161. data/lib/rubocop/cop/metrics/block_length.rb +3 -1
  162. data/lib/rubocop/cop/metrics/class_length.rb +8 -6
  163. data/lib/rubocop/cop/metrics/utils/abc_size_calculator.rb +27 -16
  164. data/lib/rubocop/cop/metrics/utils/code_length_calculator.rb +1 -0
  165. data/lib/rubocop/cop/mixin/alignment.rb +3 -0
  166. data/lib/rubocop/cop/mixin/allowed_methods.rb +2 -0
  167. data/lib/rubocop/cop/mixin/annotation_comment.rb +5 -0
  168. data/lib/rubocop/cop/mixin/check_line_breakable.rb +16 -7
  169. data/lib/rubocop/cop/mixin/comments_help.rb +48 -0
  170. data/lib/rubocop/cop/mixin/configurable_naming.rb +2 -2
  171. data/lib/rubocop/cop/mixin/configurable_numbering.rb +3 -3
  172. data/lib/rubocop/cop/mixin/empty_lines_around_body.rb +8 -7
  173. data/lib/rubocop/cop/mixin/empty_parameter.rb +3 -1
  174. data/lib/rubocop/cop/mixin/end_keyword_alignment.rb +12 -1
  175. data/lib/rubocop/cop/mixin/first_element_line_break.rb +3 -1
  176. data/lib/rubocop/cop/mixin/hash_transform_method.rb +27 -2
  177. data/lib/rubocop/cop/mixin/multiline_element_line_breaks.rb +3 -1
  178. data/lib/rubocop/cop/mixin/multiline_literal_brace_layout.rb +12 -10
  179. data/lib/rubocop/cop/mixin/negative_conditional.rb +2 -2
  180. data/lib/rubocop/cop/mixin/percent_array.rb +14 -3
  181. data/lib/rubocop/cop/mixin/rescue_node.rb +11 -1
  182. data/lib/rubocop/cop/mixin/space_after_punctuation.rb +4 -3
  183. data/lib/rubocop/cop/mixin/space_before_punctuation.rb +4 -3
  184. data/lib/rubocop/cop/mixin/statement_modifier.rb +9 -3
  185. data/lib/rubocop/cop/mixin/surrounding_space.rb +8 -4
  186. data/lib/rubocop/cop/mixin/trailing_comma.rb +7 -7
  187. data/lib/rubocop/cop/mixin/visibility_help.rb +4 -16
  188. data/lib/rubocop/cop/naming/binary_operator_parameter_name.rb +1 -1
  189. data/lib/rubocop/cop/naming/file_name.rb +1 -1
  190. data/lib/rubocop/cop/offense.rb +16 -2
  191. data/lib/rubocop/cop/security/eval.rb +1 -0
  192. data/lib/rubocop/cop/security/json_load.rb +1 -0
  193. data/lib/rubocop/cop/security/marshal_load.rb +1 -0
  194. data/lib/rubocop/cop/security/open.rb +1 -0
  195. data/lib/rubocop/cop/security/yaml_load.rb +1 -0
  196. data/lib/rubocop/cop/severity.rb +0 -8
  197. data/lib/rubocop/cop/style/access_modifier_declarations.rb +7 -11
  198. data/lib/rubocop/cop/style/accessor_grouping.rb +3 -0
  199. data/lib/rubocop/cop/style/alias.rb +2 -0
  200. data/lib/rubocop/cop/style/array_coercion.rb +4 -0
  201. data/lib/rubocop/cop/style/array_join.rb +1 -0
  202. data/lib/rubocop/cop/style/attr.rb +1 -0
  203. data/lib/rubocop/cop/style/auto_resource_cleanup.rb +2 -0
  204. data/lib/rubocop/cop/style/case_equality.rb +11 -3
  205. data/lib/rubocop/cop/style/case_like_if.rb +40 -8
  206. data/lib/rubocop/cop/style/class_and_module_children.rb +2 -0
  207. data/lib/rubocop/cop/style/class_check.rb +6 -9
  208. data/lib/rubocop/cop/style/class_equality_comparison.rb +49 -0
  209. data/lib/rubocop/cop/style/class_methods_definitions.rb +157 -0
  210. data/lib/rubocop/cop/style/class_vars.rb +1 -2
  211. data/lib/rubocop/cop/style/combinable_loops.rb +91 -0
  212. data/lib/rubocop/cop/style/comment_annotation.rb +6 -0
  213. data/lib/rubocop/cop/style/commented_keyword.rb +7 -8
  214. data/lib/rubocop/cop/style/conditional_assignment.rb +49 -60
  215. data/lib/rubocop/cop/style/date_time.rb +12 -1
  216. data/lib/rubocop/cop/style/dir.rb +1 -0
  217. data/lib/rubocop/cop/style/double_negation.rb +1 -0
  218. data/lib/rubocop/cop/style/empty_block_parameter.rb +9 -10
  219. data/lib/rubocop/cop/style/empty_lambda_parameter.rb +9 -10
  220. data/lib/rubocop/cop/style/empty_literal.rb +3 -1
  221. data/lib/rubocop/cop/style/eval_with_location.rb +1 -3
  222. data/lib/rubocop/cop/style/even_odd.rb +1 -0
  223. data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -2
  224. data/lib/rubocop/cop/style/explicit_block_argument.rb +7 -3
  225. data/lib/rubocop/cop/style/float_division.rb +2 -0
  226. data/lib/rubocop/cop/style/for.rb +0 -4
  227. data/lib/rubocop/cop/style/format_string.rb +1 -4
  228. data/lib/rubocop/cop/style/format_string_token.rb +1 -1
  229. data/lib/rubocop/cop/style/guard_clause.rb +1 -0
  230. data/lib/rubocop/cop/style/hash_as_last_array_item.rb +24 -5
  231. data/lib/rubocop/cop/style/hash_syntax.rb +6 -5
  232. data/lib/rubocop/cop/style/hash_transform_keys.rb +16 -9
  233. data/lib/rubocop/cop/style/hash_transform_values.rb +16 -9
  234. data/lib/rubocop/cop/style/if_unless_modifier.rb +2 -6
  235. data/lib/rubocop/cop/style/implicit_runtime_error.rb +1 -0
  236. data/lib/rubocop/cop/style/keyword_parameters_order.rb +53 -0
  237. data/lib/rubocop/cop/style/lambda_call.rb +3 -1
  238. data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +10 -1
  239. data/lib/rubocop/cop/style/method_def_parentheses.rb +0 -4
  240. data/lib/rubocop/cop/style/mixin_usage.rb +8 -27
  241. data/lib/rubocop/cop/style/multiline_block_chain.rb +2 -2
  242. data/lib/rubocop/cop/style/multiline_ternary_operator.rb +14 -1
  243. data/lib/rubocop/cop/style/multiline_when_then.rb +3 -2
  244. data/lib/rubocop/cop/style/negated_if.rb +6 -6
  245. data/lib/rubocop/cop/style/negated_unless.rb +6 -6
  246. data/lib/rubocop/cop/style/negated_while.rb +7 -15
  247. data/lib/rubocop/cop/style/nested_modifier.rb +10 -13
  248. data/lib/rubocop/cop/style/nested_parenthesized_calls.rb +11 -11
  249. data/lib/rubocop/cop/style/nested_ternary_operator.rb +16 -16
  250. data/lib/rubocop/cop/style/next.rb +10 -14
  251. data/lib/rubocop/cop/style/nil_comparison.rb +13 -11
  252. data/lib/rubocop/cop/style/non_nil_check.rb +34 -26
  253. data/lib/rubocop/cop/style/not.rb +20 -26
  254. data/lib/rubocop/cop/style/numeric_literal_prefix.rb +4 -9
  255. data/lib/rubocop/cop/style/numeric_predicate.rb +5 -14
  256. data/lib/rubocop/cop/style/one_line_conditional.rb +73 -23
  257. data/lib/rubocop/cop/style/option_hash.rb +1 -1
  258. data/lib/rubocop/cop/style/optional_arguments.rb +1 -1
  259. data/lib/rubocop/cop/style/optional_boolean_parameter.rb +12 -1
  260. data/lib/rubocop/cop/style/or_assignment.rb +13 -10
  261. data/lib/rubocop/cop/style/parallel_assignment.rb +14 -14
  262. data/lib/rubocop/cop/style/parentheses_around_condition.rb +6 -6
  263. data/lib/rubocop/cop/style/percent_literal_delimiters.rb +13 -19
  264. data/lib/rubocop/cop/style/percent_q_literals.rb +8 -10
  265. data/lib/rubocop/cop/style/perl_backrefs.rb +8 -10
  266. data/lib/rubocop/cop/style/preferred_hash_methods.rb +11 -14
  267. data/lib/rubocop/cop/style/proc.rb +6 -6
  268. data/lib/rubocop/cop/style/raise_args.rb +12 -24
  269. data/lib/rubocop/cop/style/random_with_offset.rb +19 -19
  270. data/lib/rubocop/cop/style/redundant_assignment.rb +8 -18
  271. data/lib/rubocop/cop/style/redundant_begin.rb +28 -12
  272. data/lib/rubocop/cop/style/redundant_capital_w.rb +6 -9
  273. data/lib/rubocop/cop/style/redundant_condition.rb +10 -7
  274. data/lib/rubocop/cop/style/redundant_conditional.rb +4 -5
  275. data/lib/rubocop/cop/style/redundant_exception.rb +1 -3
  276. data/lib/rubocop/cop/style/redundant_fetch_block.rb +3 -12
  277. data/lib/rubocop/cop/style/redundant_file_extension_in_require.rb +9 -8
  278. data/lib/rubocop/cop/style/redundant_freeze.rb +5 -7
  279. data/lib/rubocop/cop/style/redundant_interpolation.rb +31 -25
  280. data/lib/rubocop/cop/style/redundant_parentheses.rb +21 -15
  281. data/lib/rubocop/cop/style/redundant_percent_q.rb +9 -11
  282. data/lib/rubocop/cop/style/redundant_regexp_character_class.rb +44 -36
  283. data/lib/rubocop/cop/style/redundant_regexp_escape.rb +13 -29
  284. data/lib/rubocop/cop/style/redundant_return.rb +17 -17
  285. data/lib/rubocop/cop/style/redundant_self.rb +9 -11
  286. data/lib/rubocop/cop/style/redundant_self_assignment.rb +116 -0
  287. data/lib/rubocop/cop/style/redundant_sort.rb +12 -29
  288. data/lib/rubocop/cop/style/redundant_sort_by.rb +5 -9
  289. data/lib/rubocop/cop/style/regexp_literal.rb +10 -21
  290. data/lib/rubocop/cop/style/rescue_modifier.rb +29 -9
  291. data/lib/rubocop/cop/style/rescue_standard_error.rb +20 -16
  292. data/lib/rubocop/cop/style/return_nil.rb +5 -5
  293. data/lib/rubocop/cop/style/safe_navigation.rb +18 -12
  294. data/lib/rubocop/cop/style/sample.rb +12 -14
  295. data/lib/rubocop/cop/style/self_assignment.rb +26 -22
  296. data/lib/rubocop/cop/style/semicolon.rb +6 -9
  297. data/lib/rubocop/cop/style/send.rb +4 -5
  298. data/lib/rubocop/cop/style/signal_exception.rb +23 -19
  299. data/lib/rubocop/cop/style/single_argument_dig.rb +1 -0
  300. data/lib/rubocop/cop/style/single_line_block_params.rb +4 -2
  301. data/lib/rubocop/cop/style/single_line_methods.rb +17 -16
  302. data/lib/rubocop/cop/style/slicing_with_range.rb +6 -8
  303. data/lib/rubocop/cop/style/sole_nested_conditional.rb +66 -0
  304. data/lib/rubocop/cop/style/special_global_vars.rb +10 -15
  305. data/lib/rubocop/cop/style/stabby_lambda_parentheses.rb +17 -21
  306. data/lib/rubocop/cop/style/stderr_puts.rb +5 -6
  307. data/lib/rubocop/cop/style/string_concatenation.rb +17 -3
  308. data/lib/rubocop/cop/style/string_hash_keys.rb +6 -7
  309. data/lib/rubocop/cop/style/string_methods.rb +7 -17
  310. data/lib/rubocop/cop/style/strip.rb +9 -14
  311. data/lib/rubocop/cop/style/struct_inheritance.rb +3 -6
  312. data/lib/rubocop/cop/style/symbol_array.rb +5 -16
  313. data/lib/rubocop/cop/style/symbol_literal.rb +4 -6
  314. data/lib/rubocop/cop/style/symbol_proc.rb +14 -18
  315. data/lib/rubocop/cop/style/ternary_parentheses.rb +22 -22
  316. data/lib/rubocop/cop/style/trailing_body_on_class.rb +3 -6
  317. data/lib/rubocop/cop/style/trailing_body_on_method_definition.rb +4 -7
  318. data/lib/rubocop/cop/style/trailing_body_on_module.rb +3 -6
  319. data/lib/rubocop/cop/style/trailing_comma_in_arguments.rb +2 -5
  320. data/lib/rubocop/cop/style/trailing_comma_in_array_literal.rb +2 -5
  321. data/lib/rubocop/cop/style/trailing_comma_in_block_args.rb +11 -9
  322. data/lib/rubocop/cop/style/trailing_comma_in_hash_literal.rb +2 -5
  323. data/lib/rubocop/cop/style/trailing_underscore_variable.rb +8 -17
  324. data/lib/rubocop/cop/style/trivial_accessors.rb +26 -30
  325. data/lib/rubocop/cop/style/unless_else.rb +5 -8
  326. data/lib/rubocop/cop/style/unpack_first.rb +5 -8
  327. data/lib/rubocop/cop/style/variable_interpolation.rb +7 -10
  328. data/lib/rubocop/cop/style/when_then.rb +4 -6
  329. data/lib/rubocop/cop/style/while_until_do.rb +6 -16
  330. data/lib/rubocop/cop/style/while_until_modifier.rb +6 -20
  331. data/lib/rubocop/cop/style/word_array.rb +5 -23
  332. data/lib/rubocop/cop/style/yoda_condition.rb +4 -15
  333. data/lib/rubocop/cop/style/zero_length_predicate.rb +12 -18
  334. data/lib/rubocop/cop/team.rb +1 -0
  335. data/lib/rubocop/cop/util.rb +1 -2
  336. data/lib/rubocop/cop/utils/format_string.rb +3 -5
  337. data/lib/rubocop/cop/variable_force.rb +2 -0
  338. data/lib/rubocop/cop/variable_force/branch.rb +0 -4
  339. data/lib/rubocop/cops_documentation_generator.rb +4 -2
  340. data/lib/rubocop/core_ext/string.rb +2 -2
  341. data/lib/rubocop/directive_comment.rb +32 -0
  342. data/lib/rubocop/ext/regexp_node.rb +62 -0
  343. data/lib/rubocop/file_finder.rb +1 -0
  344. data/lib/rubocop/formatter/auto_gen_config_formatter.rb +2 -1
  345. data/lib/rubocop/formatter/disabled_config_formatter.rb +12 -5
  346. data/lib/rubocop/formatter/html_formatter.rb +2 -0
  347. data/lib/rubocop/formatter/progress_formatter.rb +2 -1
  348. data/lib/rubocop/formatter/quiet_formatter.rb +1 -1
  349. data/lib/rubocop/formatter/simple_text_formatter.rb +36 -6
  350. data/lib/rubocop/name_similarity.rb +1 -0
  351. data/lib/rubocop/options.rb +40 -17
  352. data/lib/rubocop/remote_config.rb +1 -0
  353. data/lib/rubocop/result_cache.rb +39 -15
  354. data/lib/rubocop/rspec/cop_helper.rb +5 -2
  355. data/lib/rubocop/rspec/expect_offense.rb +14 -9
  356. data/lib/rubocop/rspec/shared_contexts.rb +12 -0
  357. data/lib/rubocop/runner.rb +38 -18
  358. data/lib/rubocop/string_interpreter.rb +3 -0
  359. data/lib/rubocop/target_finder.rb +28 -26
  360. data/lib/rubocop/target_ruby.rb +7 -1
  361. data/lib/rubocop/version.rb +7 -1
  362. data/lib/rubocop/yaml_duplication_checker.rb +1 -0
  363. metadata +31 -17
  364. data/lib/rubocop/cop/mixin/regexp_literal_help.rb +0 -43
  365. data/lib/rubocop/cop/tokens_util.rb +0 -84
@@ -90,9 +90,11 @@ module RuboCop
90
90
  end
91
91
 
92
92
  # Make Kernel#binding public.
93
+ # rubocop:disable Lint/UselessMethodDefinition
93
94
  def binding
94
95
  super
95
96
  end
97
+ # rubocop:enable Lint/UselessMethodDefinition
96
98
 
97
99
  def decorated_message(offense)
98
100
  offense.message.gsub(/`(.+?)`/) do
@@ -45,7 +45,8 @@ module RuboCop
45
45
 
46
46
  report_summary(inspected_files.size,
47
47
  @total_offense_count,
48
- @total_correction_count)
48
+ @total_correction_count,
49
+ @total_correctable_count)
49
50
  end
50
51
 
51
52
  def report_file_as_mark(offenses)
@@ -5,7 +5,7 @@ module RuboCop
5
5
  # If no offenses are found, no output is displayed.
6
6
  # Otherwise, SimpleTextFormatter's output is displayed.
7
7
  class QuietFormatter < SimpleTextFormatter
8
- def report_summary(file_count, offense_count, correction_count)
8
+ def report_summary(file_count, offense_count, correction_count, correctable_count)
9
9
  super unless offense_count.zero?
10
10
  end
11
11
  end
@@ -23,6 +23,7 @@ module RuboCop
23
23
  def started(_target_files)
24
24
  @total_offense_count = 0
25
25
  @total_correction_count = 0
26
+ @total_correctable_count = 0
26
27
  end
27
28
 
28
29
  def file_finished(file, offenses)
@@ -35,7 +36,8 @@ module RuboCop
35
36
  def finished(inspected_files)
36
37
  report_summary(inspected_files.count,
37
38
  @total_offense_count,
38
- @total_correction_count)
39
+ @total_correction_count,
40
+ @total_correctable_count)
39
41
  end
40
42
 
41
43
  def report_file(file, offenses)
@@ -52,11 +54,13 @@ module RuboCop
52
54
  end
53
55
  end
54
56
 
55
- def report_summary(file_count, offense_count, correction_count)
57
+ def report_summary(file_count, offense_count, correction_count, correctable_count)
56
58
  report = Report.new(file_count,
57
59
  offense_count,
58
60
  correction_count,
59
- rainbow)
61
+ correctable_count,
62
+ rainbow,
63
+ safe_auto_correct: @options[:safe_auto_correct])
60
64
 
61
65
  output.puts
62
66
  output.puts report.summary
@@ -66,7 +70,9 @@ module RuboCop
66
70
 
67
71
  def count_stats(offenses)
68
72
  @total_offense_count += offenses.count
69
- @total_correction_count += offenses.count(&:corrected?)
73
+ corrected = offenses.count(&:corrected?)
74
+ @total_correction_count += corrected
75
+ @total_correctable_count += offenses.count(&:correctable?) - corrected
70
76
  end
71
77
 
72
78
  def colored_severity_code(offense)
@@ -96,16 +102,30 @@ module RuboCop
96
102
  include Colorizable
97
103
  include TextUtil
98
104
 
99
- def initialize(file_count, offense_count, correction_count, rainbow)
105
+ # rubocop:disable Metrics/ParameterLists
106
+ def initialize(
107
+ file_count, offense_count, correction_count, correctable_count, rainbow,
108
+ safe_auto_correct: false
109
+ )
100
110
  @file_count = file_count
101
111
  @offense_count = offense_count
102
112
  @correction_count = correction_count
113
+ @correctable_count = correctable_count
103
114
  @rainbow = rainbow
115
+ @safe_auto_correct = safe_auto_correct
104
116
  end
117
+ # rubocop:enable Metrics/ParameterLists
105
118
 
106
119
  def summary
107
120
  if @correction_count.positive?
108
- "#{files} inspected, #{offenses} detected, #{corrections} corrected"
121
+ if @correctable_count.positive?
122
+ "#{files} inspected, #{offenses} detected, #{corrections} corrected,"\
123
+ " #{correctable}"
124
+ else
125
+ "#{files} inspected, #{offenses} detected, #{corrections} corrected"
126
+ end
127
+ elsif @correctable_count.positive?
128
+ "#{files} inspected, #{offenses} detected, #{correctable}"
109
129
  else
110
130
  "#{files} inspected, #{offenses} detected"
111
131
  end
@@ -132,6 +152,16 @@ module RuboCop
132
152
 
133
153
  colorize(text, color)
134
154
  end
155
+
156
+ def correctable
157
+ if @safe_auto_correct
158
+ text = pluralize(@correctable_count, 'more offense')
159
+ "#{colorize(text, :yellow)} can be corrected with `rubocop -A`"
160
+ else
161
+ text = pluralize(@correctable_count, 'offense')
162
+ "#{colorize(text, :yellow)} auto-correctable"
163
+ end
164
+ end
135
165
  end
136
166
  end
137
167
  end
@@ -2,6 +2,7 @@
2
2
 
3
3
  module RuboCop
4
4
  # Common functionality for finding names that are similar to a given name.
5
+ # @api private
5
6
  module NameSimilarity
6
7
  module_function
7
8
 
@@ -8,6 +8,7 @@ module RuboCop
8
8
  class OptionArgumentError < StandardError; end
9
9
 
10
10
  # This class handles command line options.
11
+ # @api private
11
12
  class Options
12
13
  E_STDIN_NO_PATH = '-s/--stdin requires exactly one path, relative to the ' \
13
14
  'root of the project. RuboCop will use this path to determine which ' \
@@ -68,6 +69,7 @@ module RuboCop
68
69
 
69
70
  add_severity_option(opts)
70
71
  add_flags_with_optional_args(opts)
72
+ add_cache_options(opts)
71
73
  add_boolean_flags(opts)
72
74
  add_aliases(opts)
73
75
 
@@ -112,20 +114,19 @@ module RuboCop
112
114
  def add_auto_gen_options(opts)
113
115
  option(opts, '--auto-gen-config')
114
116
 
117
+ option(opts, '--regenerate-todo') do
118
+ @options.replace(ConfigRegeneration.new.options.merge(@options))
119
+ end
120
+
115
121
  option(opts, '--exclude-limit COUNT') do
116
122
  @validator.validate_exclude_limit_option
117
123
  end
118
124
 
119
125
  option(opts, '--disable-uncorrectable')
120
126
 
121
- option(opts, '--no-offense-counts') do
122
- @options[:no_offense_counts] = true
123
- end
124
-
125
- option(opts, '--auto-gen-only-exclude')
126
- option(opts, '--no-auto-gen-timestamp') do
127
- @options[:no_auto_gen_timestamp] = true
128
- end
127
+ option(opts, '--[no-]offense-counts')
128
+ option(opts, '--[no-]auto-gen-only-exclude')
129
+ option(opts, '--[no-]auto-gen-timestamp')
129
130
 
130
131
  option(opts, '--init')
131
132
  end
@@ -163,10 +164,16 @@ module RuboCop
163
164
  end
164
165
  end
165
166
 
167
+ def add_cache_options(opts)
168
+ option(opts, '-C', '--cache FLAG')
169
+ option(opts, '--cache-root DIR') do
170
+ @validator.validate_cache_enabled_for_cache_root
171
+ end
172
+ end
173
+
166
174
  # rubocop:disable Metrics/MethodLength, Metrics/AbcSize
167
175
  def add_boolean_flags(opts)
168
176
  option(opts, '-F', '--fail-fast')
169
- option(opts, '-C', '--cache FLAG')
170
177
  option(opts, '-d', '--debug')
171
178
  option(opts, '-D', '--[no-]display-cop-names')
172
179
  option(opts, '-E', '--extra-details')
@@ -232,6 +239,7 @@ module RuboCop
232
239
  end
233
240
 
234
241
  # Validates option arguments and the options' compatibility with each other.
242
+ # @api private
235
243
  class OptionsValidator
236
244
  class << self
237
245
  # Cop name validation must be done later than option parsing, so it's not
@@ -309,7 +317,7 @@ module RuboCop
309
317
 
310
318
  message = '--%<flag>s can only be used together with --auto-gen-config.'
311
319
 
312
- %i[exclude_limit no_offense_counts no_auto_gen_timestamp
320
+ %i[exclude_limit offense_counts auto_gen_timestamp
313
321
  auto_gen_only_exclude].each do |option|
314
322
  if @options.key?(option)
315
323
  raise OptionArgumentError,
@@ -390,9 +398,17 @@ module RuboCop
390
398
  # of option order.
391
399
  raise OptionParser::MissingArgument
392
400
  end
401
+
402
+ def validate_cache_enabled_for_cache_root
403
+ return unless @options[:cache] == 'false'
404
+
405
+ raise OptionArgumentError, '--cache-root can not be used with ' \
406
+ '--cache false'
407
+ end
393
408
  end
394
409
 
395
410
  # This module contains help texts for command line options.
411
+ # @api private
396
412
  module OptionsHelp
397
413
  MAX_EXCL = RuboCop::Options::DEFAULT_MAXIMUM_EXCLUSION_ITEMS.to_s
398
414
  FORMATTER_OPTION_LIST = RuboCop::Formatter::FormatterSet::BUILTIN_FORMATTERS_FOR_KEYS.keys
@@ -406,17 +422,20 @@ module RuboCop
406
422
  config: 'Specify configuration file.',
407
423
  auto_gen_config: ['Generate a configuration file acting as a',
408
424
  'TODO list.'],
409
- no_offense_counts: ['Do not include offense counts in configuration',
410
- 'file generated by --auto-gen-config.'],
411
- no_auto_gen_timestamp:
412
- ['Do not include the date and time when',
413
- 'the --auto-gen-config was run in the file it',
414
- 'generates.'],
425
+ regenerate_todo: ['Regenerate the TODO configuration file using',
426
+ 'the last configuration. If there is no existing',
427
+ 'TODO file, acts like --auto-gen-config.'],
428
+ offense_counts: ['Include offense counts in configuration',
429
+ 'file generated by --auto-gen-config.',
430
+ 'Default is true.'],
431
+ auto_gen_timestamp:
432
+ ['Include the date and time when the --auto-gen-config',
433
+ 'was run in the file it generates. Default is true.'],
415
434
  auto_gen_only_exclude:
416
435
  ['Generate only Exclude parameters and not Max',
417
436
  'when running --auto-gen-config, except if the',
418
437
  'number of files with offenses is bigger than',
419
- 'exclude-limit.'],
438
+ 'exclude-limit. Default is false.'],
420
439
  exclude_limit: ['Used together with --auto-gen-config to',
421
440
  'set the limit for how many Exclude',
422
441
  "properties to generate. Default is #{MAX_EXCL}."],
@@ -461,6 +480,10 @@ module RuboCop
461
480
  cache: ["Use result caching (FLAG=true) or don't",
462
481
  '(FLAG=false), default determined by',
463
482
  'configuration parameter AllCops: UseCache.'],
483
+ cache_root: ['Set the cache root directory.',
484
+ 'Takes precedence over the configuration',
485
+ 'parameter AllCops: CacheRootDirectory and',
486
+ 'the $RUBOCOP_CACHE_ROOT environment variable.'],
464
487
  debug: 'Display debug info.',
465
488
  display_cop_names: ['Display cop names in offense messages.',
466
489
  'Default is true.'],
@@ -5,6 +5,7 @@ require 'time'
5
5
 
6
6
  module RuboCop
7
7
  # Common methods and behaviors for dealing with remote config files.
8
+ # @api private
8
9
  class RemoteConfig
9
10
  attr_reader :uri
10
11
 
@@ -3,9 +3,11 @@
3
3
  require 'digest/sha1'
4
4
  require 'find'
5
5
  require 'etc'
6
+ require 'zlib'
6
7
 
7
8
  module RuboCop
8
9
  # Provides functionality for caching rubocop runs.
10
+ # @api private
9
11
  class ResultCache
10
12
  NON_CHANGING = %i[color format formatters out debug fail_level auto_correct
11
13
  cache fail_fast stdin parallel].freeze
@@ -29,6 +31,11 @@ module RuboCop
29
31
  end
30
32
 
31
33
  class << self
34
+ # @api private
35
+ attr_accessor :rubocop_required_features
36
+
37
+ ResultCache.rubocop_required_features = []
38
+
32
39
  private
33
40
 
34
41
  def requires_file_removal?(file_count, config_store)
@@ -60,7 +67,8 @@ module RuboCop
60
67
  end
61
68
 
62
69
  def self.cache_root(config_store)
63
- root = config_store.for_pwd.for_all_cops['CacheRootDirectory']
70
+ root = ENV['RUBOCOP_CACHE_ROOT']
71
+ root ||= config_store.for_pwd.for_all_cops['CacheRootDirectory']
64
72
  root ||= if ENV.key?('XDG_CACHE_HOME')
65
73
  # Include user ID in the path to make sure the user has write
66
74
  # access.
@@ -75,7 +83,10 @@ module RuboCop
75
83
  config_store.for_pwd.for_all_cops['AllowSymlinksInCacheRootDirectory']
76
84
  end
77
85
 
86
+ attr :path
87
+
78
88
  def initialize(file, team, options, config_store, cache_root = nil)
89
+ cache_root ||= options[:cache_root]
79
90
  cache_root ||= ResultCache.cache_root(config_store)
80
91
  @allow_symlinks_in_cache_location =
81
92
  ResultCache.allow_symlinks_in_cache_location?(config_store)
@@ -84,6 +95,11 @@ module RuboCop
84
95
  context_checksum(team, options),
85
96
  file_checksum(file, config_store))
86
97
  @cached_data = CachedData.new(file)
98
+ @debug = options[:debug]
99
+ end
100
+
101
+ def debug?
102
+ @debug
87
103
  end
88
104
 
89
105
  def valid?
@@ -91,6 +107,7 @@ module RuboCop
91
107
  end
92
108
 
93
109
  def load
110
+ puts "Loading cache from #{@path}" if debug?
94
111
  @cached_data.from_json(IO.read(@path, encoding: Encoding::UTF_8))
95
112
  end
96
113
 
@@ -158,27 +175,34 @@ module RuboCop
158
175
  end
159
176
 
160
177
  # The checksum of the rubocop program running the inspection.
161
- # rubocop:disable Metrics/AbcSize
162
178
  def rubocop_checksum
163
179
  ResultCache.source_checksum ||=
164
180
  begin
165
- lib_root = File.join(File.dirname(__FILE__), '..')
166
- exe_root = File.join(lib_root, '..', 'exe')
167
-
168
- # These are all the files we have `require`d plus everything in the
169
- # exe directory. A change to any of them could affect the cop output
170
- # so we include them in the cache hash.
171
- source_files = $LOADED_FEATURES + Find.find(exe_root).to_a
172
-
173
181
  digest = Digest::SHA1.new
174
- source_files
182
+ rubocop_extra_features
175
183
  .select { |path| File.file?(path) }
176
184
  .sort!
177
- .each { |path| digest << File.mtime(path).to_s }
185
+ .each do |path|
186
+ content = File.open(path, 'rb', &:read)
187
+ digest << Zlib.crc32(content).to_s # mtime not reliable
188
+ end
189
+ digest << RuboCop::Version::STRING << RuboCop::AST::Version::STRING
178
190
  digest.hexdigest
179
191
  end
180
192
  end
181
- # rubocop:enable Metrics/AbcSize
193
+
194
+ def rubocop_extra_features
195
+ lib_root = File.join(File.dirname(__FILE__), '..')
196
+ exe_root = File.join(lib_root, '..', 'exe')
197
+
198
+ # These are all the files we have `require`d plus everything in the
199
+ # exe directory. A change to any of them could affect the cop output
200
+ # so we include them in the cache hash.
201
+ source_files = $LOADED_FEATURES + Find.find(exe_root).to_a
202
+ source_files -= ResultCache.rubocop_required_features # Rely on gem versions
203
+
204
+ source_files
205
+ end
182
206
 
183
207
  # Return a hash of the options given at invocation, minus the ones that have
184
208
  # no effect on which offenses and disabled line ranges are found, and thus
@@ -191,8 +215,8 @@ module RuboCop
191
215
  # The external dependency checksums are cached per RuboCop team so that
192
216
  # the checksums don't need to be recomputed for each file.
193
217
  def team_checksum(team)
194
- @checksum_by_team ||= {}
195
- @checksum_by_team[team.object_id] ||= team.external_dependency_checksum
218
+ @checksum_by_team ||= {}.compare_by_identity
219
+ @checksum_by_team[team] ||= team.external_dependency_checksum
196
220
  end
197
221
 
198
222
  # We combine team and options into a single "context" checksum to avoid
@@ -17,13 +17,16 @@ module CopHelper
17
17
  RuboCop::Formatter::DisabledConfigFormatter.config_to_allow_offenses = {}
18
18
  RuboCop::Formatter::DisabledConfigFormatter.detected_styles = {}
19
19
  processed_source = parse_source(source, file)
20
- raise 'Error parsing example code' unless processed_source.valid_syntax?
20
+ unless processed_source.valid_syntax?
21
+ raise 'Error parsing example code: ' \
22
+ "#{processed_source.diagnostics.map(&:render).join("\n")}"
23
+ end
21
24
 
22
25
  _investigate(cop, processed_source)
23
26
  end
24
27
 
25
28
  def parse_source(source, file = nil)
26
- if file&.respond_to?(:write)
29
+ if file.respond_to?(:write)
27
30
  file.write(source)
28
31
  file.rewind
29
32
  file = file.path
@@ -128,14 +128,19 @@ module RuboCop
128
128
  @processed_source = parse_source(expected_annotations.plain_source,
129
129
  file)
130
130
 
131
- raise 'Error parsing example code' unless @processed_source.valid_syntax?
131
+ unless @processed_source.valid_syntax?
132
+ raise 'Error parsing example code: ' \
133
+ "#{@processed_source.diagnostics.map(&:render).join("\n")}"
134
+ end
132
135
 
133
- offenses = _investigate(cop, @processed_source)
136
+ @offenses = _investigate(cop, @processed_source)
134
137
  actual_annotations =
135
- expected_annotations.with_offense_annotations(offenses)
138
+ expected_annotations.with_offense_annotations(@offenses)
136
139
 
137
140
  expect(actual_annotations).to eq(expected_annotations), ''
138
- expect(offenses.map(&:severity).uniq).to eq([severity]) if severity
141
+ expect(@offenses.map(&:severity).uniq).to eq([severity]) if severity
142
+
143
+ @offenses
139
144
  end
140
145
 
141
146
  def expect_correction(correction, loop: true)
@@ -152,7 +157,7 @@ module RuboCop
152
157
  break corrected_source if corrected_source == @processed_source.buffer.source
153
158
 
154
159
  if iteration > RuboCop::Runner::MAX_ITERATIONS
155
- raise RuboCop::Runner::InfiniteCorrectionLoop.new(@processed_source.path, [])
160
+ raise RuboCop::Runner::InfiniteCorrectionLoop.new(@processed_source.path, [@offenses])
156
161
  end
157
162
 
158
163
  # Prepare for next loop
@@ -236,10 +241,10 @@ module RuboCop
236
241
  def match_annotations?(other)
237
242
  annotations.zip(other.annotations) do |(_actual_line, actual_annotation),
238
243
  (_expected_line, expected_annotation)|
239
- if expected_annotation&.end_with?(ABBREV)
240
- if actual_annotation.start_with?(expected_annotation[0...-ABBREV.length])
241
- expected_annotation.replace(actual_annotation)
242
- end
244
+ if expected_annotation&.end_with?(ABBREV) &&
245
+ actual_annotation.start_with?(expected_annotation[0...-ABBREV.length])
246
+
247
+ expected_annotation.replace(actual_annotation)
243
248
  end
244
249
  end
245
250