rubocop 1.75.8 → 1.77.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 (65) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +20 -14
  3. data/config/default.yml +72 -7
  4. data/config/obsoletion.yml +6 -3
  5. data/lib/rubocop/cop/bundler/ordered_gems.rb +1 -1
  6. data/lib/rubocop/cop/correctors/parentheses_corrector.rb +5 -2
  7. data/lib/rubocop/cop/gemspec/attribute_assignment.rb +91 -0
  8. data/lib/rubocop/cop/gemspec/duplicated_assignment.rb +0 -22
  9. data/lib/rubocop/cop/gemspec/ordered_dependencies.rb +1 -1
  10. data/lib/rubocop/cop/gemspec/require_mfa.rb +15 -1
  11. data/lib/rubocop/cop/internal_affairs/node_matcher_directive.rb +4 -4
  12. data/lib/rubocop/cop/internal_affairs/node_pattern_groups.rb +1 -0
  13. data/lib/rubocop/cop/layout/closing_parenthesis_indentation.rb +1 -1
  14. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +1 -1
  15. data/lib/rubocop/cop/layout/line_length.rb +26 -5
  16. data/lib/rubocop/cop/layout/space_before_brackets.rb +2 -9
  17. data/lib/rubocop/cop/layout/space_inside_array_literal_brackets.rb +7 -2
  18. data/lib/rubocop/cop/lint/ambiguous_range.rb +5 -0
  19. data/lib/rubocop/cop/lint/empty_interpolation.rb +3 -1
  20. data/lib/rubocop/cop/lint/float_comparison.rb +4 -4
  21. data/lib/rubocop/cop/lint/identity_comparison.rb +19 -15
  22. data/lib/rubocop/cop/lint/literal_as_condition.rb +19 -27
  23. data/lib/rubocop/cop/lint/redundant_regexp_quantifiers.rb +1 -1
  24. data/lib/rubocop/cop/lint/safe_navigation_chain.rb +4 -4
  25. data/lib/rubocop/cop/lint/self_assignment.rb +25 -0
  26. data/lib/rubocop/cop/lint/shadowing_outer_local_variable.rb +5 -0
  27. data/lib/rubocop/cop/lint/useless_access_modifier.rb +29 -4
  28. data/lib/rubocop/cop/lint/useless_default_value_argument.rb +90 -0
  29. data/lib/rubocop/cop/lint/useless_or.rb +98 -0
  30. data/lib/rubocop/cop/lint/useless_ruby2_keywords.rb +3 -3
  31. data/lib/rubocop/cop/mixin/alignment.rb +1 -1
  32. data/lib/rubocop/cop/mixin/frozen_string_literal.rb +1 -1
  33. data/lib/rubocop/cop/mixin/gemspec_help.rb +22 -0
  34. data/lib/rubocop/cop/mixin/line_length_help.rb +24 -8
  35. data/lib/rubocop/cop/mixin/ordered_gem_node.rb +1 -1
  36. data/lib/rubocop/cop/naming/file_name.rb +2 -2
  37. data/lib/rubocop/cop/naming/predicate_method.rb +281 -0
  38. data/lib/rubocop/cop/naming/{predicate_name.rb → predicate_prefix.rb} +4 -4
  39. data/lib/rubocop/cop/style/case_like_if.rb +1 -1
  40. data/lib/rubocop/cop/style/collection_querying.rb +167 -0
  41. data/lib/rubocop/cop/style/conditional_assignment.rb +3 -1
  42. data/lib/rubocop/cop/style/empty_string_inside_interpolation.rb +100 -0
  43. data/lib/rubocop/cop/style/exponential_notation.rb +2 -2
  44. data/lib/rubocop/cop/style/fetch_env_var.rb +32 -6
  45. data/lib/rubocop/cop/style/hash_conversion.rb +12 -3
  46. data/lib/rubocop/cop/style/if_unless_modifier.rb +13 -6
  47. data/lib/rubocop/cop/style/it_block_parameter.rb +33 -14
  48. data/lib/rubocop/cop/style/method_call_with_args_parentheses/omit_parentheses.rb +2 -2
  49. data/lib/rubocop/cop/style/min_max_comparison.rb +13 -5
  50. data/lib/rubocop/cop/style/redundant_array_flatten.rb +50 -0
  51. data/lib/rubocop/cop/style/redundant_interpolation.rb +1 -1
  52. data/lib/rubocop/cop/style/redundant_parentheses.rb +26 -5
  53. data/lib/rubocop/cop/style/redundant_self.rb +8 -5
  54. data/lib/rubocop/cop/style/safe_navigation.rb +24 -11
  55. data/lib/rubocop/cop/style/sole_nested_conditional.rb +2 -1
  56. data/lib/rubocop/cop/style/symbol_proc.rb +1 -1
  57. data/lib/rubocop/cop/style/trailing_comma_in_block_args.rb +1 -1
  58. data/lib/rubocop/formatter/fuubar_style_formatter.rb +1 -1
  59. data/lib/rubocop/formatter/offense_count_formatter.rb +1 -1
  60. data/lib/rubocop/lsp/diagnostic.rb +4 -4
  61. data/lib/rubocop/rspec/expect_offense.rb +9 -3
  62. data/lib/rubocop/version.rb +1 -1
  63. data/lib/rubocop.rb +8 -1
  64. data/lib/ruby_lsp/rubocop/addon.rb +2 -2
  65. metadata +14 -7
@@ -25,20 +25,24 @@ module RuboCop
25
25
  config.for_cop('Layout/LineLength')['AllowURI']
26
26
  end
27
27
 
28
- def allowed_uri_position?(line, uri_range)
29
- uri_range.begin < max_line_length && uri_range.end == line_length(line)
28
+ def allow_qualified_name?
29
+ config.for_cop('Layout/LineLength')['AllowQualifiedName']
30
+ end
31
+
32
+ def allowed_position?(line, range)
33
+ range.begin < max_line_length && range.end == line_length(line)
30
34
  end
31
35
 
32
36
  def line_length(line)
33
37
  line.length + indentation_difference(line)
34
38
  end
35
39
 
36
- def find_excessive_uri_range(line)
37
- last_uri_match = match_uris(line).last
38
- return nil unless last_uri_match
40
+ def find_excessive_range(line, type)
41
+ last_match = (type == :uri ? match_uris(line) : match_qualified_names(line)).last
42
+ return nil unless last_match
39
43
 
40
- begin_position, end_position = last_uri_match.offset(0)
41
- end_position = extend_uri_end_position(line, end_position)
44
+ begin_position, end_position = last_match.offset(0)
45
+ end_position = extend_end_position(line, end_position)
42
46
 
43
47
  line_indentation_difference = indentation_difference(line)
44
48
  begin_position += line_indentation_difference
@@ -57,6 +61,14 @@ module RuboCop
57
61
  matches
58
62
  end
59
63
 
64
+ def match_qualified_names(string)
65
+ matches = []
66
+ string.scan(qualified_name_regexp) do
67
+ matches << $LAST_MATCH_INFO
68
+ end
69
+ matches
70
+ end
71
+
60
72
  def indentation_difference(line)
61
73
  return 0 unless tab_indentation_width
62
74
 
@@ -70,7 +82,7 @@ module RuboCop
70
82
  index * (tab_indentation_width - 1)
71
83
  end
72
84
 
73
- def extend_uri_end_position(line, end_position)
85
+ def extend_end_position(line, end_position)
74
86
  # Extend the end position YARD comments with linked URLs of the form {<uri> <title>}
75
87
  if line&.match(/{(\s|\S)*}$/)
76
88
  match = line[end_position..line_length(line)]&.match(/(\s|\S)*}/)
@@ -101,6 +113,10 @@ module RuboCop
101
113
  end
102
114
  end
103
115
 
116
+ def qualified_name_regexp
117
+ /\b(?:[A-Z][A-Za-z0-9_]*::)+[A-Za-z_][A-Za-z0-9_]*\b/
118
+ end
119
+
104
120
  def valid_uri?(uri_ish_string)
105
121
  URI.parse(uri_ish_string)
106
122
  true
@@ -24,7 +24,7 @@ module RuboCop
24
24
  gem_canonical_name(string_a) < gem_canonical_name(string_b)
25
25
  end
26
26
 
27
- def consecutive_lines(previous, current)
27
+ def consecutive_lines?(previous, current)
28
28
  first_line = get_source_range(current, treat_comments_as_separators).first_line
29
29
  previous.source_range.last_line == first_line - 1
30
30
  end
@@ -152,7 +152,7 @@ module RuboCop
152
152
 
153
153
  const_namespace, const_name = *const
154
154
  next if name != const_name && !match_acronym?(name, const_name)
155
- next unless namespace.empty? || match_namespace(child, const_namespace, namespace)
155
+ next unless namespace.empty? || namespace_matches?(child, const_namespace, namespace)
156
156
 
157
157
  return node
158
158
  end
@@ -169,7 +169,7 @@ module RuboCop
169
169
  s(:const, namespace, name) if name
170
170
  end
171
171
 
172
- def match_namespace(node, namespace, expected)
172
+ def namespace_matches?(node, namespace, expected)
173
173
  match_partial = partial_matcher!(expected)
174
174
 
175
175
  match_partial.call(namespace)
@@ -0,0 +1,281 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Naming
6
+ # Checks that predicate methods end with `?` and non-predicate methods do not.
7
+ #
8
+ # The names of predicate methods (methods that return a boolean value) should end
9
+ # in a question mark. Methods that don't return a boolean, shouldn't
10
+ # end in a question mark.
11
+ #
12
+ # The cop assesses a predicate method as one that returns boolean values. Likewise,
13
+ # a method that only returns literal values is assessed as non-predicate. Other predicate
14
+ # method calls are assumed to return boolean values. The cop does not make an assessment
15
+ # if the return type is unknown (non-predicate method calls, variables, etc.).
16
+ #
17
+ # NOTE: Operator methods (`def ==`, etc.) are ignored.
18
+ #
19
+ # By default, the cop runs in `conservative` mode, which allows a method to be named
20
+ # with a question mark as long as at least one return value is boolean. In `aggressive`
21
+ # mode, methods with a question mark will register an offense if any known non-boolean
22
+ # return values are detected.
23
+ #
24
+ # The cop also has `AllowedMethods` configuration in order to prevent the cop from
25
+ # registering an offense from a method name that does not confirm to the naming
26
+ # guidelines. By default, `call` is allowed. The cop also has `AllowedPatterns`
27
+ # configuration to allow method names by regular expression.
28
+ #
29
+ # The cop can furthermore be configured to allow all bang methods (method names
30
+ # ending with `!`), with `AllowBangMethods: true` (default false).
31
+ #
32
+ # @example Mode: conservative (default)
33
+ # # bad
34
+ # def foo
35
+ # bar == baz
36
+ # end
37
+ #
38
+ # # good
39
+ # def foo?
40
+ # bar == baz
41
+ # end
42
+ #
43
+ # # bad
44
+ # def foo?
45
+ # 5
46
+ # end
47
+ #
48
+ # # good
49
+ # def foo
50
+ # 5
51
+ # end
52
+ #
53
+ # # bad
54
+ # def foo
55
+ # x == y
56
+ # end
57
+ #
58
+ # # good
59
+ # def foo?
60
+ # x == y
61
+ # end
62
+ #
63
+ # # bad
64
+ # def foo
65
+ # !x
66
+ # end
67
+ #
68
+ # # good
69
+ # def foo?
70
+ # !x
71
+ # end
72
+ #
73
+ # # bad - returns the value of another predicate method
74
+ # def foo
75
+ # bar?
76
+ # end
77
+ #
78
+ # # good
79
+ # def foo?
80
+ # bar?
81
+ # end
82
+ #
83
+ # # good - operator method
84
+ # def ==(other)
85
+ # hash == other.hash
86
+ # end
87
+ #
88
+ # # good - at least one return value is boolean
89
+ # def foo?
90
+ # return unless bar?
91
+ # true
92
+ # end
93
+ #
94
+ # # ok - return type is not known
95
+ # def foo?
96
+ # bar
97
+ # end
98
+ #
99
+ # # ok - return type is not known
100
+ # def foo
101
+ # bar?
102
+ # end
103
+ #
104
+ # @example Mode: aggressive
105
+ # # bad - the method returns nil in some cases
106
+ # def foo?
107
+ # return unless bar?
108
+ # true
109
+ # end
110
+ #
111
+ # @example AllowBangMethods: false (default)
112
+ # # bad
113
+ # def save!
114
+ # true
115
+ # end
116
+ #
117
+ # @example AllowBangMethods: true
118
+ # # good
119
+ # def save!
120
+ # true
121
+ # end
122
+ #
123
+ class PredicateMethod < Base
124
+ include AllowedMethods
125
+ include AllowedPattern
126
+
127
+ MSG_PREDICATE = 'Predicate method names should end with `?`.'
128
+ MSG_NON_PREDICATE = 'Non-predicate method names should not end with `?`.'
129
+
130
+ def on_def(node)
131
+ return if allowed?(node)
132
+
133
+ return_values = return_values(node.body)
134
+ return if acceptable?(return_values)
135
+
136
+ if node.predicate_method? && potential_non_predicate?(return_values)
137
+ add_offense(node.loc.name, message: MSG_NON_PREDICATE)
138
+ elsif !node.predicate_method? && all_return_values_boolean?(return_values)
139
+ add_offense(node.loc.name, message: MSG_PREDICATE)
140
+ end
141
+ end
142
+ alias on_defs on_def
143
+
144
+ private
145
+
146
+ def allowed?(node)
147
+ allowed_method?(node.method_name) ||
148
+ matches_allowed_pattern?(node.method_name) ||
149
+ allowed_bang_method?(node) ||
150
+ node.operator_method? ||
151
+ node.body.nil?
152
+ end
153
+
154
+ def acceptable?(return_values)
155
+ # In `conservative` mode, if the method returns `super`, `zsuper`, or a
156
+ # non-comparison method call, the method name is acceptable.
157
+ return false unless conservative?
158
+
159
+ return_values.any? do |value|
160
+ value.type?(:super, :zsuper) || unknown_method_call?(value)
161
+ end
162
+ end
163
+
164
+ def unknown_method_call?(value)
165
+ return false unless value.call_type?
166
+
167
+ !value.comparison_method? && !value.predicate_method? && !value.negation_method?
168
+ end
169
+
170
+ def return_values(node)
171
+ # Collect all the (implicit and explicit) return values of a node
172
+ return_values = Set.new(node.begin_type? ? [] : [extract_return_value(node)])
173
+
174
+ node.each_descendant(:return) do |return_node|
175
+ return_values << extract_return_value(return_node)
176
+ end
177
+
178
+ last_value = last_value(node)
179
+ return_values << last_value if last_value
180
+
181
+ process_return_values(return_values)
182
+ end
183
+
184
+ def all_return_values_boolean?(return_values)
185
+ values = return_values.reject { |value| value.type?(:super, :zsuper) }
186
+ return false if values.empty?
187
+
188
+ values.all? { |value| boolean_return?(value) }
189
+ end
190
+
191
+ def boolean_return?(value)
192
+ return true if value.boolean_type?
193
+ return false unless value.call_type?
194
+
195
+ value.comparison_method? || value.predicate_method? || value.negation_method?
196
+ end
197
+
198
+ def potential_non_predicate?(return_values)
199
+ # Assumes a method to be non-predicate if all return values are non-boolean literals.
200
+ #
201
+ # In `Mode: conservative`, if any of the return values is a boolean,
202
+ # the method name is acceptable.
203
+ # In `Mode: aggressive`, all return values must be booleans for a predicate
204
+ # method, or else an offense will be registered.
205
+ return false if conservative? && return_values.any? { |value| boolean_return?(value) }
206
+
207
+ return_values.any? do |value|
208
+ value.literal? && !value.boolean_type?
209
+ end
210
+ end
211
+
212
+ def extract_return_value(node)
213
+ return node unless node.return_type?
214
+
215
+ # `return` without a value is a `nil` return.
216
+ return s(:nil) if node.arguments.empty?
217
+
218
+ # When there's a multiple return, it cannot be a predicate
219
+ # so just return an `array` sexp for simplicity.
220
+ return s(:array) unless node.arguments.one?
221
+
222
+ node.first_argument
223
+ end
224
+
225
+ def last_value(node)
226
+ value = node.begin_type? ? node.children.last : node
227
+ value&.return_type? ? extract_return_value(value) : value
228
+ end
229
+
230
+ def process_return_values(return_values)
231
+ return_values.flat_map do |value|
232
+ if value.conditional?
233
+ process_return_values(extract_conditional_branches(value))
234
+ elsif and_or?(value)
235
+ process_return_values(extract_and_or_clauses(value))
236
+ else
237
+ value
238
+ end
239
+ end
240
+ end
241
+
242
+ def and_or?(node)
243
+ node.type?(:and, :or)
244
+ end
245
+
246
+ def extract_and_or_clauses(node)
247
+ # Recursively traverse an `and` or `or` node to collect all clauses within
248
+ return node unless and_or?(node)
249
+
250
+ [extract_and_or_clauses(node.lhs), extract_and_or_clauses(node.rhs)].flatten
251
+ end
252
+
253
+ def extract_conditional_branches(node)
254
+ return node unless node.conditional?
255
+
256
+ if node.type?(:while, :until)
257
+ # If there is no body, act as implicit `nil`.
258
+ node.body ? [last_value(node.body)] : [s(:nil)]
259
+ else
260
+ # Branches with no value act as an implicit `nil`.
261
+ node.branches.filter_map { |branch| branch ? last_value(branch) : s(:nil) }
262
+ end
263
+ end
264
+
265
+ def conservative?
266
+ cop_config.fetch('Mode', :conservative).to_sym == :conservative
267
+ end
268
+
269
+ def allowed_bang_method?(node)
270
+ return false unless allow_bang_methods?
271
+
272
+ node.bang_method?
273
+ end
274
+
275
+ def allow_bang_methods?
276
+ cop_config.fetch('AllowBangMethods', false)
277
+ end
278
+ end
279
+ end
280
+ end
281
+ end
@@ -100,12 +100,12 @@ module RuboCop
100
100
  # # good
101
101
  # def_node_matcher(:even?) { |value| }
102
102
  #
103
- class PredicateName < Base
103
+ class PredicatePrefix < Base
104
104
  include AllowedMethods
105
105
 
106
106
  # @!method dynamic_method_define(node)
107
107
  def_node_matcher :dynamic_method_define, <<~PATTERN
108
- (send nil? #method_definition_macros
108
+ (send nil? #method_definition_macro?
109
109
  (sym $_)
110
110
  ...)
111
111
  PATTERN
@@ -143,7 +143,7 @@ module RuboCop
143
143
  next if predicate_prefixes.include?(forbidden_prefix)
144
144
 
145
145
  raise ValidationError, <<~MSG.chomp
146
- The `Naming/PredicateName` cop is misconfigured. Prefix #{forbidden_prefix} must be included in NamePrefix because it is included in ForbiddenPrefixes.
146
+ The `Naming/PredicatePrefix` cop is misconfigured. Prefix #{forbidden_prefix} must be included in NamePrefix because it is included in ForbiddenPrefixes.
147
147
  MSG
148
148
  end
149
149
  end
@@ -195,7 +195,7 @@ module RuboCop
195
195
  cop_config['UseSorbetSigs']
196
196
  end
197
197
 
198
- def method_definition_macros(macro_name)
198
+ def method_definition_macro?(macro_name)
199
199
  cop_config['MethodDefinitionMacros'].include?(macro_name.to_s)
200
200
  end
201
201
  end
@@ -269,7 +269,7 @@ module RuboCop
269
269
  end
270
270
 
271
271
  def regexp_with_named_captures?(node)
272
- node.regexp_type? && node.each_capture(named: true).count.positive?
272
+ node.regexp_type? && node.each_capture(named: true).any?
273
273
  end
274
274
  end
275
275
  end
@@ -0,0 +1,167 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Style
6
+ # Prefer `Enumerable` predicate methods over expressions with `count`.
7
+ #
8
+ # The cop checks calls to `count` without arguments, or with a
9
+ # block. It doesn't register offenses for `count` with a positional
10
+ # argument because its behavior differs from predicate methods (`count`
11
+ # matches the argument using `==`, while `any?`, `none?` and `one?` use
12
+ # `===`).
13
+ #
14
+ # NOTE: This cop doesn't check `length` and `size` methods because they
15
+ # would yield false positives. For example, `String` implements `length`
16
+ # and `size`, but it doesn't include `Enumerable`.
17
+ #
18
+ # @safety
19
+ # The cop is unsafe because receiver might not include `Enumerable`, or
20
+ # it has nonstandard implementation of `count` or any replacement
21
+ # methods.
22
+ #
23
+ # It's also unsafe because for collections with falsey values, expressions
24
+ # with `count` without a block return a different result than methods `any?`,
25
+ # `none?` and `one?`:
26
+ #
27
+ # [source,ruby]
28
+ # ----
29
+ # [nil, false].count.positive?
30
+ # [nil].count == 1
31
+ # # => true
32
+ #
33
+ # [nil, false].any?
34
+ # [nil].one?
35
+ # # => false
36
+ #
37
+ # [nil].count == 0
38
+ # # => false
39
+ #
40
+ # [nil].none?
41
+ # # => true
42
+ # ----
43
+ #
44
+ # Autocorrection is unsafe when replacement methods don't iterate over
45
+ # every element in collection and the given block runs side effects:
46
+ #
47
+ # [source,ruby]
48
+ # ----
49
+ # x.count(&:method_with_side_effects).positive?
50
+ # # calls `method_with_side_effects` on every element
51
+ #
52
+ # x.any?(&:method_with_side_effects)
53
+ # # calls `method_with_side_effects` until first element returns a truthy value
54
+ # ----
55
+ #
56
+ # @example
57
+ #
58
+ # # bad
59
+ # x.count.positive?
60
+ # x.count > 0
61
+ # x.count != 0
62
+ #
63
+ # x.count(&:foo?).positive?
64
+ # x.count { |item| item.foo? }.positive?
65
+ #
66
+ # # good
67
+ # x.any?
68
+ #
69
+ # x.any?(&:foo?)
70
+ # x.any? { |item| item.foo? }
71
+ #
72
+ # # bad
73
+ # x.count.zero?
74
+ # x.count == 0
75
+ #
76
+ # # good
77
+ # x.none?
78
+ #
79
+ # # bad
80
+ # x.count == 1
81
+ # x.one?
82
+ #
83
+ # @example AllCops:ActiveSupportExtensionsEnabled: false (default)
84
+ #
85
+ # # good
86
+ # x.count > 1
87
+ #
88
+ # @example AllCops:ActiveSupportExtensionsEnabled: true
89
+ #
90
+ # # bad
91
+ # x.count > 1
92
+ #
93
+ # # good
94
+ # x.many?
95
+ #
96
+ class CollectionQuerying < Base
97
+ include RangeHelp
98
+ extend AutoCorrector
99
+
100
+ MSG = 'Use `%<prefer>s` instead.'
101
+
102
+ RESTRICT_ON_SEND = %i[positive? > != zero? ==].freeze
103
+
104
+ REPLACEMENTS = {
105
+ [:positive?, nil] => :any?,
106
+ [:>, 0] => :any?,
107
+ [:!=, 0] => :any?,
108
+ [:zero?, nil] => :none?,
109
+ [:==, 0] => :none?,
110
+ [:==, 1] => :one?,
111
+ [:>, 1] => :many?
112
+ }.freeze
113
+
114
+ # @!method count_predicate(node)
115
+ def_node_matcher :count_predicate, <<~PATTERN
116
+ (send
117
+ {
118
+ (any_block $(call !nil? :count) _ _)
119
+ $(call !nil? :count (block-pass _)?)
120
+ }
121
+ {
122
+ :positive? |
123
+ :> (int 0) |
124
+ :!= (int 0) |
125
+ :zero? |
126
+ :== (int 0) |
127
+ :== (int 1) |
128
+ :> (int 1)
129
+ })
130
+ PATTERN
131
+
132
+ def on_send(node)
133
+ return unless (count_node = count_predicate(node))
134
+
135
+ replacement_method = replacement_method(node)
136
+
137
+ return unless replacement_supported?(replacement_method)
138
+
139
+ offense_range = count_node.loc.selector.join(node.source_range.end)
140
+ add_offense(offense_range,
141
+ message: format(MSG, prefer: replacement_method)) do |corrector|
142
+ corrector.replace(count_node.loc.selector, replacement_method)
143
+ corrector.remove(removal_range(node))
144
+ end
145
+ end
146
+
147
+ private
148
+
149
+ def replacement_method(node)
150
+ REPLACEMENTS.fetch([node.method_name, node.first_argument&.value])
151
+ end
152
+
153
+ def replacement_supported?(method_name)
154
+ return true if active_support_extensions_enabled?
155
+
156
+ method_name != :many?
157
+ end
158
+
159
+ def removal_range(node)
160
+ range = (node.loc.dot || node.loc.selector).join(node.source_range.end)
161
+
162
+ range_with_surrounding_space(range, side: :left)
163
+ end
164
+ end
165
+ end
166
+ end
167
+ end
@@ -451,7 +451,9 @@ module RuboCop
451
451
  corrector.remove_preceding(condition.loc.else, condition.loc.else.column - column)
452
452
  end
453
453
 
454
- return unless condition.loc.end && !same_line?(condition.loc.end, condition)
454
+ return unless condition.loc.end && !same_line?(
455
+ condition.branches.last.parent.else_branch, condition.loc.end
456
+ )
455
457
 
456
458
  corrector.remove_preceding(condition.loc.end, condition.loc.end.column - column)
457
459
  end
@@ -0,0 +1,100 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Style
6
+ # Checks for empty strings being assigned inside string interpolation.
7
+ #
8
+ # Empty strings are a meaningless outcome inside of string interpolation, so we remove them.
9
+ # Alternatively, when configured to do so, we prioritise using empty strings.
10
+ #
11
+ # While this cop would also apply to variables that are only going to be used as strings,
12
+ # RuboCop can't detect that, so we only check inside of string interpolation.
13
+ #
14
+ # @example EnforcedStyle: trailing_conditional (default)
15
+ # # bad
16
+ # "#{condition ? 'foo' : ''}"
17
+ #
18
+ # # good
19
+ # "#{'foo' if condition}"
20
+ #
21
+ # # bad
22
+ # "#{condition ? '' : 'foo'}"
23
+ #
24
+ # # good
25
+ # "#{'foo' unless condition}"
26
+ #
27
+ # @example EnforcedStyle: ternary
28
+ # # bad
29
+ # "#{'foo' if condition}"
30
+ #
31
+ # # good
32
+ # "#{condition ? 'foo' : ''}"
33
+ #
34
+ # # bad
35
+ # "#{'foo' unless condition}"
36
+ #
37
+ # # good
38
+ # "#{condition ? '' : 'foo'}"
39
+ #
40
+ class EmptyStringInsideInterpolation < Base
41
+ include ConfigurableEnforcedStyle
42
+ include Interpolation
43
+ extend AutoCorrector
44
+
45
+ MSG_TRAILING_CONDITIONAL = 'Do not use trailing conditionals in string interpolation.'
46
+ MSG_TERNARY = 'Do not return empty strings in string interpolation.'
47
+
48
+ # rubocop:disable Metrics/AbcSize, Metrics/MethodLength, Metrics/PerceivedComplexity
49
+ def on_interpolation(node)
50
+ node.each_child_node(:if) do |child_node|
51
+ if style == :trailing_conditional
52
+ if empty_if_outcome?(child_node)
53
+ ternary_style_autocorrect(child_node, child_node.else_branch.source, 'unless')
54
+ end
55
+
56
+ if empty_else_outcome?(child_node)
57
+ ternary_style_autocorrect(child_node, child_node.if_branch.source, 'if')
58
+ end
59
+ elsif style == :ternary
60
+ next unless child_node.modifier_form?
61
+
62
+ ternary_component = if child_node.unless?
63
+ "'' : #{child_node.if_branch.source}"
64
+ else
65
+ "#{child_node.if_branch.source} : ''"
66
+ end
67
+
68
+ add_offense(node, message: MSG_TRAILING_CONDITIONAL) do |corrector|
69
+ corrector.replace(node, "\#{#{child_node.condition.source} ? #{ternary_component}}")
70
+ end
71
+ end
72
+ end
73
+ end
74
+ # rubocop:enable Metrics/AbcSize, Metrics/MethodLength, Metrics/PerceivedComplexity
75
+
76
+ private
77
+
78
+ def empty_if_outcome?(node)
79
+ empty_branch_outcome?(node.if_branch)
80
+ end
81
+
82
+ def empty_else_outcome?(node)
83
+ empty_branch_outcome?(node.else_branch)
84
+ end
85
+
86
+ def empty_branch_outcome?(branch)
87
+ return false unless branch
88
+
89
+ branch.nil_type? || (branch.str_type? && branch.value.empty?)
90
+ end
91
+
92
+ def ternary_style_autocorrect(node, outcome, condition)
93
+ add_offense(node, message: MSG_TERNARY) do |corrector|
94
+ corrector.replace(node, "#{outcome} #{condition} #{node.condition.source}")
95
+ end
96
+ end
97
+ end
98
+ end
99
+ end
100
+ end