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
@@ -0,0 +1,258 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Lint
6
+ # Checks for redundant uses of `to_s`, `to_sym`, `to_i`, `to_f`, `to_d`, `to_r`, `to_c`,
7
+ # `to_a`, `to_h`, and `to_set`.
8
+ #
9
+ # When one of these methods is called on an object of the same type, that object
10
+ # is returned, making the call unnecessary. The cop detects conversion methods called
11
+ # on object literals, class constructors, class `[]` methods, and the `Kernel` methods
12
+ # `String()`, `Integer()`, `Float()`, BigDecimal(), `Rational()`, `Complex()`, and `Array()`.
13
+ #
14
+ # Specifically, these cases are detected for each conversion method:
15
+ #
16
+ # * `to_s` when called on a string literal, interpolated string, heredoc,
17
+ # or with `String.new` or `String()`.
18
+ # * `to_sym` when called on a symbol literal or interpolated symbol.
19
+ # * `to_i` when called on an integer literal or with `Integer()`.
20
+ # * `to_f` when called on a float literal of with `Float()`.
21
+ # * `to_r` when called on a rational literal or with `Rational()`.
22
+ # * `to_c` when called on a complex literal of with `Complex()`.
23
+ # * `to_a` when called on an array literal, or with `Array.new`, `Array()` or `Array[]`.
24
+ # * `to_h` when called on a hash literal, or with `Hash.new`, `Hash()` or `Hash[]`.
25
+ # * `to_set` when called on `Set.new` or `Set[]`.
26
+ #
27
+ # In all cases, chaining one same `to_*` conversion methods listed above is redundant.
28
+ #
29
+ # The cop can also register an offense for chaining conversion methods on methods that are
30
+ # expected to return a specific type regardless of receiver (eg. `foo.inspect.to_s`).
31
+ #
32
+ # @example
33
+ # # bad
34
+ # "text".to_s
35
+ # :sym.to_sym
36
+ # 42.to_i
37
+ # 8.5.to_f
38
+ # 12r.to_r
39
+ # 1i.to_c
40
+ # [].to_a
41
+ # {}.to_h
42
+ # Set.new.to_set
43
+ #
44
+ # # good
45
+ # "text"
46
+ # :sym
47
+ # 42
48
+ # 8.5
49
+ # 12r
50
+ # 1i
51
+ # []
52
+ # {}
53
+ # Set.new
54
+ #
55
+ # # bad
56
+ # Integer(var).to_i
57
+ #
58
+ # # good
59
+ # Integer(var)
60
+ #
61
+ # # good - chaining to a type constructor with exceptions suppressed
62
+ # # in this case, `Integer()` could return `nil`
63
+ # Integer(var, exception: false).to_i
64
+ #
65
+ # # bad - chaining the same conversion
66
+ # foo.to_s.to_s
67
+ #
68
+ # # good
69
+ # foo.to_s
70
+ #
71
+ # # bad - chaining a conversion to a method that is expected to return the same type
72
+ # inspect.to_s
73
+ #
74
+ # # good
75
+ # inspect
76
+ #
77
+ class RedundantTypeConversion < Base
78
+ extend AutoCorrector
79
+
80
+ MSG = 'Redundant `%<method>s` detected.'
81
+
82
+ # Maps conversion methods to the node types for the literals of that type
83
+ LITERAL_NODE_TYPES = {
84
+ to_s: %i[str dstr],
85
+ to_sym: %i[sym dsym],
86
+ to_i: %i[int],
87
+ to_f: %i[float],
88
+ to_r: %i[rational],
89
+ to_c: %i[complex],
90
+ to_a: %i[array],
91
+ to_h: %i[hash],
92
+ to_set: [] # sets don't have a literal or node type
93
+ }.freeze
94
+
95
+ # Maps each conversion method to the pattern matcher for that type's constructors
96
+ # Not every type has a constructor, for instance Symbol.
97
+ CONSTRUCTOR_MAPPING = {
98
+ to_s: 'string_constructor?',
99
+ to_i: 'integer_constructor?',
100
+ to_f: 'float_constructor?',
101
+ to_d: 'bigdecimal_constructor?',
102
+ to_r: 'rational_constructor?',
103
+ to_c: 'complex_constructor?',
104
+ to_a: 'array_constructor?',
105
+ to_h: 'hash_constructor?',
106
+ to_set: 'set_constructor?'
107
+ }.freeze
108
+
109
+ # Methods that already are expected to return a given type, which makes a further
110
+ # conversion redundant.
111
+ TYPED_METHODS = { to_s: %i[inspect] }.freeze
112
+
113
+ CONVERSION_METHODS = Set[*LITERAL_NODE_TYPES.keys].freeze
114
+ RESTRICT_ON_SEND = CONVERSION_METHODS + [:to_d]
115
+
116
+ private_constant :LITERAL_NODE_TYPES, :CONSTRUCTOR_MAPPING
117
+
118
+ # @!method type_constructor?(node, type_symbol)
119
+ def_node_matcher :type_constructor?, <<~PATTERN
120
+ (send {nil? (const {cbase nil?} :Kernel)} %1 ...)
121
+ PATTERN
122
+
123
+ # @!method string_constructor?(node)
124
+ def_node_matcher :string_constructor?, <<~PATTERN
125
+ {
126
+ (send (const {cbase nil?} :String) :new ...)
127
+ #type_constructor?(:String)
128
+ }
129
+ PATTERN
130
+
131
+ # @!method integer_constructor?(node)
132
+ def_node_matcher :integer_constructor?, <<~PATTERN
133
+ #type_constructor?(:Integer)
134
+ PATTERN
135
+
136
+ # @!method float_constructor?(node)
137
+ def_node_matcher :float_constructor?, <<~PATTERN
138
+ #type_constructor?(:Float)
139
+ PATTERN
140
+
141
+ # @!method bigdecimal_constructor?(node)
142
+ def_node_matcher :bigdecimal_constructor?, <<~PATTERN
143
+ #type_constructor?(:BigDecimal)
144
+ PATTERN
145
+
146
+ # @!method rational_constructor?(node)
147
+ def_node_matcher :rational_constructor?, <<~PATTERN
148
+ #type_constructor?(:Rational)
149
+ PATTERN
150
+
151
+ # @!method complex_constructor?(node)
152
+ def_node_matcher :complex_constructor?, <<~PATTERN
153
+ #type_constructor?(:Complex)
154
+ PATTERN
155
+
156
+ # @!method array_constructor?(node)
157
+ def_node_matcher :array_constructor?, <<~PATTERN
158
+ {
159
+ (send (const {cbase nil?} :Array) {:new :[]} ...)
160
+ #type_constructor?(:Array)
161
+ }
162
+ PATTERN
163
+
164
+ # @!method hash_constructor?(node)
165
+ def_node_matcher :hash_constructor?, <<~PATTERN
166
+ {
167
+ (block (send (const {cbase nil?} :Hash) :new) ...)
168
+ (send (const {cbase nil?} :Hash) {:new :[]} ...)
169
+ (send {nil? (const {cbase nil?} :Kernel)} :Hash ...)
170
+ }
171
+ PATTERN
172
+
173
+ # @!method set_constructor?(node)
174
+ def_node_matcher :set_constructor?, <<~PATTERN
175
+ {
176
+ (send (const {cbase nil?} :Set) {:new :[]} ...)
177
+ }
178
+ PATTERN
179
+
180
+ # @!method exception_false_keyword_argument?(node)
181
+ def_node_matcher :exception_false_keyword_argument?, <<~PATTERN
182
+ (hash (pair (sym :exception) false))
183
+ PATTERN
184
+
185
+ # rubocop:disable Metrics/AbcSize
186
+ def on_send(node)
187
+ return if hash_or_set_with_block?(node)
188
+
189
+ receiver = find_receiver(node)
190
+ return unless literal_receiver?(node, receiver) ||
191
+ constructor?(node, receiver) ||
192
+ chained_conversion?(node, receiver) ||
193
+ chained_to_typed_method?(node, receiver)
194
+
195
+ message = format(MSG, method: node.method_name)
196
+
197
+ add_offense(node.loc.selector, message: message) do |corrector|
198
+ corrector.remove(node.loc.dot.join(node.loc.selector))
199
+ end
200
+ end
201
+ # rubocop:enable Metrics/AbcSize
202
+ alias on_csend on_send
203
+
204
+ private
205
+
206
+ def hash_or_set_with_block?(node)
207
+ return false if !node.method?(:to_h) && !node.method?(:to_set)
208
+
209
+ node.parent&.any_block_type? || node.last_argument&.block_pass_type?
210
+ end
211
+
212
+ def find_receiver(node)
213
+ receiver = node.receiver
214
+ return unless receiver
215
+
216
+ while receiver.begin_type?
217
+ break unless receiver.children.one?
218
+
219
+ receiver = receiver.children.first
220
+ end
221
+
222
+ receiver
223
+ end
224
+
225
+ def literal_receiver?(node, receiver)
226
+ return false unless receiver
227
+
228
+ receiver.type?(*LITERAL_NODE_TYPES[node.method_name])
229
+ end
230
+
231
+ def constructor?(node, receiver)
232
+ matcher = CONSTRUCTOR_MAPPING[node.method_name]
233
+ return false unless matcher
234
+
235
+ public_send(matcher, receiver) && !constructor_suppresses_exceptions?(receiver)
236
+ end
237
+
238
+ def constructor_suppresses_exceptions?(receiver)
239
+ # If the constructor suppresses exceptions with `exception: false`, it is possible
240
+ # it could return `nil`, and therefore a chained conversion is not redundant.
241
+ receiver.arguments.any? { |arg| exception_false_keyword_argument?(arg) }
242
+ end
243
+
244
+ def chained_conversion?(node, receiver)
245
+ return false unless receiver&.call_type?
246
+
247
+ receiver.method?(node.method_name)
248
+ end
249
+
250
+ def chained_to_typed_method?(node, receiver)
251
+ return false unless receiver&.call_type?
252
+
253
+ TYPED_METHODS.fetch(node.method_name, []).include?(receiver.method_name)
254
+ end
255
+ end
256
+ end
257
+ end
258
+ end
@@ -35,22 +35,15 @@ module RuboCop
35
35
  def on_return(return_node)
36
36
  return unless return_node.descendants.any?
37
37
 
38
- context_node = non_void_context(return_node)
39
-
40
- return unless context_node&.def_type?
41
- return unless context_node&.void_context?
38
+ def_node = return_node.each_ancestor(:def).first
39
+ return unless def_node&.void_context?
40
+ return if return_node.each_ancestor(:any_block).any?(&:lambda?)
42
41
 
43
42
  add_offense(
44
43
  return_node.loc.keyword,
45
- message: format(message, method: context_node.method_name)
44
+ message: format(message, method: def_node.method_name)
46
45
  )
47
46
  end
48
-
49
- private
50
-
51
- def non_void_context(return_node)
52
- return_node.each_ancestor(:block, :def, :defs).first
53
- end
54
47
  end
55
48
  end
56
49
  end
@@ -51,7 +51,18 @@ module RuboCop
51
51
 
52
52
  # @!method hash_initialized_with_mutable_shared_object?(node)
53
53
  def_node_matcher :hash_initialized_with_mutable_shared_object?, <<~PATTERN
54
- (send (const {nil? cbase} :Hash) :new {array hash (send (const {nil? cbase} {:Array :Hash}) :new)})
54
+ {
55
+ (send (const {nil? cbase} :Hash) :new [
56
+ {array hash (send (const {nil? cbase} {:Array :Hash}) :new)}
57
+ !#capacity_keyword_argument?
58
+ ])
59
+ (send (const {nil? cbase} :Hash) :new hash #capacity_keyword_argument?)
60
+ }
61
+ PATTERN
62
+
63
+ # @!method capacity_keyword_argument?(node)
64
+ def_node_matcher :capacity_keyword_argument?, <<~PATTERN
65
+ (hash (pair (sym :capacity) _))
55
66
  PATTERN
56
67
 
57
68
  def on_send(node)
@@ -0,0 +1,111 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Lint
6
+ # Checks for cases where exceptions unrelated to the numeric constructors `Integer()`,
7
+ # `Float()`, `BigDecimal()`, `Complex()`, and `Rational()` may be unintentionally swallowed.
8
+ #
9
+ # @safety
10
+ # The cop is unsafe for autocorrection because unexpected errors occurring in the argument
11
+ # passed to numeric constructor (e.g., `Integer()`) can lead to incompatible behavior.
12
+ # For example, changing it to `Integer(potential_exception_method_call, exception: false)`
13
+ # ensures that exceptions raised by `potential_exception_method_call` are not ignored.
14
+ #
15
+ # [source,ruby]
16
+ # ----
17
+ # Integer(potential_exception_method_call) rescue nil
18
+ # ----
19
+ #
20
+ # @example
21
+ #
22
+ # # bad
23
+ # Integer(arg) rescue nil
24
+ #
25
+ # # bad
26
+ # begin
27
+ # Integer(arg)
28
+ # rescue
29
+ # nil
30
+ # end
31
+ #
32
+ # # bad
33
+ # begin
34
+ # Integer(arg)
35
+ # rescue
36
+ # end
37
+ #
38
+ # # good
39
+ # Integer(arg, exception: false)
40
+ #
41
+ class SuppressedExceptionInNumberConversion < Base
42
+ extend AutoCorrector
43
+ extend TargetRubyVersion
44
+
45
+ MSG = 'Use `%<prefer>s` instead.'
46
+ EXPECTED_EXCEPTION_CLASSES = %w[ArgumentError TypeError ::ArgumentError ::TypeError].freeze
47
+
48
+ # @!method numeric_constructor_rescue_nil(node)
49
+ def_node_matcher :numeric_constructor_rescue_nil, <<~PATTERN
50
+ (rescue
51
+ $#numeric_method?
52
+ (resbody nil? nil? (nil)) nil?)
53
+ PATTERN
54
+
55
+ # @!method begin_numeric_constructor_rescue_nil(node)
56
+ def_node_matcher :begin_numeric_constructor_rescue_nil, <<~PATTERN
57
+ (kwbegin
58
+ (rescue
59
+ $#numeric_method?
60
+ (resbody $_? nil?
61
+ {(nil) nil?}) nil?))
62
+ PATTERN
63
+
64
+ # @!method numeric_method?(node)
65
+ def_node_matcher :numeric_method?, <<~PATTERN
66
+ {
67
+ (call #constructor_receiver? {:Integer :BigDecimal :Complex :Rational}
68
+ _ _?)
69
+ (call #constructor_receiver? :Float
70
+ _)
71
+ }
72
+ PATTERN
73
+
74
+ # @!method constructor_receiver?(node)
75
+ def_node_matcher :constructor_receiver?, <<~PATTERN
76
+ {nil? (const {nil? cbase} :Kernel)}
77
+ PATTERN
78
+
79
+ minimum_target_ruby_version 2.6
80
+
81
+ # rubocop:disable Metrics/AbcSize
82
+ def on_rescue(node)
83
+ if (method, exception_classes = begin_numeric_constructor_rescue_nil(node.parent))
84
+ return unless expected_exception_classes_only?(exception_classes)
85
+
86
+ node = node.parent
87
+ else
88
+ return unless (method = numeric_constructor_rescue_nil(node))
89
+ end
90
+
91
+ arguments = method.arguments.map(&:source) << 'exception: false'
92
+ prefer = "#{method.method_name}(#{arguments.join(', ')})"
93
+ prefer = "#{method.receiver.source}#{method.loc.dot.source}#{prefer}" if method.receiver
94
+
95
+ add_offense(node, message: format(MSG, prefer: prefer)) do |corrector|
96
+ corrector.replace(node, prefer)
97
+ end
98
+ end
99
+ # rubocop:enable Metrics/AbcSize
100
+
101
+ private
102
+
103
+ def expected_exception_classes_only?(exception_classes)
104
+ return true unless (arguments = exception_classes.first)
105
+
106
+ (arguments.values.map(&:source) - EXPECTED_EXCEPTION_CLASSES).none?
107
+ end
108
+ end
109
+ end
110
+ end
111
+ end
@@ -0,0 +1,71 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Lint
6
+ # Checks for useless constant scoping. Private constants must be defined using
7
+ # `private_constant`. Even if `private` access modifier is used, it is public scope despite
8
+ # its appearance.
9
+ #
10
+ # It does not support autocorrection due to behavior change and multiple ways to fix it.
11
+ # Or a public constant may be intended.
12
+ #
13
+ # @example
14
+ #
15
+ # # bad
16
+ # class Foo
17
+ # private
18
+ # PRIVATE_CONST = 42
19
+ # end
20
+ #
21
+ # # good
22
+ # class Foo
23
+ # PRIVATE_CONST = 42
24
+ # private_constant :PRIVATE_CONST
25
+ # end
26
+ #
27
+ # # good
28
+ # class Foo
29
+ # PUBLIC_CONST = 42 # If private scope is not intended.
30
+ # end
31
+ #
32
+ class UselessConstantScoping < Base
33
+ MSG = 'Useless `private` access modifier for constant scope.'
34
+
35
+ # @!method private_constants(node)
36
+ def_node_matcher :private_constants, <<~PATTERN
37
+ (send nil? :private_constant $...)
38
+ PATTERN
39
+
40
+ def on_casgn(node)
41
+ return unless after_private_modifier?(node.left_siblings)
42
+ return if private_constantize?(node.right_siblings, node.name)
43
+
44
+ add_offense(node)
45
+ end
46
+
47
+ private
48
+
49
+ def after_private_modifier?(left_siblings)
50
+ access_modifier_candidates = left_siblings.compact.select do |left_sibling|
51
+ left_sibling.respond_to?(:send_type?) && left_sibling.send_type?
52
+ end
53
+
54
+ access_modifier_candidates.any? do |candidate|
55
+ candidate.command?(:private) && candidate.arguments.none?
56
+ end
57
+ end
58
+
59
+ def private_constantize?(right_siblings, const_value)
60
+ private_constant_arguments = right_siblings.map { |node| private_constants(node) }
61
+
62
+ private_constant_values = private_constant_arguments.flatten.filter_map do |constant|
63
+ constant.value.to_sym if constant.respond_to?(:value)
64
+ end
65
+
66
+ private_constant_values.include?(const_value)
67
+ end
68
+ end
69
+ end
70
+ end
71
+ end
@@ -125,9 +125,14 @@ module RuboCop
125
125
  check_nonmutating(expr)
126
126
  end
127
127
 
128
+ # rubocop:disable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity
128
129
  def check_void_op(node, &block)
129
130
  node = node.children.first while node.begin_type?
130
131
  return unless node.call_type? && OPERATORS.include?(node.method_name)
132
+ if !UNARY_OPERATORS.include?(node.method_name) && node.loc.dot && node.arguments.none?
133
+ return
134
+ end
135
+
131
136
  return if block && yield(node)
132
137
 
133
138
  add_offense(node.loc.selector,
@@ -135,6 +140,7 @@ module RuboCop
135
140
  autocorrect_void_op(corrector, node)
136
141
  end
137
142
  end
143
+ # rubocop:enable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity
138
144
 
139
145
  def check_var(node)
140
146
  return unless node.variable? || node.const_type?
@@ -26,6 +26,13 @@ module RuboCop
26
26
  extend NodePattern::Macros
27
27
  include RuboCop::AST::Sexp
28
28
 
29
+ VAR_SETTER_TO_GETTER = {
30
+ lvasgn: :lvar,
31
+ ivasgn: :ivar,
32
+ cvasgn: :cvar,
33
+ gvasgn: :gvar
34
+ }.freeze
35
+
29
36
  # Plug into the calculator
30
37
  def initialize(node, discount_repeated_attributes: false)
31
38
  super(node)
@@ -114,13 +121,6 @@ module RuboCop
114
121
  calls.fetch(value) { yield [calls, value] }
115
122
  end
116
123
 
117
- VAR_SETTER_TO_GETTER = {
118
- lvasgn: :lvar,
119
- ivasgn: :ivar,
120
- cvasgn: :cvar,
121
- gvasgn: :gvar
122
- }.freeze
123
-
124
124
  # @returns `[receiver, method | nil]` for the given setter `node`
125
125
  # or `nil` if it is not a setter.
126
126
  def setter_to_getter(node)
@@ -5,10 +5,10 @@ module RuboCop
5
5
  # This module checks for nodes that should be aligned to the left or right.
6
6
  # This amount is determined by the instance variable @column_delta.
7
7
  module Alignment
8
- private
9
-
10
8
  SPACE = ' '
11
9
 
10
+ private
11
+
12
12
  attr_reader :column_delta
13
13
 
14
14
  def configured_indentation_width
@@ -18,12 +18,12 @@ module RuboCop
18
18
  end
19
19
 
20
20
  # @deprecated Use allowed_line? instead
21
- def ignored_line?
21
+ def ignored_line?(line)
22
22
  warn Rainbow(<<~WARNING).yellow, uplevel: 1
23
23
  `ignored_line?` is deprecated. Use `allowed_line?` instead.
24
24
  WARNING
25
25
 
26
- allowed_line?
26
+ allowed_line?(line)
27
27
  end
28
28
 
29
29
  def matches_allowed_pattern?(line)
@@ -31,12 +31,12 @@ module RuboCop
31
31
  end
32
32
 
33
33
  # @deprecated Use matches_allowed_pattern? instead
34
- def matches_ignored_pattern?
34
+ def matches_ignored_pattern?(line)
35
35
  warn Rainbow(<<~WARNING).yellow, uplevel: 1
36
36
  `matches_ignored_pattern?` is deprecated. Use `matches_allowed_pattern?` instead.
37
37
  WARNING
38
38
 
39
- matches_allowed_pattern?
39
+ matches_allowed_pattern?(line)
40
40
  end
41
41
 
42
42
  def allowed_patterns
@@ -35,7 +35,7 @@ module RuboCop
35
35
  comment_line_numbers = processed_source.comments.map { |comment| comment.loc.line }
36
36
 
37
37
  comment_line_numbers.any? do |comment_line_number|
38
- comment_line_number >= node.first_line && comment_line_number <= node.last_line
38
+ comment_line_number.between?(node.first_line, node.last_line)
39
39
  end
40
40
  end
41
41
 
@@ -82,7 +82,7 @@ module RuboCop
82
82
  next_sibling.source_range
83
83
  next_sibling.loc.line
84
84
  elsif (parent = node.parent)
85
- if parent.loc.respond_to?(:end) && parent.loc.end
85
+ if parent.loc?(:end)
86
86
  parent.loc.end.line
87
87
  else
88
88
  parent.loc.line
@@ -11,6 +11,24 @@ module RuboCop
11
11
  DO_NOT_MIX_OMIT_VALUE_MSG = "#{DO_NOT_MIX_MSG_PREFIX} #{OMIT_HASH_VALUE_MSG}"
12
12
  DO_NOT_MIX_EXPLICIT_VALUE_MSG = "#{DO_NOT_MIX_MSG_PREFIX} #{EXPLICIT_HASH_VALUE_MSG}"
13
13
 
14
+ DefNode = Struct.new(:node) do
15
+ def selector
16
+ if node.loc.respond_to?(:selector)
17
+ node.loc.selector
18
+ else
19
+ node.loc.keyword
20
+ end
21
+ end
22
+
23
+ def first_argument
24
+ node.first_argument
25
+ end
26
+
27
+ def last_argument
28
+ node.last_argument
29
+ end
30
+ end
31
+
14
32
  def on_hash_for_mixed_shorthand(hash_node)
15
33
  return if ignore_mixed_hash_shorthand_syntax?(hash_node)
16
34
 
@@ -212,24 +230,6 @@ module RuboCop
212
230
  register_offense(pair_node, OMIT_HASH_VALUE_MSG, replacement)
213
231
  end
214
232
  end
215
-
216
- DefNode = Struct.new(:node) do
217
- def selector
218
- if node.loc.respond_to?(:selector)
219
- node.loc.selector
220
- else
221
- node.loc.keyword
222
- end
223
- end
224
-
225
- def first_argument
226
- node.first_argument
227
- end
228
-
229
- def last_argument
230
- node.last_argument
231
- end
232
- end
233
233
  end
234
234
  end
235
235
  # rubocop:enable Metrics/ModuleLength