rubocop 1.73.1 → 1.75.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (127) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +1 -1
  3. data/config/default.yml +64 -10
  4. data/config/internal_affairs.yml +4 -0
  5. data/config/obsoletion.yml +3 -1
  6. data/lib/rubocop/cli.rb +1 -1
  7. data/lib/rubocop/config.rb +35 -6
  8. data/lib/rubocop/config_loader.rb +4 -1
  9. data/lib/rubocop/config_loader_resolver.rb +2 -1
  10. data/lib/rubocop/config_obsoletion/extracted_cop.rb +4 -3
  11. data/lib/rubocop/config_obsoletion/renamed_cop.rb +18 -3
  12. data/lib/rubocop/config_obsoletion.rb +46 -2
  13. data/lib/rubocop/config_validator.rb +1 -0
  14. data/lib/rubocop/cop/internal_affairs/example_description.rb +3 -1
  15. data/lib/rubocop/cop/internal_affairs/node_pattern_groups.rb +1 -1
  16. data/lib/rubocop/cop/internal_affairs/node_type_group.rb +91 -0
  17. data/lib/rubocop/cop/internal_affairs/redundant_described_class_as_subject.rb +6 -5
  18. data/lib/rubocop/cop/internal_affairs.rb +1 -0
  19. data/lib/rubocop/cop/layout/block_alignment.rb +1 -0
  20. data/lib/rubocop/cop/layout/block_end_newline.rb +1 -0
  21. data/lib/rubocop/cop/layout/else_alignment.rb +1 -1
  22. data/lib/rubocop/cop/layout/empty_line_between_defs.rb +1 -1
  23. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +1 -0
  24. data/lib/rubocop/cop/layout/empty_lines_around_block_body.rb +1 -0
  25. data/lib/rubocop/cop/layout/indentation_width.rb +1 -0
  26. data/lib/rubocop/cop/layout/line_length.rb +5 -1
  27. data/lib/rubocop/cop/layout/multiline_block_layout.rb +1 -0
  28. data/lib/rubocop/cop/layout/multiline_method_parameter_line_breaks.rb +1 -0
  29. data/lib/rubocop/cop/layout/redundant_line_break.rb +9 -5
  30. data/lib/rubocop/cop/layout/rescue_ensure_alignment.rb +1 -1
  31. data/lib/rubocop/cop/layout/space_around_operators.rb +4 -1
  32. data/lib/rubocop/cop/layout/space_before_block_braces.rb +1 -0
  33. data/lib/rubocop/cop/layout/space_inside_block_braces.rb +1 -0
  34. data/lib/rubocop/cop/lint/debugger.rb +2 -2
  35. data/lib/rubocop/cop/lint/empty_conditional_body.rb +15 -70
  36. data/lib/rubocop/cop/lint/erb_new_arguments.rb +0 -6
  37. data/lib/rubocop/cop/lint/literal_as_condition.rb +4 -0
  38. data/lib/rubocop/cop/lint/non_local_exit_from_iterator.rb +2 -2
  39. data/lib/rubocop/cop/lint/raise_exception.rb +29 -10
  40. data/lib/rubocop/cop/lint/redundant_type_conversion.rb +9 -3
  41. data/lib/rubocop/cop/lint/redundant_with_index.rb +3 -0
  42. data/lib/rubocop/cop/lint/redundant_with_object.rb +3 -0
  43. data/lib/rubocop/cop/lint/return_in_void_context.rb +4 -11
  44. data/lib/rubocop/cop/lint/shadowing_outer_local_variable.rb +8 -1
  45. data/lib/rubocop/cop/lint/shared_mutable_default.rb +12 -1
  46. data/lib/rubocop/cop/lint/unexpected_block_arity.rb +2 -0
  47. data/lib/rubocop/cop/lint/unreachable_code.rb +1 -0
  48. data/lib/rubocop/cop/lint/unreachable_loop.rb +5 -5
  49. data/lib/rubocop/cop/lint/useless_access_modifier.rb +1 -0
  50. data/lib/rubocop/cop/lint/useless_constant_scoping.rb +2 -11
  51. data/lib/rubocop/cop/lint/void.rb +1 -0
  52. data/lib/rubocop/cop/metrics/block_length.rb +1 -0
  53. data/lib/rubocop/cop/metrics/method_length.rb +1 -0
  54. data/lib/rubocop/cop/metrics/utils/code_length_calculator.rb +1 -1
  55. data/lib/rubocop/cop/mixin/check_line_breakable.rb +2 -2
  56. data/lib/rubocop/cop/mixin/check_single_line_suitability.rb +1 -1
  57. data/lib/rubocop/cop/mixin/forbidden_identifiers.rb +20 -0
  58. data/lib/rubocop/cop/mixin/forbidden_pattern.rb +16 -0
  59. data/lib/rubocop/cop/mixin/method_complexity.rb +1 -0
  60. data/lib/rubocop/cop/mixin/range_help.rb +12 -0
  61. data/lib/rubocop/cop/mixin/target_ruby_version.rb +1 -1
  62. data/lib/rubocop/cop/naming/method_name.rb +64 -8
  63. data/lib/rubocop/cop/naming/variable_name.rb +6 -19
  64. data/lib/rubocop/cop/registry.rb +9 -6
  65. data/lib/rubocop/cop/style/array_intersect.rb +39 -28
  66. data/lib/rubocop/cop/style/block_delimiters.rb +2 -1
  67. data/lib/rubocop/cop/style/class_and_module_children.rb +29 -7
  68. data/lib/rubocop/cop/style/collection_methods.rb +1 -0
  69. data/lib/rubocop/cop/style/combinable_loops.rb +1 -0
  70. data/lib/rubocop/cop/style/commented_keyword.rb +9 -2
  71. data/lib/rubocop/cop/style/comparable_between.rb +75 -0
  72. data/lib/rubocop/cop/style/double_negation.rb +1 -1
  73. data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -7
  74. data/lib/rubocop/cop/style/exponential_notation.rb +2 -2
  75. data/lib/rubocop/cop/style/for.rb +1 -0
  76. data/lib/rubocop/cop/style/format_string_token.rb +38 -11
  77. data/lib/rubocop/cop/style/guard_clause.rb +2 -1
  78. data/lib/rubocop/cop/style/hash_each_methods.rb +3 -2
  79. data/lib/rubocop/cop/style/hash_fetch_chain.rb +105 -0
  80. data/lib/rubocop/cop/style/if_inside_else.rb +10 -13
  81. data/lib/rubocop/cop/style/if_unless_modifier.rb +2 -2
  82. data/lib/rubocop/cop/style/inverse_methods.rb +9 -5
  83. data/lib/rubocop/cop/style/invertible_unless_condition.rb +2 -2
  84. data/lib/rubocop/cop/style/ip_addresses.rb +2 -2
  85. data/lib/rubocop/cop/style/it_block_parameter.rb +100 -0
  86. data/lib/rubocop/cop/style/keyword_parameters_order.rb +13 -7
  87. data/lib/rubocop/cop/style/lambda.rb +1 -0
  88. data/lib/rubocop/cop/style/map_into_array.rb +1 -0
  89. data/lib/rubocop/cop/style/method_call_with_args_parentheses/omit_parentheses.rb +3 -3
  90. data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +1 -0
  91. data/lib/rubocop/cop/style/multiline_block_chain.rb +2 -1
  92. data/lib/rubocop/cop/style/multiline_method_signature.rb +1 -9
  93. data/lib/rubocop/cop/style/next.rb +44 -0
  94. data/lib/rubocop/cop/style/object_then.rb +1 -0
  95. data/lib/rubocop/cop/style/proc.rb +1 -0
  96. data/lib/rubocop/cop/style/raise_args.rb +8 -8
  97. data/lib/rubocop/cop/style/redundant_begin.rb +1 -0
  98. data/lib/rubocop/cop/style/redundant_condition.rb +2 -3
  99. data/lib/rubocop/cop/style/redundant_current_directory_in_path.rb +14 -4
  100. data/lib/rubocop/cop/style/redundant_format.rb +10 -3
  101. data/lib/rubocop/cop/style/redundant_freeze.rb +1 -1
  102. data/lib/rubocop/cop/style/redundant_parentheses.rb +2 -1
  103. data/lib/rubocop/cop/style/redundant_self.rb +1 -0
  104. data/lib/rubocop/cop/style/redundant_sort_by.rb +17 -1
  105. data/lib/rubocop/cop/style/rescue_modifier.rb +3 -0
  106. data/lib/rubocop/cop/style/select_by_regexp.rb +4 -1
  107. data/lib/rubocop/cop/style/single_line_do_end_block.rb +3 -1
  108. data/lib/rubocop/cop/style/sole_nested_conditional.rb +41 -106
  109. data/lib/rubocop/cop/style/symbol_proc.rb +2 -0
  110. data/lib/rubocop/cop/style/top_level_method_definition.rb +1 -0
  111. data/lib/rubocop/cop/utils/format_string.rb +5 -2
  112. data/lib/rubocop/cop/variable_force/scope.rb +1 -1
  113. data/lib/rubocop/cop/variable_force/variable.rb +1 -6
  114. data/lib/rubocop/cop/variable_force.rb +1 -1
  115. data/lib/rubocop/directive_comment.rb +1 -1
  116. data/lib/rubocop/ext/regexp_node.rb +0 -1
  117. data/lib/rubocop/lsp/runtime.rb +4 -4
  118. data/lib/rubocop/lsp/stdin_runner.rb +3 -1
  119. data/lib/rubocop/rspec/cop_helper.rb +4 -1
  120. data/lib/rubocop/rspec/shared_contexts.rb +20 -0
  121. data/lib/rubocop/rspec/support.rb +2 -0
  122. data/lib/rubocop/runner.rb +5 -1
  123. data/lib/rubocop/target_ruby.rb +1 -1
  124. data/lib/rubocop/version.rb +14 -7
  125. data/lib/rubocop.rb +5 -0
  126. data/lib/ruby_lsp/rubocop/runtime_adapter.rb +20 -2
  127. metadata +12 -6
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: a4215b7d095dfd177a36af3bdf5d5aecd786877a6feca3480078d9cb1a5917ad
4
- data.tar.gz: fc20be6f63f1a0e3c06cd2fc13f8619f872e9a81642dfa833f90f88f70997857
3
+ metadata.gz: a064330786019e0efee6b496e761b05bea0286f0afdcbb4fffad2f07433aef9d
4
+ data.tar.gz: eb1ee00d5fe060ea49203635df3c7457df6f416de93c9801a687386aa02f4768
5
5
  SHA512:
6
- metadata.gz: aee7de1fa54b8b7108f5ad7f29c3fbc1818547f28507010dca91adc0f648b02ce519c96846d2ad98407d8f8847047cd8f88547e7d5c81028d0f54cfafb44500b
7
- data.tar.gz: afc90ab7e887e183ddfac0bf49e9f644c04564e7fd605423d29f2848f6cc509f2394fd642eccd8870d105b21055a5cb4d0dea051b92ba47ef3befbf5b7d3dad5
6
+ metadata.gz: 336c6e38cc6162a00ca4c9780dcc71fb4b03a785af0eb6abb6c30baf2e1c8512af12e921a2036564102290324e418278cb83ceb06d7a1ec44ddf314923275f89
7
+ data.tar.gz: cbe9d5ab5685dbab70de2b281ecc807494631a294273260d87315d6875b753d9c2b60f6d7c14379eaf5b384894a4400c747ef91a57323c418c66a51076b33f06
data/README.md CHANGED
@@ -52,7 +52,7 @@ To prevent an unwanted RuboCop update you might want to use a conservative versi
52
52
  in your `Gemfile`:
53
53
 
54
54
  ```rb
55
- gem 'rubocop', '~> 1.73', require: false
55
+ gem 'rubocop', '~> 1.75', require: false
56
56
  ```
57
57
 
58
58
  See [our versioning policy](https://docs.rubocop.org/rubocop/versioning.html) for further details.
data/config/default.yml CHANGED
@@ -146,12 +146,14 @@ AllCops:
146
146
  # Ruby version is still unresolved, RuboCop will use the oldest officially
147
147
  # supported Ruby version (currently Ruby 2.7).
148
148
  TargetRubyVersion: ~
149
- # You can specify the parser engine. There are two options available:
149
+ # RuboCop choses the parser engine automatically but you can also specify it yourself.
150
+ # These options are available:
151
+ # - `default`
150
152
  # - `parser_whitequark` ... https://github.com/whitequark/parser
151
153
  # - `parser_prism` ... https://github.com/ruby/prism (`Prism::Translation::Parser`)
152
- # By default, `parser` is used. For the `TargetRubyVersion` value, `parser` can be specified for versions `2.0` and above.
153
- # `parser_prism` can be specified for versions `3.3` and above. `parser_prism` is faster but still considered experimental.
154
- ParserEngine: parser_whitequark
154
+ # Not every parser engine can handle every `TargetRubyVersion`. `parser_whitequark`
155
+ # only works with versions 3.4 and below, `parser_prism` with versions 3.3 and above.
156
+ ParserEngine: default
155
157
  # Determines if a notification for extension libraries should be shown when
156
158
  # rubocop is run. Keys are the name of the extension, and values are an array
157
159
  # of gems in the Gemfile that the extension is suggested for, if not already
@@ -1887,10 +1889,9 @@ Lint/EmptyConditionalBody:
1887
1889
  Description: 'Checks for the presence of `if`, `elsif` and `unless` branches without a body.'
1888
1890
  Enabled: true
1889
1891
  AutoCorrect: contextual
1890
- SafeAutoCorrect: false
1891
1892
  AllowComments: true
1892
1893
  VersionAdded: '0.89'
1893
- VersionChanged: '1.61'
1894
+ VersionChanged: '1.73'
1894
1895
 
1895
1896
  Lint/EmptyEnsure:
1896
1897
  Description: 'Checks for empty ensure block.'
@@ -2996,6 +2997,7 @@ Naming/MethodName:
2996
2997
  StyleGuide: '#snake-case-symbols-methods-vars'
2997
2998
  Enabled: true
2998
2999
  VersionAdded: '0.50'
3000
+ VersionChanged: '1.75'
2999
3001
  EnforcedStyle: snake_case
3000
3002
  SupportedStyles:
3001
3003
  - snake_case
@@ -3007,6 +3009,10 @@ Naming/MethodName:
3007
3009
  # - '\A\s*onSelectionCleared\s*'
3008
3010
  #
3009
3011
  AllowedPatterns: []
3012
+ ForbiddenIdentifiers:
3013
+ - __id__
3014
+ - __send__
3015
+ ForbiddenPatterns: []
3010
3016
 
3011
3017
  Naming/MethodParameterName:
3012
3018
  Description: >-
@@ -3043,17 +3049,19 @@ Naming/PredicateName:
3043
3049
  StyleGuide: '#bool-methods-qmark'
3044
3050
  Enabled: true
3045
3051
  VersionAdded: '0.50'
3046
- VersionChanged: '0.77'
3052
+ VersionChanged: '1.75'
3047
3053
  # Predicate name prefixes.
3048
3054
  NamePrefix:
3049
3055
  - is_
3050
3056
  - has_
3051
3057
  - have_
3058
+ - does_
3052
3059
  # Predicate name prefixes that should be removed.
3053
3060
  ForbiddenPrefixes:
3054
3061
  - is_
3055
3062
  - has_
3056
3063
  - have_
3064
+ - does_
3057
3065
  # Predicate names which, despite having a forbidden prefix, or no `?`,
3058
3066
  # should still be accepted
3059
3067
  AllowedMethods:
@@ -3105,6 +3113,8 @@ Naming/VariableNumber:
3105
3113
  CheckMethodNames: true
3106
3114
  CheckSymbols: true
3107
3115
  AllowedIdentifiers:
3116
+ - TLS1_1 # OpenSSL::SSL::TLS1_1_VERSION
3117
+ - TLS1_2 # OpenSSL::SSL::TLS1_2_VERSION
3108
3118
  - capture3 # Open3.capture3
3109
3119
  - iso8601 # Time#iso8601
3110
3120
  - rfc1123_date # CGI.rfc1123_date
@@ -3500,6 +3510,7 @@ Style/ClassAndModuleChildren:
3500
3510
  SafeAutoCorrect: false
3501
3511
  Enabled: true
3502
3512
  VersionAdded: '0.19'
3513
+ VersionChanged: '1.74'
3503
3514
  #
3504
3515
  # Basically there are two different styles:
3505
3516
  #
@@ -3515,7 +3526,21 @@ Style/ClassAndModuleChildren:
3515
3526
  #
3516
3527
  # The compact style is only forced, for classes or modules with one child.
3517
3528
  EnforcedStyle: nested
3518
- SupportedStyles:
3529
+ SupportedStyles: &supported_styles
3530
+ - nested
3531
+ - compact
3532
+ # Configure classes separately, if desired. If not set, or set to `nil`,
3533
+ # the `EnforcedStyle` value will be used.
3534
+ EnforcedStyleForClasses: ~
3535
+ SupportedStylesForClasses:
3536
+ - ~
3537
+ - nested
3538
+ - compact
3539
+ # Configure modules separately, if desired. If not set, or set to `nil`,
3540
+ # the `EnforcedStyle` value will be used.
3541
+ EnforcedStyleForModules: ~
3542
+ SupportedStylesForModules:
3543
+ - ~
3519
3544
  - nested
3520
3545
  - compact
3521
3546
 
@@ -3668,6 +3693,12 @@ Style/CommentedKeyword:
3668
3693
  VersionAdded: '0.51'
3669
3694
  VersionChanged: '1.19'
3670
3695
 
3696
+ Style/ComparableBetween:
3697
+ Description: 'Enforces the use of `Comparable#between?` instead of logical comparison.'
3698
+ Enabled: pending
3699
+ VersionAdded: '1.74'
3700
+ StyleGuide: '#ranges-or-between'
3701
+
3671
3702
  Style/ComparableClamp:
3672
3703
  Description: 'Enforces the use of `Comparable#clamp` instead of comparison by minimum and maximum.'
3673
3704
  Enabled: pending
@@ -4070,8 +4101,14 @@ Style/FormatStringToken:
4070
4101
  # style token in a format string to be allowed when enforced style is not
4071
4102
  # `unannotated`.
4072
4103
  MaxUnannotatedPlaceholdersAllowed: 1
4104
+ # The mode the cop operates in. Two values are allowed:
4105
+ # * aggressive (default): all strings are considered
4106
+ # * conservative:
4107
+ # only register offenses for strings given to `printf`, `sprintf`,
4108
+ # format` and `%` methods. Other strings are not considered.
4109
+ Mode: aggressive
4073
4110
  VersionAdded: '0.49'
4074
- VersionChanged: '1.0'
4111
+ VersionChanged: '1.74'
4075
4112
  AllowedMethods: []
4076
4113
  AllowedPatterns: []
4077
4114
 
@@ -4167,6 +4204,12 @@ Style/HashExcept:
4167
4204
  VersionAdded: '1.7'
4168
4205
  VersionChanged: '1.39'
4169
4206
 
4207
+ Style/HashFetchChain:
4208
+ Description: 'Use `Hash#dig` instead of chained `fetch` calls.'
4209
+ Enabled: pending
4210
+ Safe: false
4211
+ VersionAdded: '1.75'
4212
+
4170
4213
  Style/HashLikeCase:
4171
4214
  Description: >-
4172
4215
  Checks for places where `case-when` represents a simple 1:1
@@ -4377,6 +4420,16 @@ Style/ItAssignment:
4377
4420
  Enabled: pending
4378
4421
  VersionAdded: '1.70'
4379
4422
 
4423
+ Style/ItBlockParameter:
4424
+ Description: 'Checks for blocks with one argument where `it` block parameter can be used.'
4425
+ Enabled: pending
4426
+ EnforcedStyle: only_numbered_parameters
4427
+ SupportedStyles:
4428
+ - only_numbered_parameters
4429
+ - always
4430
+ - disallow
4431
+ VersionAdded: '1.75'
4432
+
4380
4433
  Style/KeywordArgumentsMerging:
4381
4434
  Description: >-
4382
4435
  When passing an existing hash as keyword arguments, provide additional arguments
@@ -4765,7 +4818,7 @@ Style/Next:
4765
4818
  StyleGuide: '#no-nested-conditionals'
4766
4819
  Enabled: true
4767
4820
  VersionAdded: '0.22'
4768
- VersionChanged: '0.35'
4821
+ VersionChanged: '1.75'
4769
4822
  # With `always` all conditions at the end of an iteration needs to be
4770
4823
  # replaced by next - with `skip_modifier_ifs` the modifier if like this one
4771
4824
  # are ignored: [1, 2].each { |a| return 'yes' if a == 1 }
@@ -4773,6 +4826,7 @@ Style/Next:
4773
4826
  # `MinBodyLength` defines the number of lines of the a body of an `if` or `unless`
4774
4827
  # needs to have to trigger this cop
4775
4828
  MinBodyLength: 3
4829
+ AllowConsecutiveConditionals: false
4776
4830
  SupportedStyles:
4777
4831
  - skip_modifier_ifs
4778
4832
  - always
@@ -14,6 +14,10 @@ InternalAffairs/ExampleDescription:
14
14
  Include:
15
15
  - 'spec/rubocop/cop/**/*.rb'
16
16
 
17
+ InternalAffairs/NodeTypeGroup:
18
+ Include:
19
+ - 'lib/rubocop/cop/**/*.rb'
20
+
17
21
  InternalAffairs/OnSendWithoutOnCSend:
18
22
  Include:
19
23
  - 'lib/rubocop/cop/**/*.rb'
@@ -43,7 +43,9 @@ renamed:
43
43
  Style/MethodCallParentheses: Style/MethodCallWithoutArgsParentheses
44
44
  Style/MethodName: Naming/MethodName
45
45
  Style/OpMethod: Naming/BinaryOperatorParameterName
46
- Style/PredicateName: Naming/PredicateName
46
+ Style/PredicateName:
47
+ new_name: Naming/PredicateName
48
+ severity: warning
47
49
  Style/SingleSpaceBeforeFirstArg: Layout/SpaceBeforeFirstArg
48
50
  Style/UnneededCapitalW: Style/RedundantCapitalW
49
51
  Style/UnneededCondition: Style/RedundantCondition
data/lib/rubocop/cli.rb CHANGED
@@ -52,7 +52,7 @@ module RuboCop
52
52
  end
53
53
  end
54
54
  rescue ConfigNotFoundError, IncorrectCopNameError, OptionArgumentError => e
55
- warn e.message
55
+ warn Rainbow(e.message).red
56
56
  STATUS_ERROR
57
57
  rescue RuboCop::Error => e
58
58
  warn Rainbow("Error: #{e.message}").red
@@ -27,14 +27,43 @@ module RuboCop
27
27
  config
28
28
  end
29
29
 
30
- # rubocop:disable Metrics/AbcSize
30
+ # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
31
31
  def initialize(hash = RuboCop::ConfigLoader.default_configuration, loaded_path = nil)
32
32
  @loaded_path = loaded_path
33
33
  @for_cop = Hash.new do |h, cop|
34
34
  cop_name = cop.respond_to?(:cop_name) ? cop.cop_name : cop
35
- qualified_cop_name = Cop::Registry.qualified_cop_name(cop_name, loaded_path)
36
- cop_options = self[qualified_cop_name].dup || {}
37
- cop_options['Enabled'] = enable_cop?(qualified_cop_name, cop_options)
35
+
36
+ if ConfigObsoletion.deprecated_cop_name?(cop)
37
+ # Since a deprecated cop will no longer have a qualified name (as the badge is no
38
+ # longer valid), and since we do not want to automatically enable the cop, we just
39
+ # set the configuration to an empty hash if it is unset.
40
+ # This is necessary to allow a renamed cop have its old configuration merged in
41
+ # before being used (which is necessary to allow it to be disabled via config).
42
+ cop_options = self[cop_name].dup || {}
43
+ else
44
+ qualified_cop_name = Cop::Registry.qualified_cop_name(cop_name, loaded_path, warn: false)
45
+ cop_options = self[qualified_cop_name].dup || {}
46
+ cop_options['Enabled'] = enable_cop?(qualified_cop_name, cop_options)
47
+
48
+ # If the cop has deprecated names (ie. it has been renamed), it is possible that
49
+ # users will still have old configuration for the cop's old name. In this case,
50
+ # if `ConfigObsoletion` is configured to warn rather than error (and therefore
51
+ # RuboCop runs), we want to respect the old configuration, so merge it in.
52
+ #
53
+ # NOTE: If there is configuration for both the cop and a deprecated names, the old
54
+ # configuration will be merged on top of the new configuration!
55
+ ConfigObsoletion.deprecated_names_for(cop).each do |deprecated_cop_name|
56
+ deprecated_config = @for_cop[deprecated_cop_name]
57
+ next if deprecated_config.empty?
58
+
59
+ warn Rainbow(<<~WARNING).yellow
60
+ Warning: Using `#{deprecated_cop_name}` configuration in #{loaded_path} for `#{cop}`.
61
+ WARNING
62
+
63
+ cop_options.merge!(@for_cop[deprecated_cop_name])
64
+ end
65
+ end
66
+
38
67
  h[cop] = h[cop_name] = cop_options
39
68
  end
40
69
  @hash = hash
@@ -43,7 +72,7 @@ module RuboCop
43
72
  @badge_config_cache = {}.compare_by_identity
44
73
  @clusivity_config_exists_cache = {}
45
74
  end
46
- # rubocop:enable Metrics/AbcSize
75
+ # rubocop:enable Metrics/AbcSize, Metrics/MethodLength
47
76
 
48
77
  def loaded_plugins
49
78
  @loaded_plugins ||= ConfigLoader.loaded_plugins
@@ -262,7 +291,7 @@ module RuboCop
262
291
  end
263
292
 
264
293
  def parser_engine
265
- @parser_engine ||= for_all_cops.fetch('ParserEngine', :parser_whitequark).to_sym
294
+ @parser_engine ||= for_all_cops.fetch('ParserEngine', :default).to_sym
266
295
  end
267
296
 
268
297
  def target_rails_version
@@ -63,7 +63,6 @@ module RuboCop
63
63
  loaded_features = resolver.resolve_requires(path, hash)
64
64
  add_loaded_features(loaded_features)
65
65
 
66
- resolver.override_department_setting_for_cops({}, hash)
67
66
  resolver.resolve_inheritance_from_gems(hash)
68
67
  resolver.resolve_inheritance(path, hash, file, debug?)
69
68
  hash.delete('inherit_from')
@@ -92,8 +91,12 @@ module RuboCop
92
91
  def add_missing_namespaces(path, hash)
93
92
  # Using `hash.each_key` will cause the
94
93
  # `can't add a new key into hash during iteration` error
94
+ obsoletion = ConfigObsoletion.new(hash)
95
+
95
96
  hash_keys = hash.keys
96
97
  hash_keys.each do |key|
98
+ next if obsoletion.deprecated_cop_name?(key)
99
+
97
100
  q = Cop::Registry.qualified_cop_name(key, path)
98
101
  next if q == key
99
102
 
@@ -9,7 +9,8 @@ module RuboCop
9
9
  # @api private
10
10
  class ConfigLoaderResolver # rubocop:disable Metrics/ClassLength
11
11
  def resolve_plugins(rubocop_config, plugins)
12
- return if (plugins = Array(plugins)).empty?
12
+ plugins = Array(plugins) - ConfigLoader.loaded_plugins.map { |plugin| plugin.about.name }
13
+ return if plugins.empty?
13
14
 
14
15
  Plugin.integrate_plugins(rubocop_config, plugins)
15
16
  end
@@ -15,7 +15,7 @@ module RuboCop
15
15
  end
16
16
 
17
17
  def violated?
18
- return false if feature_loaded?
18
+ return false if plugin_loaded?
19
19
 
20
20
  affected_cops.any?
21
21
  end
@@ -38,8 +38,9 @@ module RuboCop
38
38
  end
39
39
  end
40
40
 
41
- def feature_loaded?
42
- config.loaded_features.include?(gem)
41
+ def plugin_loaded?
42
+ # Plugins loaded via `require` are included in `loaded_features`.
43
+ config.loaded_plugins.include?(gem) || config.loaded_features.include?(gem)
43
44
  end
44
45
  end
45
46
  end
@@ -6,17 +6,28 @@ module RuboCop
6
6
  # a cop or moving it to a new department.
7
7
  # @api private
8
8
  class RenamedCop < CopRule
9
- attr_reader :new_name
9
+ attr_reader :new_name, :metadata
10
10
 
11
- def initialize(config, old_name, new_name)
11
+ def initialize(config, old_name, name_or_hash)
12
12
  super(config, old_name)
13
- @new_name = new_name
13
+
14
+ if name_or_hash.is_a?(Hash)
15
+ @metadata = name_or_hash
16
+ @new_name = name_or_hash['new_name']
17
+ else
18
+ @metadata = {}
19
+ @new_name = name_or_hash
20
+ end
14
21
  end
15
22
 
16
23
  def rule_message
17
24
  "The `#{old_name}` cop has been #{verb} to `#{new_name}`."
18
25
  end
19
26
 
27
+ def warning?
28
+ severity == 'warning'
29
+ end
30
+
20
31
  private
21
32
 
22
33
  def moved?
@@ -29,6 +40,10 @@ module RuboCop
29
40
  def verb
30
41
  moved? ? 'moved' : 'renamed'
31
42
  end
43
+
44
+ def severity
45
+ metadata['severity']
46
+ end
32
47
  end
33
48
  end
34
49
  end
@@ -23,9 +23,40 @@ module RuboCop
23
23
  class << self
24
24
  attr_accessor :files
25
25
 
26
+ def global
27
+ @global ||= new(Config.new)
28
+ end
29
+
30
+ def reset!
31
+ @global = nil
32
+ @deprecated_names = {}
33
+ LOAD_RULES_CACHE[rules_cache_key] = nil
34
+ end
35
+
36
+ def rules_cache_key
37
+ files.hash
38
+ end
39
+
26
40
  def legacy_cop_names
27
41
  # Used by DepartmentName#qualified_legacy_cop_name
28
- new(Config.new).rules.select(&:cop_rule?).map(&:old_name)
42
+ global.legacy_cop_names
43
+ end
44
+
45
+ def deprecated_cop_name?(name)
46
+ global.deprecated_cop_name?(name)
47
+ end
48
+
49
+ def deprecated_names_for(cop)
50
+ @deprecated_names ||= {}
51
+ return @deprecated_names[cop] if @deprecated_names.key?(cop)
52
+
53
+ @deprecated_names[cop] = global.rules.filter_map do |rule|
54
+ next unless rule.cop_rule?
55
+ next unless rule.respond_to?(:new_name)
56
+ next unless rule.new_name == cop
57
+
58
+ rule.old_name
59
+ end
29
60
  end
30
61
  end
31
62
 
@@ -45,12 +76,21 @@ module RuboCop
45
76
  raise ValidationError, messages.join("\n")
46
77
  end
47
78
 
79
+ def legacy_cop_names
80
+ # Used by DepartmentName#qualified_legacy_cop_name
81
+ cop_rules.map(&:old_name)
82
+ end
83
+
84
+ def deprecated_cop_name?(name)
85
+ legacy_cop_names.include?(name)
86
+ end
87
+
48
88
  private
49
89
 
50
90
  # Default rules for obsoletions are in config/obsoletion.yml
51
91
  # Additional rules files can be added with `RuboCop::ConfigObsoletion.files << filename`
52
92
  def load_rules # rubocop:disable Metrics/AbcSize
53
- rules = LOAD_RULES_CACHE[self.class.files] ||=
93
+ rules = LOAD_RULES_CACHE[self.class.rules_cache_key] ||=
54
94
  self.class.files.each_with_object({}) do |filename, hash|
55
95
  hash.merge!(YAML.safe_load(File.read(filename)) || {}) do |_key, first, second|
56
96
  case first
@@ -107,5 +147,9 @@ module RuboCop
107
147
  rule.message
108
148
  end
109
149
  end
150
+
151
+ def cop_rules
152
+ rules.select(&:cop_rule?)
153
+ end
110
154
  end
111
155
  end
@@ -118,6 +118,7 @@ module RuboCop
118
118
  invalid_cop_names.each do |name|
119
119
  # There could be a custom cop with this name. If so, don't warn
120
120
  next if Cop::Registry.global.contains_cop_matching?([name])
121
+ next if ConfigObsoletion.deprecated_cop_name?(name)
121
122
 
122
123
  # Special case for inherit_mode, which is a directive that we keep in
123
124
  # the configuration (even though it's not a cop), because it's easier
@@ -50,10 +50,12 @@ module RuboCop
50
50
  }.freeze
51
51
 
52
52
  EXPECT_NO_CORRECTIONS_DESCRIPTION_MAPPING = {
53
- /\A(auto[- ]?)?correct/ => 'does not correct'
53
+ /\A(auto[- ]?)?corrects?/ => 'does not correct',
54
+ /\band (auto[- ]?)?corrects/ => 'but does not correct'
54
55
  }.freeze
55
56
 
56
57
  EXPECT_CORRECTION_DESCRIPTION_MAPPING = {
58
+ /\bbut (does not|doesn't) (auto[- ]?)?correct/ => 'and autocorrects',
57
59
  /\b(does not|doesn't) (auto[- ]?)?correct/ => 'autocorrects'
58
60
  }.freeze
59
61
 
@@ -27,7 +27,7 @@ module RuboCop
27
27
  MSG = 'Replace `%<names>s` in node pattern union with `%<replacement>s`.'
28
28
  RESTRICT_ON_SEND = %i[def_node_matcher def_node_search].freeze
29
29
  NODE_GROUPS = {
30
- any_block: %i[block numblock],
30
+ any_block: %i[block numblock itblock],
31
31
  argument: %i[arg optarg restarg kwarg kwoptarg kwrestarg blockarg forward_arg shadowarg],
32
32
  boolean: %i[true false],
33
33
  call: %i[send csend],
@@ -0,0 +1,91 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module InternalAffairs
6
+ # Checks that node types are checked against their group when all types of a
7
+ # group are checked.
8
+ #
9
+ # @example
10
+ # # bad
11
+ # node.type?(:irange, :erange)
12
+ #
13
+ # # good
14
+ # node.range_type?
15
+ #
16
+ # # bad
17
+ # node.type?(:irange, :erange, :send, :csend)
18
+ #
19
+ # # good
20
+ # node.type?(:range, :call)
21
+ #
22
+ class NodeTypeGroup < Base
23
+ extend AutoCorrector
24
+ include RangeHelp
25
+
26
+ MSG = 'Use `:%<group>s` instead of individually listing group types.'
27
+
28
+ RESTRICT_ON_SEND = %i[type? each_ancestor each_child_node each_descendant each_node].freeze
29
+
30
+ def on_send(node)
31
+ return unless node.receiver
32
+
33
+ symbol_args = node.arguments.select(&:sym_type?)
34
+ return if symbol_args.none?
35
+
36
+ NodePatternGroups::NODE_GROUPS.each do |group_name, group_types|
37
+ next unless group_satisfied?(group_types, symbol_args)
38
+
39
+ offense_range = arguments_range(node)
40
+ add_offense(offense_range, message: format(MSG, group: group_name)) do |corrector|
41
+ autocorrect(corrector, node, symbol_args, group_name, group_types)
42
+ end
43
+ end
44
+ end
45
+ alias on_csend on_send
46
+
47
+ private
48
+
49
+ def arguments_range(node)
50
+ range_between(
51
+ node.first_argument.source_range.begin_pos,
52
+ node.last_argument.source_range.end_pos
53
+ )
54
+ end
55
+
56
+ def group_satisfied?(group_types, symbol_args)
57
+ group_types.all? { |type| symbol_args.any? { |arg| arg.value == type } }
58
+ end
59
+
60
+ def autocorrect(corrector, node, symbol_args, group_name, group_types)
61
+ if node.method?(:type?) && node.arguments.count == group_types.count
62
+ autocorrect_to_explicit_predicate(corrector, node, group_name)
63
+ else
64
+ autocorrect_keep_method(corrector, symbol_args, group_name, group_types)
65
+ end
66
+ end
67
+
68
+ def autocorrect_to_explicit_predicate(corrector, node, group_name)
69
+ corrector.replace(node.selector, "#{group_name}_type?")
70
+ corrector.remove(arguments_range(node))
71
+ end
72
+
73
+ def autocorrect_keep_method(corrector, symbol_args, group_name, group_types)
74
+ first_replaced = false
75
+ symbol_args.each do |arg|
76
+ next unless group_types.include?(arg.value)
77
+
78
+ if first_replaced
79
+ range = range_with_surrounding_space(arg.source_range)
80
+ range = range_with_surrounding_comma(range, :left)
81
+ corrector.remove(range)
82
+ else
83
+ first_replaced = true
84
+ corrector.replace(arg, ":#{group_name}")
85
+ end
86
+ end
87
+ end
88
+ end
89
+ end
90
+ end
91
+ end
@@ -38,23 +38,24 @@ module RuboCop
38
38
 
39
39
  describe = find_describe_method_node(node)
40
40
 
41
- unless (exist_config = describe.last_argument.source == ':config')
42
- additional_message = ' and specify `:config` in `describe`'
43
- end
41
+ should_append_config = describe && describe.last_argument.source != ':config'
42
+ additional_message = ' and specify `:config` in `describe`' if should_append_config
44
43
 
45
44
  message = format(MSG, additional_message: additional_message)
46
45
 
47
46
  add_offense(node, message: message) do |corrector|
48
47
  corrector.remove(range_by_whole_lines(node.source_range, include_final_newline: true))
49
48
 
50
- corrector.insert_after(describe.last_argument, ', :config') unless exist_config
49
+ corrector.insert_after(describe.last_argument, ', :config') if should_append_config
51
50
  end
52
51
  end
53
52
 
54
53
  private
55
54
 
56
55
  def find_describe_method_node(block_node)
57
- block_node.ancestors.find { |node| node.block_type? && node.method?(:describe) }.send_node
56
+ block_node.ancestors.find do |node|
57
+ node.block_type? && node.method?(:describe)
58
+ end&.send_node
58
59
  end
59
60
  end
60
61
  end
@@ -17,6 +17,7 @@ require_relative 'internal_affairs/node_destructuring'
17
17
  require_relative 'internal_affairs/node_first_or_last_argument'
18
18
  require_relative 'internal_affairs/node_matcher_directive'
19
19
  require_relative 'internal_affairs/node_pattern_groups'
20
+ require_relative 'internal_affairs/node_type_group'
20
21
  require_relative 'internal_affairs/node_type_multiple_predicates'
21
22
  require_relative 'internal_affairs/node_type_predicate'
22
23
  require_relative 'internal_affairs/numblock_handler'
@@ -87,6 +87,7 @@ module RuboCop
87
87
  end
88
88
 
89
89
  alias on_numblock on_block
90
+ alias on_itblock on_block
90
91
 
91
92
  def style_parameter_name
92
93
  'EnforcedStyleAlignWith'
@@ -43,6 +43,7 @@ module RuboCop
43
43
  end
44
44
 
45
45
  alias on_numblock on_block
46
+ alias on_itblock on_block
46
47
 
47
48
  private
48
49
 
@@ -92,7 +92,7 @@ module RuboCop
92
92
  case parent.type
93
93
  when :def, :defs then base_for_method_definition(parent)
94
94
  when :kwbegin then parent.loc.begin
95
- when :block, :numblock
95
+ when :block, :numblock, :itblock
96
96
  assignment_node = assignment_node(parent)
97
97
  if same_line?(parent, assignment_node)
98
98
  assignment_node.source_range