rubocop 1.71.2 → 1.73.2

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 (101) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +3 -3
  3. data/config/default.yml +55 -13
  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 +23 -9
  10. data/lib/rubocop/config_validator.rb +1 -1
  11. data/lib/rubocop/cop/internal_affairs/example_description.rb +7 -3
  12. data/lib/rubocop/cop/internal_affairs/location_exists.rb +116 -0
  13. data/lib/rubocop/cop/internal_affairs/node_pattern_groups/ast_walker.rb +1 -1
  14. data/lib/rubocop/cop/internal_affairs/node_type_group.rb +91 -0
  15. data/lib/rubocop/cop/internal_affairs/plugin.rb +33 -0
  16. data/lib/rubocop/cop/internal_affairs/undefined_config.rb +7 -1
  17. data/lib/rubocop/cop/internal_affairs.rb +2 -16
  18. data/lib/rubocop/cop/layout/block_alignment.rb +2 -0
  19. data/lib/rubocop/cop/layout/closing_parenthesis_indentation.rb +4 -4
  20. data/lib/rubocop/cop/layout/else_alignment.rb +1 -1
  21. data/lib/rubocop/cop/layout/empty_line_after_guard_clause.rb +1 -1
  22. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +26 -1
  23. data/lib/rubocop/cop/layout/empty_lines_around_method_body.rb +22 -2
  24. data/lib/rubocop/cop/layout/heredoc_argument_closing_parenthesis.rb +1 -1
  25. data/lib/rubocop/cop/layout/line_length.rb +3 -3
  26. data/lib/rubocop/cop/layout/multiline_method_call_indentation.rb +2 -2
  27. data/lib/rubocop/cop/layout/space_around_method_call_operator.rb +1 -1
  28. data/lib/rubocop/cop/lint/cop_directive_syntax.rb +84 -0
  29. data/lib/rubocop/cop/lint/duplicate_methods.rb +0 -14
  30. data/lib/rubocop/cop/lint/empty_conditional_body.rb +14 -64
  31. data/lib/rubocop/cop/lint/erb_new_arguments.rb +0 -6
  32. data/lib/rubocop/cop/lint/float_comparison.rb +1 -6
  33. data/lib/rubocop/cop/lint/format_parameter_mismatch.rb +1 -1
  34. data/lib/rubocop/cop/lint/literal_as_condition.rb +99 -9
  35. data/lib/rubocop/cop/lint/mixed_case_range.rb +2 -2
  36. data/lib/rubocop/cop/lint/redundant_require_statement.rb +0 -21
  37. data/lib/rubocop/cop/lint/redundant_type_conversion.rb +252 -0
  38. data/lib/rubocop/cop/lint/suppressed_exception_in_number_conversion.rb +111 -0
  39. data/lib/rubocop/cop/lint/useless_constant_scoping.rb +80 -0
  40. data/lib/rubocop/cop/lint/void.rb +6 -0
  41. data/lib/rubocop/cop/metrics/utils/repeated_attribute_discount.rb +7 -7
  42. data/lib/rubocop/cop/mixin/alignment.rb +2 -2
  43. data/lib/rubocop/cop/mixin/allowed_pattern.rb +4 -4
  44. data/lib/rubocop/cop/mixin/comments_help.rb +1 -1
  45. data/lib/rubocop/cop/mixin/hash_shorthand_syntax.rb +18 -18
  46. data/lib/rubocop/cop/mixin/hash_subset.rb +19 -4
  47. data/lib/rubocop/cop/mixin/hash_transform_method.rb +74 -74
  48. data/lib/rubocop/cop/mixin/percent_literal.rb +1 -1
  49. data/lib/rubocop/cop/mixin/range_help.rb +15 -3
  50. data/lib/rubocop/cop/mixin/string_help.rb +1 -1
  51. data/lib/rubocop/cop/mixin/trailing_comma.rb +12 -0
  52. data/lib/rubocop/cop/naming/block_forwarding.rb +3 -3
  53. data/lib/rubocop/cop/naming/predicate_name.rb +44 -0
  54. data/lib/rubocop/cop/naming/variable_name.rb +64 -6
  55. data/lib/rubocop/cop/style/accessor_grouping.rb +19 -5
  56. data/lib/rubocop/cop/style/arguments_forwarding.rb +3 -3
  57. data/lib/rubocop/cop/style/commented_keyword.rb +1 -1
  58. data/lib/rubocop/cop/style/endless_method.rb +163 -18
  59. data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -7
  60. data/lib/rubocop/cop/style/inverse_methods.rb +8 -5
  61. data/lib/rubocop/cop/style/keyword_parameters_order.rb +13 -7
  62. data/lib/rubocop/cop/style/line_end_concatenation.rb +10 -4
  63. data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +1 -1
  64. data/lib/rubocop/cop/style/multiline_block_chain.rb +1 -1
  65. data/lib/rubocop/cop/style/multiline_method_signature.rb +1 -9
  66. data/lib/rubocop/cop/style/redundant_condition.rb +45 -0
  67. data/lib/rubocop/cop/style/redundant_format.rb +250 -0
  68. data/lib/rubocop/cop/style/redundant_freeze.rb +1 -1
  69. data/lib/rubocop/cop/style/redundant_parentheses.rb +18 -4
  70. data/lib/rubocop/cop/style/redundant_self_assignment.rb +1 -1
  71. data/lib/rubocop/cop/style/single_line_methods.rb +3 -3
  72. data/lib/rubocop/cop/style/sole_nested_conditional.rb +0 -6
  73. data/lib/rubocop/cop/style/string_concatenation.rb +1 -1
  74. data/lib/rubocop/cop/style/trailing_comma_in_array_literal.rb +47 -6
  75. data/lib/rubocop/cop/style/trailing_comma_in_hash_literal.rb +48 -6
  76. data/lib/rubocop/cop/style/trivial_accessors.rb +1 -1
  77. data/lib/rubocop/cop/util.rb +1 -1
  78. data/lib/rubocop/cop/utils/format_string.rb +10 -5
  79. data/lib/rubocop/cops_documentation_generator.rb +12 -1
  80. data/lib/rubocop/directive_comment.rb +35 -2
  81. data/lib/rubocop/lsp/runtime.rb +2 -0
  82. data/lib/rubocop/lsp/server.rb +0 -2
  83. data/lib/rubocop/options.rb +26 -11
  84. data/lib/rubocop/path_util.rb +4 -0
  85. data/lib/rubocop/plugin/configuration_integrator.rb +143 -0
  86. data/lib/rubocop/plugin/load_error.rb +26 -0
  87. data/lib/rubocop/plugin/loader.rb +100 -0
  88. data/lib/rubocop/plugin/not_supported_error.rb +29 -0
  89. data/lib/rubocop/plugin.rb +46 -0
  90. data/lib/rubocop/rake_task.rb +4 -1
  91. data/lib/rubocop/rspec/cop_helper.rb +9 -0
  92. data/lib/rubocop/rspec/shared_contexts.rb +15 -0
  93. data/lib/rubocop/rspec/support.rb +1 -0
  94. data/lib/rubocop/server/cache.rb +35 -2
  95. data/lib/rubocop/server/cli.rb +2 -2
  96. data/lib/rubocop/version.rb +17 -2
  97. data/lib/rubocop.rb +5 -1
  98. data/lib/ruby_lsp/rubocop/addon.rb +7 -10
  99. data/lib/ruby_lsp/rubocop/{wraps_built_in_lsp_runtime.rb → runtime_adapter.rb} +5 -8
  100. metadata +36 -10
  101. 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: 671bf05708f62eb3abe3b448168f7daa905cffe60e2652ad4ec21964594adf70
4
+ data.tar.gz: caef12b9b307560cda7125652fa0249fcc0908725e97cc963a10f4c1e0782faa
5
5
  SHA512:
6
- metadata.gz: 8729a8ea3adae11fcb2501bfea338821a8f9894209ea6fb482b62ebfae87019e07ded7bf02d3371c18726b28c2491c79952e7b86c193388a045f886a833b427c
7
- data.tar.gz: 44251afbbbc2aa7c121033ea05f25e1b3447afba7487d04a81f9463347ba28bcea05bdc2e434248f15cd235d9e452cb3f9f378f94adb5507eeb6d5ce58ba4119
6
+ metadata.gz: fea1053c5a5f72ffc9eea331b438f1fc1d876eed21c0f251569ab541cf3f9724193e1c3bf72a0154e84608ac4346cb7a122fcc4a5266fe7b520109c56c5dea1a
7
+ data.tar.gz: e2b4c854a638b119a3b28e39e77dd05d86fe85b890180bc9a70ac10727f9e74e1ed64e659fe397042610e4bc87b3153d5783d472a596de110628395708da6177
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.73', 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.'
@@ -3899,6 +3923,8 @@ Style/EndlessMethod:
3899
3923
  - allow_single_line
3900
3924
  - allow_always
3901
3925
  - disallow
3926
+ - require_single_line
3927
+ - require_always
3902
3928
 
3903
3929
  Style/EnvHome:
3904
3930
  Description: "Checks for consistent usage of `ENV['HOME']`."
@@ -5081,6 +5107,9 @@ Style/RedundantCondition:
5081
5107
  Description: 'Checks for unnecessary conditional expressions.'
5082
5108
  Enabled: true
5083
5109
  VersionAdded: '0.76'
5110
+ VersionChanged: '1.73'
5111
+ AllowedMethods:
5112
+ - nonzero?
5084
5113
 
5085
5114
  Style/RedundantConditional:
5086
5115
  Description: "Don't return true/false from a conditional."
@@ -5145,6 +5174,13 @@ Style/RedundantFilterChain:
5145
5174
  VersionAdded: '1.52'
5146
5175
  VersionChanged: '1.57'
5147
5176
 
5177
+ Style/RedundantFormat:
5178
+ Description: 'Checks for usages of `Kernel#format` or `Kernel#sprintf` with only a single argument.'
5179
+ Enabled: pending
5180
+ SafeAutoCorrect: false
5181
+ VersionAdded: '1.72'
5182
+ VersionChanged: '1.72'
5183
+
5148
5184
  Style/RedundantFreeze:
5149
5185
  Description: "Checks usages of Object#freeze on immutable objects."
5150
5186
  Enabled: true
@@ -5671,14 +5707,17 @@ Style/TrailingCommaInArrayLiteral:
5671
5707
  StyleGuide: '#no-trailing-array-commas'
5672
5708
  Enabled: true
5673
5709
  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.
5710
+ # If `comma`, the cop requires a comma after the last item in an array, but only when each item is
5711
+ # on its own line.
5712
+ # If `consistent_comma`, the cop requires a comma after the last item of all non-empty, multiline
5713
+ # array literals.
5714
+ # If `diff_comma`, the cop requires a comma after the last item of all non-empty, multiline array
5715
+ # literals, but only when that last item immediately precedes a newline.
5678
5716
  EnforcedStyleForMultiline: no_comma
5679
5717
  SupportedStylesForMultiline:
5680
5718
  - comma
5681
5719
  - consistent_comma
5720
+ - diff_comma
5682
5721
  - no_comma
5683
5722
 
5684
5723
  Style/TrailingCommaInBlockArgs:
@@ -5690,14 +5729,17 @@ Style/TrailingCommaInBlockArgs:
5690
5729
  Style/TrailingCommaInHashLiteral:
5691
5730
  Description: 'Checks for trailing comma in hash literals.'
5692
5731
  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.
5732
+ # If `comma`, the cop requires a comma after the last item in a hash, but only when each item is
5733
+ # on its own line.
5734
+ # If `consistent_comma`, the cop requires a comma after the last item of all non-empty, multiline
5735
+ # hash literals.
5736
+ # If `diff_comma`, the cop requires a comma after the last item of all non-empty, multiline hash
5737
+ # literals, but only when that last item immediately precedes a newline.
5697
5738
  EnforcedStyleForMultiline: no_comma
5698
5739
  SupportedStylesForMultiline:
5699
5740
  - comma
5700
5741
  - consistent_comma
5742
+ - diff_comma
5701
5743
  - no_comma
5702
5744
  VersionAdded: '0.53'
5703
5745
 
@@ -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,34 @@
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
+ return if (plugins = Array(plugins)).empty?
13
+
14
+ Plugin.integrate_plugins(rubocop_config, plugins)
15
+ end
16
+
10
17
  def resolve_requires(path, hash)
11
18
  config_dir = File.dirname(path)
12
19
  hash.delete('require').tap do |loaded_features|
13
20
  Array(loaded_features).each do |feature|
14
- FeatureLoader.load(config_directory_path: config_dir, feature: feature)
21
+ if Plugin.plugin_capable?(feature)
22
+ # NOTE: Compatibility for before plugins style.
23
+ warn Rainbow(<<~MESSAGE).yellow
24
+ #{feature} extension supports plugin, specify `plugins: #{feature}` instead of `require: #{feature}` in #{path}.
25
+ For more information, see https://docs.rubocop.org/rubocop/plugin_migration_guide.html.
26
+ MESSAGE
27
+ rubocop_config = Config.create(hash, path, check: false)
28
+
29
+ resolve_plugins(rubocop_config, feature)
30
+ else
31
+ FeatureLoader.load(config_directory_path: config_dir, feature: feature)
32
+ end
15
33
  end
16
34
  end
17
35
  end
@@ -105,7 +123,7 @@ module RuboCop
105
123
  elsif merge_hashes?(base_hash, derived_hash, key)
106
124
  result[key] = merge(base_hash[key], derived_hash[key], **opts)
107
125
  elsif should_union?(derived_hash, base_hash, opts[:inherit_mode], key)
108
- result[key] = base_hash[key] | derived_hash[key]
126
+ result[key] = Array(base_hash[key]) | Array(derived_hash[key])
109
127
  elsif opts[:debug]
110
128
  warn_on_duplicate_setting(base_hash, derived_hash, key, **opts)
111
129
  end
@@ -157,7 +175,7 @@ module RuboCop
157
175
  return false if inherited_file.nil? # Not inheritance resolving merge
158
176
  return false if inherited_file.start_with?('..') # Legitimate override
159
177
  return false if base_hash[key] == derived_hash[key] # Same value
160
- return false if remote_file?(inherited_file) # Can't change
178
+ return false if PathUtil.remote_file?(inherited_file) # Can't change
161
179
 
162
180
  Gem.path.none? { |dir| inherited_file.start_with?(dir) } # Can change?
163
181
  end
@@ -187,7 +205,7 @@ module RuboCop
187
205
  end
188
206
 
189
207
  def should_union?(derived_hash, base_hash, root_mode, key)
190
- return false unless base_hash[key].is_a?(Array)
208
+ return false unless base_hash[key].is_a?(Array) || derived_hash[key].is_a?(Array)
191
209
 
192
210
  derived_mode = derived_hash['inherit_mode']
193
211
  return false if should_override?(derived_mode, key)
@@ -225,7 +243,7 @@ module RuboCop
225
243
  end
226
244
 
227
245
  def inherited_file(path, inherit_from, file)
228
- if remote_file?(inherit_from)
246
+ if PathUtil.remote_file?(inherit_from)
229
247
  # A remote configuration, e.g. `inherit_from: http://example.com/rubocop.yml`.
230
248
  RemoteConfig.new(inherit_from, File.dirname(path))
231
249
  elsif Pathname.new(inherit_from).absolute?
@@ -245,10 +263,6 @@ module RuboCop
245
263
  end
246
264
  end
247
265
 
248
- def remote_file?(uri)
249
- uri.start_with?('http://', 'https://')
250
- end
251
-
252
266
  def remote_config?(file)
253
267
  file.is_a?(RemoteConfig)
254
268
  end
@@ -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
@@ -0,0 +1,116 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module InternalAffairs
6
+ # When a node location may not exist, `Node#loc?` or `Node#loc_is?`
7
+ # can be used instead of calling `Node#respond_to?` before using
8
+ # the value.
9
+ #
10
+ # @example
11
+ # # bad
12
+ # node.loc.respond_to?(:begin) && node.loc.begin
13
+ #
14
+ # # good
15
+ # node.loc?(:begin)
16
+ #
17
+ # # bad
18
+ # node.loc.respond_to?(:begin) && node.loc.begin.is?('(')
19
+ #
20
+ # # good
21
+ # node.loc_is?(:begin, '(')
22
+ #
23
+ # # bad
24
+ # node.loc.respond_to?(:begin) && node.loc.begin.source == '('
25
+ #
26
+ # # good
27
+ # node.loc_is?(:begin, '(')
28
+ #
29
+ class LocationExists < Base
30
+ extend AutoCorrector
31
+
32
+ MSG = 'Use `%<replacement>s` instead of `%<source>s`.'
33
+
34
+ # @!method replaceable_with_loc_is(node)
35
+ def_node_matcher :replaceable_with_loc_is, <<~PATTERN
36
+ (and
37
+ (call
38
+ (call $_receiver :loc) :respond_to?
39
+ $(sym _location))
40
+ {
41
+ (call
42
+ (call
43
+ (call _receiver :loc) _location) :is?
44
+ $(str _))
45
+ (call
46
+ (call
47
+ (call
48
+ (call _receiver :loc) _location) :source) :==
49
+ $(str _))
50
+ })
51
+ PATTERN
52
+
53
+ # @!method replaceable_with_loc(node)
54
+ def_node_matcher :replaceable_with_loc, <<~PATTERN
55
+ (and
56
+ (call
57
+ (call $_receiver :loc) :respond_to?
58
+ $(sym _location))
59
+ (call
60
+ (call _receiver :loc) _location))
61
+ PATTERN
62
+
63
+ def on_and(node)
64
+ replace_with_loc(node) || replace_with_loc_is(node)
65
+ end
66
+
67
+ private
68
+
69
+ def replace_with_loc(node)
70
+ replaceable_with_loc(node) do |receiver, location|
71
+ if node.parent&.assignment?
72
+ register_offense(node, replace_assignment(receiver, location))
73
+ else
74
+ register_offense(node, replacement(receiver, "loc?(#{location.source})"))
75
+ end
76
+ end
77
+ end
78
+
79
+ def replace_with_loc_is(node)
80
+ replaceable_with_loc_is(node) do |receiver, location, value|
81
+ replacement = replacement(receiver, "loc_is?(#{location.source}, #{value.source})")
82
+ register_offense(node, replacement)
83
+ end
84
+ end
85
+
86
+ def register_offense(node, replacement)
87
+ message = format(MSG, replacement: replacement, source: node.source)
88
+
89
+ add_offense(node, message: message) do |corrector|
90
+ corrector.replace(node, replacement)
91
+ end
92
+ end
93
+
94
+ def replacement(receiver, rest)
95
+ "#{replace_receiver(receiver)}#{rest}"
96
+ end
97
+
98
+ def replace_assignment(receiver, location)
99
+ prefix = replace_receiver(receiver)
100
+
101
+ "#{prefix}loc#{dot(receiver)}#{location.value} if #{prefix}loc?(#{location.source})"
102
+ end
103
+
104
+ def replace_receiver(receiver)
105
+ return '' unless receiver
106
+
107
+ "#{receiver.source}#{dot(receiver)}"
108
+ end
109
+
110
+ def dot(node)
111
+ node.parent.loc.dot.source
112
+ end
113
+ end
114
+ end
115
+ end
116
+ end
@@ -3,7 +3,7 @@
3
3
  module RuboCop
4
4
  module Cop
5
5
  module InternalAffairs
6
- # rubocop:disable InternalAffairs/RedundantSourceRange - node here is a `NodePattern::Node`
6
+ # rubocop:disable InternalAffairs/RedundantSourceRange -- node here is a `NodePattern::Node`
7
7
  class NodePatternGroups
8
8
  # Walks an AST that has been processed by `InternalAffairs::NodePatternGroups::Processor`
9
9
  # in order to find `node_type` and `node_sequence` nodes that can be replaced with a node