rubocop 1.71.2 → 1.74.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 (118) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +3 -3
  3. data/config/default.yml +86 -15
  4. data/config/internal_affairs.yml +20 -0
  5. data/lib/rubocop/cli/command/suggest_extensions.rb +7 -1
  6. data/lib/rubocop/comment_config.rb +1 -1
  7. data/lib/rubocop/config.rb +4 -0
  8. data/lib/rubocop/config_loader.rb +44 -9
  9. data/lib/rubocop/config_loader_resolver.rb +24 -9
  10. data/lib/rubocop/config_obsoletion/extracted_cop.rb +4 -3
  11. data/lib/rubocop/config_obsoletion.rb +1 -1
  12. data/lib/rubocop/config_validator.rb +1 -1
  13. data/lib/rubocop/cop/internal_affairs/example_description.rb +7 -3
  14. data/lib/rubocop/cop/internal_affairs/location_exists.rb +116 -0
  15. data/lib/rubocop/cop/internal_affairs/node_pattern_groups/ast_walker.rb +1 -1
  16. data/lib/rubocop/cop/internal_affairs/node_type_group.rb +91 -0
  17. data/lib/rubocop/cop/internal_affairs/plugin.rb +33 -0
  18. data/lib/rubocop/cop/internal_affairs/undefined_config.rb +7 -1
  19. data/lib/rubocop/cop/internal_affairs.rb +2 -16
  20. data/lib/rubocop/cop/layout/block_alignment.rb +2 -0
  21. data/lib/rubocop/cop/layout/closing_parenthesis_indentation.rb +4 -4
  22. data/lib/rubocop/cop/layout/else_alignment.rb +1 -1
  23. data/lib/rubocop/cop/layout/empty_line_after_guard_clause.rb +1 -1
  24. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +26 -1
  25. data/lib/rubocop/cop/layout/empty_lines_around_method_body.rb +22 -2
  26. data/lib/rubocop/cop/layout/heredoc_argument_closing_parenthesis.rb +1 -1
  27. data/lib/rubocop/cop/layout/line_length.rb +3 -3
  28. data/lib/rubocop/cop/layout/multiline_method_call_indentation.rb +2 -2
  29. data/lib/rubocop/cop/layout/space_around_method_call_operator.rb +1 -1
  30. data/lib/rubocop/cop/lint/cop_directive_syntax.rb +84 -0
  31. data/lib/rubocop/cop/lint/duplicate_methods.rb +0 -14
  32. data/lib/rubocop/cop/lint/empty_conditional_body.rb +14 -64
  33. data/lib/rubocop/cop/lint/erb_new_arguments.rb +0 -6
  34. data/lib/rubocop/cop/lint/float_comparison.rb +1 -6
  35. data/lib/rubocop/cop/lint/format_parameter_mismatch.rb +1 -1
  36. data/lib/rubocop/cop/lint/literal_as_condition.rb +103 -9
  37. data/lib/rubocop/cop/lint/mixed_case_range.rb +2 -2
  38. data/lib/rubocop/cop/lint/non_local_exit_from_iterator.rb +2 -2
  39. data/lib/rubocop/cop/lint/redundant_require_statement.rb +0 -21
  40. data/lib/rubocop/cop/lint/redundant_type_conversion.rb +258 -0
  41. data/lib/rubocop/cop/lint/return_in_void_context.rb +4 -11
  42. data/lib/rubocop/cop/lint/shared_mutable_default.rb +12 -1
  43. data/lib/rubocop/cop/lint/suppressed_exception_in_number_conversion.rb +111 -0
  44. data/lib/rubocop/cop/lint/useless_constant_scoping.rb +71 -0
  45. data/lib/rubocop/cop/lint/void.rb +6 -0
  46. data/lib/rubocop/cop/metrics/utils/repeated_attribute_discount.rb +7 -7
  47. data/lib/rubocop/cop/mixin/alignment.rb +2 -2
  48. data/lib/rubocop/cop/mixin/allowed_pattern.rb +4 -4
  49. data/lib/rubocop/cop/mixin/check_single_line_suitability.rb +1 -1
  50. data/lib/rubocop/cop/mixin/comments_help.rb +1 -1
  51. data/lib/rubocop/cop/mixin/hash_shorthand_syntax.rb +18 -18
  52. data/lib/rubocop/cop/mixin/hash_subset.rb +19 -4
  53. data/lib/rubocop/cop/mixin/hash_transform_method.rb +74 -74
  54. data/lib/rubocop/cop/mixin/percent_literal.rb +1 -1
  55. data/lib/rubocop/cop/mixin/range_help.rb +15 -3
  56. data/lib/rubocop/cop/mixin/string_help.rb +1 -1
  57. data/lib/rubocop/cop/mixin/target_ruby_version.rb +1 -1
  58. data/lib/rubocop/cop/mixin/trailing_comma.rb +12 -0
  59. data/lib/rubocop/cop/naming/block_forwarding.rb +3 -3
  60. data/lib/rubocop/cop/naming/predicate_name.rb +44 -0
  61. data/lib/rubocop/cop/naming/variable_name.rb +64 -6
  62. data/lib/rubocop/cop/style/accessor_grouping.rb +19 -5
  63. data/lib/rubocop/cop/style/arguments_forwarding.rb +3 -3
  64. data/lib/rubocop/cop/style/class_and_module_children.rb +29 -7
  65. data/lib/rubocop/cop/style/commented_keyword.rb +10 -3
  66. data/lib/rubocop/cop/style/comparable_between.rb +75 -0
  67. data/lib/rubocop/cop/style/double_negation.rb +1 -1
  68. data/lib/rubocop/cop/style/endless_method.rb +163 -18
  69. data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -7
  70. data/lib/rubocop/cop/style/exponential_notation.rb +2 -2
  71. data/lib/rubocop/cop/style/format_string_token.rb +38 -11
  72. data/lib/rubocop/cop/style/if_unless_modifier.rb +2 -2
  73. data/lib/rubocop/cop/style/inverse_methods.rb +8 -5
  74. data/lib/rubocop/cop/style/keyword_parameters_order.rb +13 -7
  75. data/lib/rubocop/cop/style/line_end_concatenation.rb +10 -4
  76. data/lib/rubocop/cop/style/method_call_with_args_parentheses/omit_parentheses.rb +3 -3
  77. data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +1 -1
  78. data/lib/rubocop/cop/style/multiline_block_chain.rb +1 -1
  79. data/lib/rubocop/cop/style/multiline_method_signature.rb +1 -9
  80. data/lib/rubocop/cop/style/redundant_condition.rb +45 -0
  81. data/lib/rubocop/cop/style/redundant_current_directory_in_path.rb +14 -4
  82. data/lib/rubocop/cop/style/redundant_format.rb +250 -0
  83. data/lib/rubocop/cop/style/redundant_freeze.rb +1 -1
  84. data/lib/rubocop/cop/style/redundant_parentheses.rb +18 -4
  85. data/lib/rubocop/cop/style/redundant_self_assignment.rb +1 -1
  86. data/lib/rubocop/cop/style/rescue_modifier.rb +3 -0
  87. data/lib/rubocop/cop/style/single_line_methods.rb +3 -3
  88. data/lib/rubocop/cop/style/sole_nested_conditional.rb +0 -6
  89. data/lib/rubocop/cop/style/string_concatenation.rb +1 -1
  90. data/lib/rubocop/cop/style/trailing_comma_in_array_literal.rb +47 -6
  91. data/lib/rubocop/cop/style/trailing_comma_in_hash_literal.rb +48 -6
  92. data/lib/rubocop/cop/style/trivial_accessors.rb +1 -1
  93. data/lib/rubocop/cop/util.rb +1 -1
  94. data/lib/rubocop/cop/utils/format_string.rb +10 -5
  95. data/lib/rubocop/cops_documentation_generator.rb +12 -1
  96. data/lib/rubocop/directive_comment.rb +36 -3
  97. data/lib/rubocop/ext/regexp_node.rb +0 -1
  98. data/lib/rubocop/lsp/runtime.rb +2 -0
  99. data/lib/rubocop/lsp/server.rb +0 -2
  100. data/lib/rubocop/options.rb +26 -11
  101. data/lib/rubocop/path_util.rb +4 -0
  102. data/lib/rubocop/plugin/configuration_integrator.rb +143 -0
  103. data/lib/rubocop/plugin/load_error.rb +26 -0
  104. data/lib/rubocop/plugin/loader.rb +100 -0
  105. data/lib/rubocop/plugin/not_supported_error.rb +29 -0
  106. data/lib/rubocop/plugin.rb +46 -0
  107. data/lib/rubocop/rake_task.rb +4 -1
  108. data/lib/rubocop/rspec/cop_helper.rb +9 -0
  109. data/lib/rubocop/rspec/shared_contexts.rb +15 -0
  110. data/lib/rubocop/rspec/support.rb +1 -0
  111. data/lib/rubocop/server/cache.rb +35 -2
  112. data/lib/rubocop/server/cli.rb +2 -2
  113. data/lib/rubocop/version.rb +17 -2
  114. data/lib/rubocop.rb +6 -1
  115. data/lib/ruby_lsp/rubocop/addon.rb +7 -10
  116. data/lib/ruby_lsp/rubocop/{wraps_built_in_lsp_runtime.rb → runtime_adapter.rb} +5 -8
  117. metadata +37 -10
  118. data/lib/rubocop/cop/utils/regexp_ranges.rb +0 -113
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 8d7c6408a777042eced58bec2bc889312b24d8701f5db99da3b471088f589d6f
4
- data.tar.gz: 27423e682d55f04840a3bf3d5e0301556e4a1c419857e3422ced38ec423c0003
3
+ metadata.gz: 440bee0e28f294bab2eba4c11e9c681342917c69ad4c00e36eceb88743767c29
4
+ data.tar.gz: 68e9e209a22e891a38b677f97344d2d10e9a8913dd999751472f301deed43b31
5
5
  SHA512:
6
- metadata.gz: 8729a8ea3adae11fcb2501bfea338821a8f9894209ea6fb482b62ebfae87019e07ded7bf02d3371c18726b28c2491c79952e7b86c193388a045f886a833b427c
7
- data.tar.gz: 44251afbbbc2aa7c121033ea05f25e1b3447afba7487d04a81f9463347ba28bcea05bdc2e434248f15cd235d9e452cb3f9f378f94adb5507eeb6d5ce58ba4119
6
+ metadata.gz: 70db27a5b7a0e00696672a5b4fcd4e3bf35aa40c85ef65ccb630914ab110a5e5b657927aa1b1c8637be76b7041a90e2ad3c4ed188110266c048d0c4d6a4be0b4
7
+ data.tar.gz: e5599d30a4f776d85d202acad2ac03c7e4568e64a1c4038cb87b8ee3372b2428882924a91ea49853c98cf0614280d002918363eef41fe0990dc8b1895d8e9776
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.71', require: false
55
+ gem 'rubocop', '~> 1.74', require: false
56
56
  ```
57
57
 
58
58
  See [our versioning policy](https://docs.rubocop.org/rubocop/versioning.html) for further details.
@@ -241,9 +241,9 @@ Become a sponsor and get your logo on our README on GitHub with a link to your s
241
241
  <a href="https://opencollective.com/rubocop/organization/28/website" target="_blank"><img src="https://opencollective.com/rubocop/organization/28/avatar.svg"></a>
242
242
  <a href="https://opencollective.com/rubocop/organization/29/website" target="_blank"><img src="https://opencollective.com/rubocop/organization/29/avatar.svg"></a>
243
243
 
244
- ## Changelog
244
+ ## Release Notes
245
245
 
246
- RuboCop's changelog is available [here](CHANGELOG.md).
246
+ RuboCop's release notes are available [here](https://github.com/rubocop/rubocop/releases).
247
247
 
248
248
  ## Copyright
249
249
 
data/config/default.yml CHANGED
@@ -1685,6 +1685,11 @@ Lint/ConstantResolution:
1685
1685
  # Restrict this cop from only looking at certain names
1686
1686
  Ignore: []
1687
1687
 
1688
+ Lint/CopDirectiveSyntax:
1689
+ Description: 'Checks that `# rubocop:` directives are strictly formatted.'
1690
+ Enabled: pending
1691
+ VersionAdded: '1.72'
1692
+
1688
1693
  Lint/Debugger:
1689
1694
  Description: 'Check for debugger calls.'
1690
1695
  Enabled: true
@@ -1882,10 +1887,9 @@ Lint/EmptyConditionalBody:
1882
1887
  Description: 'Checks for the presence of `if`, `elsif` and `unless` branches without a body.'
1883
1888
  Enabled: true
1884
1889
  AutoCorrect: contextual
1885
- SafeAutoCorrect: false
1886
1890
  AllowComments: true
1887
1891
  VersionAdded: '0.89'
1888
- VersionChanged: '1.61'
1892
+ VersionChanged: '1.73'
1889
1893
 
1890
1894
  Lint/EmptyEnsure:
1891
1895
  Description: 'Checks for empty ensure block.'
@@ -2041,6 +2045,7 @@ Lint/LambdaWithoutLiteralBlock:
2041
2045
  Lint/LiteralAsCondition:
2042
2046
  Description: 'Checks of literals used in conditions.'
2043
2047
  Enabled: true
2048
+ AutoCorrect: contextual
2044
2049
  VersionAdded: '0.51'
2045
2050
 
2046
2051
  Lint/LiteralAssignmentInCondition:
@@ -2254,9 +2259,8 @@ Lint/RedundantRegexpQuantifiers:
2254
2259
  Lint/RedundantRequireStatement:
2255
2260
  Description: 'Checks for unnecessary `require` statement.'
2256
2261
  Enabled: true
2257
- SafeAutoCorrect: false
2258
2262
  VersionAdded: '0.76'
2259
- VersionChanged: '1.57'
2263
+ VersionChanged: '1.73'
2260
2264
 
2261
2265
  Lint/RedundantSafeNavigation:
2262
2266
  Description: 'Checks for redundant safe navigation calls.'
@@ -2285,6 +2289,11 @@ Lint/RedundantStringCoercion:
2285
2289
  VersionAdded: '0.19'
2286
2290
  VersionChanged: '0.77'
2287
2291
 
2292
+ Lint/RedundantTypeConversion:
2293
+ Description: 'Checks for redundantly converting a literal to the same type.'
2294
+ Enabled: pending
2295
+ VersionAdded: '1.72'
2296
+
2288
2297
  Lint/RedundantWithIndex:
2289
2298
  Description: 'Checks for redundant `with_index`.'
2290
2299
  Enabled: true
@@ -2433,6 +2442,12 @@ Lint/SuppressedException:
2433
2442
  VersionAdded: '0.9'
2434
2443
  VersionChanged: '1.12'
2435
2444
 
2445
+ Lint/SuppressedExceptionInNumberConversion:
2446
+ Description: 'Checks for cases where exceptions unrelated to the numeric constructors may be unintentionally swallowed.'
2447
+ Enabled: pending
2448
+ SafeAutoCorrect: false
2449
+ VersionAdded: '1.72'
2450
+
2436
2451
  Lint/SymbolConversion:
2437
2452
  Description: 'Checks for unnecessary symbol conversions.'
2438
2453
  Enabled: pending
@@ -2588,6 +2603,11 @@ Lint/UselessAssignment:
2588
2603
  VersionAdded: '0.11'
2589
2604
  VersionChanged: '1.66'
2590
2605
 
2606
+ Lint/UselessConstantScoping:
2607
+ Description: 'Checks for useless constant scoping.'
2608
+ Enabled: pending
2609
+ VersionAdded: '1.72'
2610
+
2591
2611
  Lint/UselessDefined:
2592
2612
  Description: 'Checks for calls to `defined?` with strings and symbols. The result of such a call will always be truthy.'
2593
2613
  Enabled: pending
@@ -3041,6 +3061,8 @@ Naming/PredicateName:
3041
3061
  MethodDefinitionMacros:
3042
3062
  - define_method
3043
3063
  - define_singleton_method
3064
+ # Use Sorbet's T::Boolean return type to detect predicate methods.
3065
+ UseSorbetSigs: false
3044
3066
  # Exclude Rspec specs because there is a strong convention to write spec
3045
3067
  # helpers in the form of `have_something` or `be_something`.
3046
3068
  Exclude:
@@ -3058,13 +3080,15 @@ Naming/VariableName:
3058
3080
  StyleGuide: '#snake-case-symbols-methods-vars'
3059
3081
  Enabled: true
3060
3082
  VersionAdded: '0.50'
3061
- VersionChanged: '1.8'
3083
+ VersionChanged: '1.73'
3062
3084
  EnforcedStyle: snake_case
3063
3085
  SupportedStyles:
3064
3086
  - snake_case
3065
3087
  - camelCase
3066
3088
  AllowedIdentifiers: []
3067
3089
  AllowedPatterns: []
3090
+ ForbiddenIdentifiers: []
3091
+ ForbiddenPatterns: []
3068
3092
 
3069
3093
  Naming/VariableNumber:
3070
3094
  Description: 'Use the configured style when numbering symbols, methods and variables.'
@@ -3080,6 +3104,8 @@ Naming/VariableNumber:
3080
3104
  CheckMethodNames: true
3081
3105
  CheckSymbols: true
3082
3106
  AllowedIdentifiers:
3107
+ - TLS1_1 # OpenSSL::SSL::TLS1_1_VERSION
3108
+ - TLS1_2 # OpenSSL::SSL::TLS1_2_VERSION
3083
3109
  - capture3 # Open3.capture3
3084
3110
  - iso8601 # Time#iso8601
3085
3111
  - rfc1123_date # CGI.rfc1123_date
@@ -3475,6 +3501,7 @@ Style/ClassAndModuleChildren:
3475
3501
  SafeAutoCorrect: false
3476
3502
  Enabled: true
3477
3503
  VersionAdded: '0.19'
3504
+ VersionChanged: '1.74'
3478
3505
  #
3479
3506
  # Basically there are two different styles:
3480
3507
  #
@@ -3490,7 +3517,21 @@ Style/ClassAndModuleChildren:
3490
3517
  #
3491
3518
  # The compact style is only forced, for classes or modules with one child.
3492
3519
  EnforcedStyle: nested
3493
- SupportedStyles:
3520
+ SupportedStyles: &supported_styles
3521
+ - nested
3522
+ - compact
3523
+ # Configure classes separately, if desired. If not set, or set to `nil`,
3524
+ # the `EnforcedStyle` value will be used.
3525
+ EnforcedStyleForClasses: ~
3526
+ SupportedStylesForClasses:
3527
+ - ~
3528
+ - nested
3529
+ - compact
3530
+ # Configure modules separately, if desired. If not set, or set to `nil`,
3531
+ # the `EnforcedStyle` value will be used.
3532
+ EnforcedStyleForModules: ~
3533
+ SupportedStylesForModules:
3534
+ - ~
3494
3535
  - nested
3495
3536
  - compact
3496
3537
 
@@ -3643,6 +3684,12 @@ Style/CommentedKeyword:
3643
3684
  VersionAdded: '0.51'
3644
3685
  VersionChanged: '1.19'
3645
3686
 
3687
+ Style/ComparableBetween:
3688
+ Description: 'Enforces the use of `Comparable#between?` instead of logical comparison.'
3689
+ Enabled: pending
3690
+ VersionAdded: '1.74'
3691
+ StyleGuide: '#ranges-or-between'
3692
+
3646
3693
  Style/ComparableClamp:
3647
3694
  Description: 'Enforces the use of `Comparable#clamp` instead of comparison by minimum and maximum.'
3648
3695
  Enabled: pending
@@ -3899,6 +3946,8 @@ Style/EndlessMethod:
3899
3946
  - allow_single_line
3900
3947
  - allow_always
3901
3948
  - disallow
3949
+ - require_single_line
3950
+ - require_always
3902
3951
 
3903
3952
  Style/EnvHome:
3904
3953
  Description: "Checks for consistent usage of `ENV['HOME']`."
@@ -4043,8 +4092,14 @@ Style/FormatStringToken:
4043
4092
  # style token in a format string to be allowed when enforced style is not
4044
4093
  # `unannotated`.
4045
4094
  MaxUnannotatedPlaceholdersAllowed: 1
4095
+ # The mode the cop operates in. Two values are allowed:
4096
+ # * aggressive (default): all strings are considered
4097
+ # * conservative:
4098
+ # only register offenses for strings given to `printf`, `sprintf`,
4099
+ # format` and `%` methods. Other strings are not considered.
4100
+ Mode: aggressive
4046
4101
  VersionAdded: '0.49'
4047
- VersionChanged: '1.0'
4102
+ VersionChanged: '1.74'
4048
4103
  AllowedMethods: []
4049
4104
  AllowedPatterns: []
4050
4105
 
@@ -5081,6 +5136,9 @@ Style/RedundantCondition:
5081
5136
  Description: 'Checks for unnecessary conditional expressions.'
5082
5137
  Enabled: true
5083
5138
  VersionAdded: '0.76'
5139
+ VersionChanged: '1.73'
5140
+ AllowedMethods:
5141
+ - nonzero?
5084
5142
 
5085
5143
  Style/RedundantConditional:
5086
5144
  Description: "Don't return true/false from a conditional."
@@ -5145,6 +5203,13 @@ Style/RedundantFilterChain:
5145
5203
  VersionAdded: '1.52'
5146
5204
  VersionChanged: '1.57'
5147
5205
 
5206
+ Style/RedundantFormat:
5207
+ Description: 'Checks for usages of `Kernel#format` or `Kernel#sprintf` with only a single argument.'
5208
+ Enabled: pending
5209
+ SafeAutoCorrect: false
5210
+ VersionAdded: '1.72'
5211
+ VersionChanged: '1.72'
5212
+
5148
5213
  Style/RedundantFreeze:
5149
5214
  Description: "Checks usages of Object#freeze on immutable objects."
5150
5215
  Enabled: true
@@ -5671,14 +5736,17 @@ Style/TrailingCommaInArrayLiteral:
5671
5736
  StyleGuide: '#no-trailing-array-commas'
5672
5737
  Enabled: true
5673
5738
  VersionAdded: '0.53'
5674
- # If `comma`, the cop requires a comma after the last item in an array,
5675
- # but only when each item is on its own line.
5676
- # If `consistent_comma`, the cop requires a comma after the last item of all
5677
- # non-empty, multiline array literals.
5739
+ # If `comma`, the cop requires a comma after the last item in an array, but only when each item is
5740
+ # on its own line.
5741
+ # If `consistent_comma`, the cop requires a comma after the last item of all non-empty, multiline
5742
+ # array literals.
5743
+ # If `diff_comma`, the cop requires a comma after the last item of all non-empty, multiline array
5744
+ # literals, but only when that last item immediately precedes a newline.
5678
5745
  EnforcedStyleForMultiline: no_comma
5679
5746
  SupportedStylesForMultiline:
5680
5747
  - comma
5681
5748
  - consistent_comma
5749
+ - diff_comma
5682
5750
  - no_comma
5683
5751
 
5684
5752
  Style/TrailingCommaInBlockArgs:
@@ -5690,14 +5758,17 @@ Style/TrailingCommaInBlockArgs:
5690
5758
  Style/TrailingCommaInHashLiteral:
5691
5759
  Description: 'Checks for trailing comma in hash literals.'
5692
5760
  Enabled: true
5693
- # If `comma`, the cop requires a comma after the last item in a hash,
5694
- # but only when each item is on its own line.
5695
- # If `consistent_comma`, the cop requires a comma after the last item of all
5696
- # non-empty, multiline hash literals.
5761
+ # If `comma`, the cop requires a comma after the last item in a hash, but only when each item is
5762
+ # on its own line.
5763
+ # If `consistent_comma`, the cop requires a comma after the last item of all non-empty, multiline
5764
+ # hash literals.
5765
+ # If `diff_comma`, the cop requires a comma after the last item of all non-empty, multiline hash
5766
+ # literals, but only when that last item immediately precedes a newline.
5697
5767
  EnforcedStyleForMultiline: no_comma
5698
5768
  SupportedStylesForMultiline:
5699
5769
  - comma
5700
5770
  - consistent_comma
5771
+ - diff_comma
5701
5772
  - no_comma
5702
5773
  VersionAdded: '0.53'
5703
5774
 
@@ -6,6 +6,26 @@ InternalAffairs/CopDescription:
6
6
  Include:
7
7
  - 'lib/rubocop/cop/**/*.rb'
8
8
 
9
+ InternalAffairs/ExampleHeredocDelimiter:
10
+ Include:
11
+ - 'spec/rubocop/cop/**/*.rb'
12
+
13
+ InternalAffairs/ExampleDescription:
14
+ Include:
15
+ - 'spec/rubocop/cop/**/*.rb'
16
+
17
+ InternalAffairs/NodeTypeGroup:
18
+ Include:
19
+ - 'lib/rubocop/cop/**/*.rb'
20
+
21
+ InternalAffairs/OnSendWithoutOnCSend:
22
+ Include:
23
+ - 'lib/rubocop/cop/**/*.rb'
24
+
25
+ InternalAffairs/UndefinedConfig:
26
+ Include:
27
+ - 'lib/rubocop/cop/**/*.rb'
28
+
9
29
  InternalAffairs/UselessMessageAssertion:
10
30
  Include:
11
31
  - '**/*_spec.rb'
@@ -97,7 +97,13 @@ module RuboCop
97
97
  end
98
98
 
99
99
  def loaded_extensions
100
- @config_store.for_pwd.loaded_features.to_a
100
+ rubocop_config = @config_store.for_pwd
101
+
102
+ plugin_names = rubocop_config.loaded_plugins.map do |plugin|
103
+ plugin.about.name
104
+ end
105
+
106
+ plugin_names + rubocop_config.loaded_features.to_a
101
107
  end
102
108
 
103
109
  def installed_and_not_loaded_extensions
@@ -205,7 +205,7 @@ module RuboCop
205
205
  directive.cop_names.each do |name|
206
206
  if directive.disabled?
207
207
  names[name] += 1
208
- elsif (names[name]).positive?
208
+ elsif names[name].positive?
209
209
  names[name] -= 1
210
210
  else
211
211
  extras[directive.comment] << name
@@ -45,6 +45,10 @@ module RuboCop
45
45
  end
46
46
  # rubocop:enable Metrics/AbcSize
47
47
 
48
+ def loaded_plugins
49
+ @loaded_plugins ||= ConfigLoader.loaded_plugins
50
+ end
51
+
48
52
  def loaded_features
49
53
  @loaded_features ||= ConfigLoader.loaded_features
50
54
  end
@@ -33,13 +33,14 @@ module RuboCop
33
33
  attr_accessor :debug, :ignore_parent_exclusion, :disable_pending_cops, :enable_pending_cops,
34
34
  :ignore_unrecognized_cops
35
35
  attr_writer :default_configuration
36
- attr_reader :loaded_features
36
+ attr_reader :loaded_plugins, :loaded_features
37
37
 
38
38
  alias debug? debug
39
39
  alias ignore_parent_exclusion? ignore_parent_exclusion
40
40
 
41
41
  def clear_options
42
42
  @debug = nil
43
+ @loaded_plugins = Set.new
43
44
  @loaded_features = Set.new
44
45
  @disable_pending_cops = nil
45
46
  @enable_pending_cops = nil
@@ -48,15 +49,20 @@ module RuboCop
48
49
  FileFinder.root_level = nil
49
50
  end
50
51
 
52
+ # rubocop:disable Metrics/AbcSize
51
53
  def load_file(file, check: true)
52
54
  path = file_path(file)
53
55
 
54
56
  hash = load_yaml_configuration(path)
55
57
 
58
+ rubocop_config = Config.create(hash, path, check: false)
59
+ plugins = hash.delete('plugins')
60
+ loaded_plugins = resolver.resolve_plugins(rubocop_config, plugins)
61
+ add_loaded_plugins(loaded_plugins)
62
+
56
63
  loaded_features = resolver.resolve_requires(path, hash)
57
64
  add_loaded_features(loaded_features)
58
65
 
59
- resolver.override_department_setting_for_cops({}, hash)
60
66
  resolver.resolve_inheritance_from_gems(hash)
61
67
  resolver.resolve_inheritance(path, hash, file, debug?)
62
68
  hash.delete('inherit_from')
@@ -67,6 +73,7 @@ module RuboCop
67
73
 
68
74
  Config.create(hash, path, check: check)
69
75
  end
76
+ # rubocop:enable Metrics/AbcSize
70
77
 
71
78
  def load_yaml_configuration(absolute_path)
72
79
  file_contents = read_file(absolute_path)
@@ -155,14 +162,35 @@ module RuboCop
155
162
  end
156
163
  end
157
164
 
158
- # @api private
159
- def inject_defaults!(project_root)
160
- path = File.join(project_root, 'config', 'default.yml')
161
- config = load_file(path)
162
- new_config = ConfigLoader.merge_with_default(config, path)
163
- puts "configuration from #{path}" if debug?
164
- @default_configuration = new_config
165
+ # This API is primarily intended for testing and documenting plugins.
166
+ # When testing a plugin using `rubocop/rspec/support`, the plugin is loaded automatically,
167
+ # so this API is usually not needed. It is intended to be used only when implementing tests
168
+ # that do not use `rubocop/rspec/support`.
169
+ # rubocop:disable Metrics/MethodLength
170
+ def inject_defaults!(config_yml_path)
171
+ if Pathname(config_yml_path).directory?
172
+ # TODO: Since the warning noise is expected to be high until some time after the release,
173
+ # warnings will only be issued when `RUBYOPT=-w` is specified.
174
+ # To proceed step by step, the next step is to remove `$VERBOSE` and always issue warning.
175
+ # Eventually, `project_root` will no longer be accepted.
176
+ if $VERBOSE
177
+ warn Rainbow(<<~MESSAGE).yellow, uplevel: 1
178
+ Use config YAML file path instead of project root directory.
179
+ e.g., `path/to/config/default.yml`
180
+ MESSAGE
181
+ end
182
+ # NOTE: For compatibility.
183
+ project_root = config_yml_path
184
+ path = File.join(project_root, 'config', 'default.yml')
185
+ config = load_file(path)
186
+ else
187
+ hash = ConfigLoader.load_yaml_configuration(config_yml_path.to_s)
188
+ config = Config.new(hash, config_yml_path).tap(&:make_excludes_absolute)
189
+ end
190
+
191
+ @default_configuration = ConfigLoader.merge_with_default(config, path)
165
192
  end
193
+ # rubocop:enable Metrics/MethodLength
166
194
 
167
195
  # Returns the path RuboCop inferred as the root of the project. No file
168
196
  # searches will go past this directory.
@@ -196,6 +224,13 @@ module RuboCop
196
224
  resolver.merge_with_default(config, config_file, unset_nil: unset_nil)
197
225
  end
198
226
 
227
+ # @api private
228
+ # Used to add plugins that were required inside a config or from
229
+ # the CLI using `--plugin`.
230
+ def add_loaded_plugins(loaded_plugins)
231
+ @loaded_plugins.merge(Array(loaded_plugins))
232
+ end
233
+
199
234
  # @api private
200
235
  # Used to add features that were required inside a config or from
201
236
  # the CLI using `--require`.
@@ -2,16 +2,35 @@
2
2
 
3
3
  require 'pathname'
4
4
  require 'yaml'
5
+ require_relative 'plugin'
5
6
 
6
7
  module RuboCop
7
8
  # A help class for ConfigLoader that handles configuration resolution.
8
9
  # @api private
9
10
  class ConfigLoaderResolver # rubocop:disable Metrics/ClassLength
11
+ def resolve_plugins(rubocop_config, plugins)
12
+ plugins = Array(plugins) - ConfigLoader.loaded_plugins.map { |plugin| plugin.about.name }
13
+ return if plugins.empty?
14
+
15
+ Plugin.integrate_plugins(rubocop_config, plugins)
16
+ end
17
+
10
18
  def resolve_requires(path, hash)
11
19
  config_dir = File.dirname(path)
12
20
  hash.delete('require').tap do |loaded_features|
13
21
  Array(loaded_features).each do |feature|
14
- FeatureLoader.load(config_directory_path: config_dir, feature: feature)
22
+ if Plugin.plugin_capable?(feature)
23
+ # NOTE: Compatibility for before plugins style.
24
+ warn Rainbow(<<~MESSAGE).yellow
25
+ #{feature} extension supports plugin, specify `plugins: #{feature}` instead of `require: #{feature}` in #{path}.
26
+ For more information, see https://docs.rubocop.org/rubocop/plugin_migration_guide.html.
27
+ MESSAGE
28
+ rubocop_config = Config.create(hash, path, check: false)
29
+
30
+ resolve_plugins(rubocop_config, feature)
31
+ else
32
+ FeatureLoader.load(config_directory_path: config_dir, feature: feature)
33
+ end
15
34
  end
16
35
  end
17
36
  end
@@ -105,7 +124,7 @@ module RuboCop
105
124
  elsif merge_hashes?(base_hash, derived_hash, key)
106
125
  result[key] = merge(base_hash[key], derived_hash[key], **opts)
107
126
  elsif should_union?(derived_hash, base_hash, opts[:inherit_mode], key)
108
- result[key] = base_hash[key] | derived_hash[key]
127
+ result[key] = Array(base_hash[key]) | Array(derived_hash[key])
109
128
  elsif opts[:debug]
110
129
  warn_on_duplicate_setting(base_hash, derived_hash, key, **opts)
111
130
  end
@@ -157,7 +176,7 @@ module RuboCop
157
176
  return false if inherited_file.nil? # Not inheritance resolving merge
158
177
  return false if inherited_file.start_with?('..') # Legitimate override
159
178
  return false if base_hash[key] == derived_hash[key] # Same value
160
- return false if remote_file?(inherited_file) # Can't change
179
+ return false if PathUtil.remote_file?(inherited_file) # Can't change
161
180
 
162
181
  Gem.path.none? { |dir| inherited_file.start_with?(dir) } # Can change?
163
182
  end
@@ -187,7 +206,7 @@ module RuboCop
187
206
  end
188
207
 
189
208
  def should_union?(derived_hash, base_hash, root_mode, key)
190
- return false unless base_hash[key].is_a?(Array)
209
+ return false unless base_hash[key].is_a?(Array) || derived_hash[key].is_a?(Array)
191
210
 
192
211
  derived_mode = derived_hash['inherit_mode']
193
212
  return false if should_override?(derived_mode, key)
@@ -225,7 +244,7 @@ module RuboCop
225
244
  end
226
245
 
227
246
  def inherited_file(path, inherit_from, file)
228
- if remote_file?(inherit_from)
247
+ if PathUtil.remote_file?(inherit_from)
229
248
  # A remote configuration, e.g. `inherit_from: http://example.com/rubocop.yml`.
230
249
  RemoteConfig.new(inherit_from, File.dirname(path))
231
250
  elsif Pathname.new(inherit_from).absolute?
@@ -245,10 +264,6 @@ module RuboCop
245
264
  end
246
265
  end
247
266
 
248
- def remote_file?(uri)
249
- uri.start_with?('http://', 'https://')
250
- end
251
-
252
267
  def remote_config?(file)
253
268
  file.is_a?(RemoteConfig)
254
269
  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
@@ -50,7 +50,7 @@ module RuboCop
50
50
  # Default rules for obsoletions are in config/obsoletion.yml
51
51
  # Additional rules files can be added with `RuboCop::ConfigObsoletion.files << filename`
52
52
  def load_rules # rubocop:disable Metrics/AbcSize
53
- rules = LOAD_RULES_CACHE[self.class.files] ||=
53
+ rules = LOAD_RULES_CACHE[self.class.files.hash] ||=
54
54
  self.class.files.each_with_object({}) do |filename, hash|
55
55
  hash.merge!(YAML.safe_load(File.read(filename)) || {}) do |_key, first, second|
56
56
  case first
@@ -9,7 +9,7 @@ module RuboCop
9
9
 
10
10
  # @api private
11
11
  COMMON_PARAMS = %w[Exclude Include Severity inherit_mode AutoCorrect StyleGuide Details
12
- Enabled].freeze
12
+ Enabled Reference].freeze
13
13
  # @api private
14
14
  INTERNAL_PARAMS = %w[Description StyleGuide
15
15
  VersionAdded VersionChanged VersionRemoved
@@ -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
 
@@ -90,8 +92,10 @@ module RuboCop
90
92
  description_text = string_contents(current_description)
91
93
  return unless (new_description = correct_description(description_text, description_map))
92
94
 
95
+ quote = current_description.dstr_type? ? '"' : "'"
96
+
93
97
  add_offense(current_description, message: message) do |corrector|
94
- corrector.replace(current_description, "'#{new_description}'")
98
+ corrector.replace(current_description, "#{quote}#{new_description}#{quote}")
95
99
  end
96
100
  end
97
101
 
@@ -106,7 +110,7 @@ module RuboCop
106
110
  end
107
111
 
108
112
  def string_contents(node)
109
- node.str_type? ? node.value : node.source
113
+ node.type?(:str, :dstr) ? node.value : node.source
110
114
  end
111
115
  end
112
116
  end