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
@@ -42,7 +42,28 @@ module RuboCop
42
42
  # c
43
43
  # end
44
44
  #
45
+ # # bad
46
+ # a.nil? ? true : a
47
+ #
48
+ # # good
49
+ # a.nil? || a
50
+ #
51
+ # # bad
52
+ # if a.nil?
53
+ # true
54
+ # else
55
+ # a
56
+ # end
57
+ #
58
+ # # good
59
+ # a.nil? || a
60
+ #
61
+ # @example AllowedMethods: ['nonzero?'] (default)
62
+ # # good
63
+ # num.nonzero? ? true : false
64
+ #
45
65
  class RedundantCondition < Base
66
+ include AllowedMethods
46
67
  include CommentsHelp
47
68
  include RangeHelp
48
69
  extend AutoCorrector
@@ -128,6 +149,16 @@ module RuboCop
128
149
  # end
129
150
  return true if condition == if_branch
130
151
 
152
+ # e.g.
153
+ # a.nil? ? true : a
154
+ # or
155
+ # if a.nil?
156
+ # true
157
+ # else
158
+ # a
159
+ # end
160
+ return true if if_branch_is_true_type_and_else_is_not?(node)
161
+
131
162
  # e.g.
132
163
  # if foo
133
164
  # @value = foo
@@ -146,6 +177,18 @@ module RuboCop
146
177
  !use_hash_key_access?(if_branch)
147
178
  end
148
179
 
180
+ # rubocop:disable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity
181
+ def if_branch_is_true_type_and_else_is_not?(node)
182
+ return false unless node.ternary? || node.if?
183
+
184
+ cond = node.condition
185
+ return false unless cond.call_type?
186
+ return false if !cond.predicate_method? || allowed_method?(cond.method_name)
187
+
188
+ node.if_branch&.true_type? && node.else_branch && !node.else_branch.true_type?
189
+ end
190
+ # rubocop:enable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity
191
+
149
192
  def branches_have_assignment?(node)
150
193
  _condition, if_branch, else_branch = *node # rubocop:disable InternalAffairs/NodeDestructuring
151
194
 
@@ -194,6 +237,8 @@ module RuboCop
194
237
  argument_source = if_branch.first_argument.source
195
238
 
196
239
  "#{if_branch.receiver.source} #{if_branch.method_name} (#{argument_source}"
240
+ elsif if_branch.true_type?
241
+ if_branch.parent.condition.source
197
242
  else
198
243
  if_branch.source
199
244
  end
@@ -20,20 +20,30 @@ module RuboCop
20
20
 
21
21
  MSG = 'Remove the redundant current directory path.'
22
22
  RESTRICT_ON_SEND = %i[require_relative].freeze
23
- CURRENT_DIRECTORY_PATH = './'
23
+ CURRENT_DIRECTORY_PREFIX = %r{./+}.freeze
24
+ REDUNDANT_CURRENT_DIRECTORY_PREFIX = /\A#{CURRENT_DIRECTORY_PREFIX}/.freeze
24
25
 
25
26
  def on_send(node)
26
27
  return unless (first_argument = node.first_argument)
27
- return unless first_argument.str_content&.start_with?(CURRENT_DIRECTORY_PATH)
28
- return unless (index = first_argument.source.index(CURRENT_DIRECTORY_PATH))
28
+ return unless (index = first_argument.source.index(CURRENT_DIRECTORY_PREFIX))
29
+ return unless (redundant_length = redundant_path_length(first_argument.str_content))
29
30
 
30
31
  begin_pos = first_argument.source_range.begin.begin_pos + index
31
- range = range_between(begin_pos, begin_pos + 2)
32
+ end_pos = begin_pos + redundant_length
33
+ range = range_between(begin_pos, end_pos)
32
34
 
33
35
  add_offense(range) do |corrector|
34
36
  corrector.remove(range)
35
37
  end
36
38
  end
39
+
40
+ private
41
+
42
+ def redundant_path_length(path)
43
+ return unless (match = path&.match(REDUNDANT_CURRENT_DIRECTORY_PREFIX))
44
+
45
+ match[0].length
46
+ end
37
47
  end
38
48
  end
39
49
  end
@@ -0,0 +1,250 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Style
6
+ # Checks for calls to `Kernel#format` or `Kernel#sprintf` that are redundant.
7
+ #
8
+ # Calling `format` with only a single string argument is redundant, as it can be
9
+ # replaced by the string itself.
10
+ #
11
+ # Also looks for `format` calls where the arguments are literals that can be
12
+ # inlined into a string easily. This applies to the `%s`, `%d`, `%i`, `%u`, and
13
+ # `%f` format specifiers.
14
+ #
15
+ # @safety
16
+ # This cop's autocorrection is unsafe because string object returned by
17
+ # `format` and `sprintf` are never frozen. If `format('string')` is autocorrected to
18
+ # `'string'`, `FrozenError` may occur when calling a destructive method like `String#<<`.
19
+ # Consider using `'string'.dup` instead of `format('string')`.
20
+ # Additionally, since the necessity of `dup` cannot be determined automatically,
21
+ # this autocorrection is inherently unsafe.
22
+ #
23
+ # [source,ruby]
24
+ # ----
25
+ # # frozen_string_literal: true
26
+ #
27
+ # format('template').frozen? # => false
28
+ # 'template'.frozen? # => true
29
+ # ----
30
+ #
31
+ # @example
32
+ #
33
+ # # bad
34
+ # format('the quick brown fox jumps over the lazy dog.')
35
+ # sprintf('the quick brown fox jumps over the lazy dog.')
36
+ #
37
+ # # good
38
+ # 'the quick brown fox jumps over the lazy dog.'
39
+ #
40
+ # # bad
41
+ # format('%s %s', 'foo', 'bar')
42
+ # sprintf('%s %s', 'foo', 'bar')
43
+ #
44
+ # # good
45
+ # 'foo bar'
46
+ #
47
+ class RedundantFormat < Base
48
+ extend AutoCorrector
49
+
50
+ MSG = 'Use `%<prefer>s` directly instead of `%<method_name>s`.'
51
+
52
+ RESTRICT_ON_SEND = %i[format sprintf].to_set.freeze
53
+ ACCEPTABLE_LITERAL_TYPES = %i[str dstr sym dsym numeric boolean nil].freeze
54
+
55
+ # @!method format_without_additional_args?(node)
56
+ def_node_matcher :format_without_additional_args?, <<~PATTERN
57
+ (send {(const {nil? cbase} :Kernel) nil?} %RESTRICT_ON_SEND ${str dstr})
58
+ PATTERN
59
+
60
+ # @!method rational_number?(node)
61
+ def_node_matcher :rational_number?, <<~PATTERN
62
+ {rational (send int :/ rational) (begin rational) (begin (send int :/ rational))}
63
+ PATTERN
64
+
65
+ # @!method complex_number?(node)
66
+ def_node_matcher :complex_number?, <<~PATTERN
67
+ {complex (send int :+ complex) (begin complex) (begin (send int :+ complex))}
68
+ PATTERN
69
+
70
+ # @!method find_hash_value_node(node, name)
71
+ def_node_search :find_hash_value_node, <<~PATTERN
72
+ (pair (sym %1) $_)
73
+ PATTERN
74
+
75
+ # @!method splatted_arguments?(node)
76
+ def_node_matcher :splatted_arguments?, <<~PATTERN
77
+ (send _ %RESTRICT_ON_SEND <{
78
+ splat
79
+ (hash <kwsplat ...>)
80
+ } ...>)
81
+ PATTERN
82
+
83
+ def on_send(node)
84
+ format_without_additional_args?(node) do |value|
85
+ replacement = value.source
86
+
87
+ add_offense(node, message: message(node, replacement)) do |corrector|
88
+ corrector.replace(node, replacement)
89
+ end
90
+ return
91
+ end
92
+
93
+ detect_unnecessary_fields(node)
94
+ end
95
+
96
+ private
97
+
98
+ def message(node, prefer)
99
+ format(MSG, prefer: prefer, method_name: node.method_name)
100
+ end
101
+
102
+ def detect_unnecessary_fields(node)
103
+ return unless node.first_argument&.str_type?
104
+
105
+ string = node.first_argument.value
106
+ arguments = node.arguments[1..]
107
+
108
+ return unless string && arguments.any?
109
+ return if splatted_arguments?(node)
110
+
111
+ register_all_fields_literal(node, string, arguments)
112
+ end
113
+
114
+ def register_all_fields_literal(node, string, arguments)
115
+ return unless all_fields_literal?(string, arguments.dup)
116
+
117
+ formatted_string = format(string, *argument_values(arguments))
118
+ replacement = quote(formatted_string, node)
119
+
120
+ add_offense(node, message: message(node, replacement)) do |corrector|
121
+ corrector.replace(node, replacement)
122
+ end
123
+ end
124
+
125
+ def all_fields_literal?(string, arguments)
126
+ count = 0
127
+ sequences = RuboCop::Cop::Utils::FormatString.new(string).format_sequences
128
+ return false unless sequences.any?
129
+
130
+ sequences.each do |sequence|
131
+ next if sequence.percent?
132
+
133
+ hash = arguments.detect(&:hash_type?)
134
+ next unless (argument = find_argument(sequence, arguments, hash))
135
+ next unless matching_argument?(sequence, argument)
136
+
137
+ count += 1
138
+ end
139
+
140
+ sequences.size == count
141
+ end
142
+
143
+ def find_argument(sequence, arguments, hash)
144
+ if hash && (sequence.annotated? || sequence.template?)
145
+ find_hash_value_node(hash, sequence.name.to_sym).first
146
+ elsif sequence.arg_number
147
+ arguments[sequence.arg_number.to_i - 1]
148
+ else
149
+ # If the specifier contains `*`, the following arguments will be used
150
+ # to specify the width and can be ignored.
151
+ (sequence.arity - 1).times { arguments.shift }
152
+ arguments.shift
153
+ end
154
+ end
155
+
156
+ def matching_argument?(sequence, argument)
157
+ # Template specifiers don't give a type, any acceptable literal type is ok.
158
+ return argument.type?(*ACCEPTABLE_LITERAL_TYPES) if sequence.template?
159
+
160
+ # An argument matches a specifier if it can be easily converted
161
+ # to that type.
162
+ case sequence.type
163
+ when 's'
164
+ argument.type?(*ACCEPTABLE_LITERAL_TYPES)
165
+ when 'd', 'i', 'u'
166
+ integer?(argument)
167
+ when 'f'
168
+ float?(argument)
169
+ else
170
+ false
171
+ end
172
+ end
173
+
174
+ def numeric?(argument)
175
+ argument.type?(:numeric, :str) ||
176
+ rational_number?(argument) ||
177
+ complex_number?(argument)
178
+ end
179
+
180
+ def integer?(argument)
181
+ numeric?(argument) && Integer(argument_value(argument), exception: false)
182
+ end
183
+
184
+ def float?(argument)
185
+ numeric?(argument) && Float(argument_value(argument), exception: false)
186
+ end
187
+
188
+ # Add correct quotes to the formatted string, preferring retaining the existing
189
+ # quotes if possible.
190
+ def quote(string, node)
191
+ str_node = node.first_argument
192
+ start_delimiter = str_node.loc.begin.source
193
+ end_delimiter = str_node.loc.end.source
194
+
195
+ # If there is any interpolation, the delimiters need to be changed potentially
196
+ if node.each_descendant(:dstr, :dsym).any?
197
+ case start_delimiter
198
+ when "'"
199
+ start_delimiter = end_delimiter = '"'
200
+ when /\A%q(.)/
201
+ start_delimiter = "%Q#{Regexp.last_match[1]}"
202
+ end
203
+ end
204
+
205
+ "#{start_delimiter}#{string}#{end_delimiter}"
206
+ end
207
+
208
+ def argument_values(arguments)
209
+ arguments.map { |argument| argument_value(argument) }
210
+ end
211
+
212
+ def argument_value(argument)
213
+ argument = argument.children.first if argument.begin_type?
214
+
215
+ if argument.dsym_type?
216
+ dsym_value(argument)
217
+ elsif argument.hash_type?
218
+ hash_value(argument)
219
+ elsif rational_number?(argument)
220
+ rational_value(argument)
221
+ elsif complex_number?(argument)
222
+ complex_value(argument)
223
+ elsif argument.respond_to?(:value)
224
+ argument.value
225
+ else
226
+ argument.source
227
+ end
228
+ end
229
+
230
+ def dsym_value(dsym_node)
231
+ dsym_node.children.first.source
232
+ end
233
+
234
+ def hash_value(hash_node)
235
+ hash_node.each_pair.with_object({}) do |pair, hash|
236
+ hash[pair.key.value] = argument_value(pair.value)
237
+ end
238
+ end
239
+
240
+ def rational_value(rational_node)
241
+ rational_node.source.to_r
242
+ end
243
+
244
+ def complex_value(complex_node)
245
+ Complex(complex_node.source)
246
+ end
247
+ end
248
+ end
249
+ end
250
+ end
@@ -60,7 +60,7 @@ module RuboCop
60
60
  (begin (send !{(str _) array} {:+ :- :* :** :/ :%} {float int}))
61
61
  (begin (send _ {:== :=== :!= :<= :>= :< :>} _))
62
62
  (send _ {:count :length :size} ...)
63
- (block (send _ {:count :length :size} ...) ...)
63
+ (any_block (send _ {:count :length :size} ...) ...)
64
64
  }
65
65
  PATTERN
66
66
  end
@@ -13,14 +13,16 @@ module RuboCop
13
13
  # # good
14
14
  # x if y.z.nil?
15
15
  #
16
- class RedundantParentheses < Base
16
+ class RedundantParentheses < Base # rubocop:disable Metrics/ClassLength
17
17
  include Parentheses
18
18
  extend AutoCorrector
19
19
 
20
20
  ALLOWED_NODE_TYPES = %i[and or send splat kwsplat].freeze
21
21
 
22
22
  # @!method square_brackets?(node)
23
- def_node_matcher :square_brackets?, '(send {(send _recv _msg) str array hash} :[] ...)'
23
+ def_node_matcher :square_brackets?, <<~PATTERN
24
+ (send `{(send _recv _msg) str array hash const #variable?} :[] ...)
25
+ PATTERN
24
26
 
25
27
  # @!method method_node_and_args(node)
26
28
  def_node_matcher :method_node_and_args, '$(call _recv _msg $...)'
@@ -39,6 +41,10 @@ module RuboCop
39
41
 
40
42
  private
41
43
 
44
+ def variable?(node)
45
+ node.respond_to?(:variable?) && node.variable?
46
+ end
47
+
42
48
  def parens_allowed?(node)
43
49
  empty_parentheses?(node) ||
44
50
  first_arg_begins_with_hash_literal?(node) ||
@@ -128,6 +134,8 @@ module RuboCop
128
134
  node = begin_node.children.first
129
135
 
130
136
  if (message = find_offense_message(begin_node, node))
137
+ begin_node = begin_node.parent if node.range_type?
138
+
131
139
  return offense(begin_node, message)
132
140
  end
133
141
 
@@ -137,7 +145,7 @@ module RuboCop
137
145
  # rubocop:disable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/MethodLength, Metrics/PerceivedComplexity
138
146
  def find_offense_message(begin_node, node)
139
147
  return 'a keyword' if keyword_with_redundant_parentheses?(node)
140
- return 'a literal' if disallowed_literal?(begin_node, node)
148
+ return 'a literal' if node.literal? && disallowed_literal?(begin_node, node)
141
149
  return 'a variable' if node.variable?
142
150
  return 'a constant' if node.const_type?
143
151
  if node.assignment? && (begin_node.parent.nil? || begin_node.parent.begin_type?)
@@ -207,7 +215,13 @@ module RuboCop
207
215
  end
208
216
 
209
217
  def disallowed_literal?(begin_node, node)
210
- node.literal? && !node.range_type? && !raised_to_power_negative_numeric?(begin_node, node)
218
+ if node.range_type?
219
+ return false unless (parent = begin_node.parent)
220
+
221
+ parent.begin_type? && parent.children.one?
222
+ else
223
+ !raised_to_power_negative_numeric?(begin_node, node)
224
+ end
211
225
  end
212
226
 
213
227
  def raised_to_power_negative_numeric?(begin_node, node)
@@ -58,7 +58,7 @@ module RuboCop
58
58
  # rubocop:disable Metrics/AbcSize
59
59
  def on_lvasgn(node)
60
60
  return unless (rhs = node.rhs)
61
- return unless rhs.call_type? && method_returning_self?(rhs.method_name)
61
+ return unless rhs.type?(:any_block, :call) && method_returning_self?(rhs.method_name)
62
62
  return unless (receiver = rhs.receiver)
63
63
 
64
64
  receiver_type = ASSIGNMENT_TYPE_TO_RECEIVER_TYPE[node.type]
@@ -67,11 +67,13 @@ module RuboCop
67
67
  node.parent && parentheses?(node.parent)
68
68
  end
69
69
 
70
+ # rubocop:disable Metrics/AbcSize
70
71
  def correct_rescue_block(corrector, node, parenthesized)
71
72
  operation = node.body
72
73
 
73
74
  node_indentation, node_offset = indentation_and_offset(node, parenthesized)
74
75
 
76
+ corrector.wrap(operation, '[', ']') if operation.array_type? && !operation.bracketed?
75
77
  corrector.remove(range_between(operation.source_range.end_pos, node.source_range.end_pos))
76
78
  corrector.insert_before(operation, "begin\n#{node_indentation}")
77
79
  corrector.insert_after(heredoc_end(operation) || operation, <<~RESCUE_CLAUSE.chop)
@@ -81,6 +83,7 @@ module RuboCop
81
83
  #{node_offset}end
82
84
  RESCUE_CLAUSE
83
85
  end
86
+ # rubocop:enable Metrics/AbcSize
84
87
 
85
88
  def indentation_and_offset(node, parenthesized)
86
89
  node_indentation = indentation(node)
@@ -8,9 +8,9 @@ module RuboCop
8
8
  #
9
9
  # Endless methods added in Ruby 3.0 are also accepted by this cop.
10
10
  #
11
- # If `Style/EndlessMethod` is enabled with `EnforcedStyle: allow_single_line` or
12
- # `allow_always`, single-line methods will be autocorrected to endless
13
- # methods if there is only one statement in the body.
11
+ # If `Style/EndlessMethod` is enabled with `EnforcedStyle: allow_single_line`, `allow_always`,
12
+ # `require_single_line`, or `require_always`, single-line methods will be autocorrected
13
+ # to endless methods if there is only one statement in the body.
14
14
  #
15
15
  # @example
16
16
  # # bad
@@ -230,12 +230,6 @@ module RuboCop
230
230
  !condition.comparison_method?
231
231
  end
232
232
 
233
- def arguments_range(node)
234
- range_between(
235
- node.first_argument.source_range.begin_pos, node.last_argument.source_range.end_pos
236
- )
237
- end
238
-
239
233
  def wrap_condition?(node)
240
234
  node.operator_keyword? || (node.call_type? && node.arguments.any? && !node.parenthesized?)
241
235
  end
@@ -128,7 +128,7 @@ module RuboCop
128
128
  end
129
129
 
130
130
  def uncorrectable?(part)
131
- part.multiline? || heredoc?(part) || part.each_descendant(:block).any?
131
+ part.multiline? || heredoc?(part) || part.each_descendant(:any_block).any?
132
132
  end
133
133
 
134
134
  def heredoc?(node)
@@ -6,12 +6,13 @@ module RuboCop
6
6
  # Checks for trailing comma in array literals.
7
7
  # The configuration options are:
8
8
  #
9
- # * `consistent_comma`: Requires a comma after the
10
- # last item of all non-empty, multiline array literals.
11
- # * `comma`: Requires a comma after last item in an array,
12
- # but only when each item is on its own line.
13
- # * `no_comma`: Does not require a comma after the
14
- # last item in an array
9
+ # * `consistent_comma`: Requires a comma after the last item of all non-empty, multiline array
10
+ # literals.
11
+ # * `comma`: Requires a comma after the last item in an array, but only when each item is on
12
+ # its own line.
13
+ # * `diff_comma`: Requires a comma after the last item in an array, but only when that item is
14
+ # followed by an immediate newline.
15
+ # * `no_comma`: Does not require a comma after the last item in an array
15
16
  #
16
17
  # @example EnforcedStyleForMultiline: consistent_comma
17
18
  # # bad
@@ -37,6 +38,14 @@ module RuboCop
37
38
  # 2,
38
39
  # ]
39
40
  #
41
+ # # bad
42
+ # a = [1, 2,
43
+ # 3, 4]
44
+ #
45
+ # # good
46
+ # a = [1, 2,
47
+ # 3, 4,]
48
+ #
40
49
  # @example EnforcedStyleForMultiline: comma
41
50
  # # bad
42
51
  # a = [1, 2,]
@@ -72,6 +81,38 @@ module RuboCop
72
81
  # 2,
73
82
  # ]
74
83
  #
84
+ # @example EnforcedStyleForMultiline: diff_comma
85
+ # # bad
86
+ # a = [1, 2,]
87
+ #
88
+ # # good
89
+ # a = [1, 2]
90
+ #
91
+ # # good
92
+ # a = [
93
+ # 1, 2,
94
+ # 3,
95
+ # ]
96
+ #
97
+ # # good
98
+ # a = [
99
+ # 1, 2, 3,
100
+ # ]
101
+ #
102
+ # # good
103
+ # a = [
104
+ # 1,
105
+ # 2,
106
+ # ]
107
+ #
108
+ # # bad
109
+ # a = [1, 2,
110
+ # 3, 4,]
111
+ #
112
+ # # good
113
+ # a = [1, 2,
114
+ # 3, 4]
115
+ #
75
116
  # @example EnforcedStyleForMultiline: no_comma (default)
76
117
  # # bad
77
118
  # a = [1, 2,]
@@ -6,12 +6,13 @@ module RuboCop
6
6
  # Checks for trailing comma in hash literals.
7
7
  # The configuration options are:
8
8
  #
9
- # * `consistent_comma`: Requires a comma after the
10
- # last item of all non-empty, multiline hash literals.
11
- # * `comma`: Requires a comma after the last item in a hash,
12
- # but only when each item is on its own line.
13
- # * `no_comma`: Does not require a comma after the
14
- # last item in a hash
9
+ # * `consistent_comma`: Requires a comma after the last item of all non-empty, multiline hash
10
+ # literals.
11
+ # * `comma`: Requires a comma after the last item in a hash, but only when each item is on its
12
+ # own line.
13
+ # * `diff_comma`: Requires a comma after the last item in a hash, but only when that item is
14
+ # followed by an immediate newline.
15
+ # * `no_comma`: Does not require a comma after the last item in a hash
15
16
  #
16
17
  # @example EnforcedStyleForMultiline: consistent_comma
17
18
  #
@@ -38,6 +39,14 @@ module RuboCop
38
39
  # bar: 2,
39
40
  # }
40
41
  #
42
+ # # bad
43
+ # a = { foo: 1, bar: 2,
44
+ # baz: 3, qux: 4 }
45
+ #
46
+ # # good
47
+ # a = { foo: 1, bar: 2,
48
+ # baz: 3, qux: 4, }
49
+ #
41
50
  # @example EnforcedStyleForMultiline: comma
42
51
  #
43
52
  # # bad
@@ -74,6 +83,39 @@ module RuboCop
74
83
  # bar: 2,
75
84
  # }
76
85
  #
86
+ # @example EnforcedStyleForMultiline: diff_comma
87
+ #
88
+ # # bad
89
+ # a = { foo: 1, bar: 2, }
90
+ #
91
+ # # good
92
+ # a = { foo: 1, bar: 2 }
93
+ #
94
+ # # good
95
+ # a = {
96
+ # foo: 1, bar: 2,
97
+ # qux: 3,
98
+ # }
99
+ #
100
+ # # good
101
+ # a = {
102
+ # foo: 1, bar: 2, qux: 3,
103
+ # }
104
+ #
105
+ # # good
106
+ # a = {
107
+ # foo: 1,
108
+ # bar: 2,
109
+ # }
110
+ #
111
+ # # bad
112
+ # a = { foo: 1, bar: 2,
113
+ # baz: 3, qux: 4, }
114
+ #
115
+ # # good
116
+ # a = { foo: 1, bar: 2,
117
+ # baz: 3, qux: 4 }
118
+ #
77
119
  # @example EnforcedStyleForMultiline: no_comma (default)
78
120
  #
79
121
  # # bad
@@ -113,7 +113,7 @@ module RuboCop
113
113
  private
114
114
 
115
115
  def in_module_or_instance_eval?(node)
116
- node.each_ancestor(:block, :class, :sclass, :module).each do |pnode|
116
+ node.each_ancestor(:any_block, :class, :sclass, :module).each do |pnode|
117
117
  case pnode.type
118
118
  when :class, :sclass
119
119
  return false
@@ -32,7 +32,7 @@ module RuboCop
32
32
  end
33
33
 
34
34
  def parentheses?(node)
35
- node.loc.respond_to?(:end) && node.loc.end&.is?(')')
35
+ node.loc_is?(:end, ')')
36
36
  end
37
37
 
38
38
  # rubocop:disable Metrics/AbcSize, Metrics/MethodLength