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
@@ -81,21 +81,16 @@ module RuboCop
81
81
  (node.numeric_type? && node.value.zero?) || node.nil_type?
82
82
  end
83
83
 
84
- # rubocop:disable Metrics/PerceivedComplexity
85
84
  def check_send(node)
86
85
  if node.arithmetic_operation?
87
86
  float?(node.receiver) || float?(node.first_argument)
88
87
  elsif FLOAT_RETURNING_METHODS.include?(node.method_name)
89
88
  true
90
89
  elsif node.receiver&.float_type?
91
- if FLOAT_INSTANCE_METHODS.include?(node.method_name)
92
- true
93
- else
90
+ FLOAT_INSTANCE_METHODS.include?(node.method_name) ||
94
91
  check_numeric_returning_method(node)
95
- end
96
92
  end
97
93
  end
98
- # rubocop:enable Metrics/PerceivedComplexity
99
94
 
100
95
  def check_numeric_returning_method(node)
101
96
  return false unless node.receiver
@@ -7,7 +7,7 @@ module RuboCop
7
7
  # expected fields for format/sprintf/#% and what is actually
8
8
  # passed as arguments.
9
9
  #
10
- # In addition it checks whether different formats are used in the same
10
+ # In addition, it checks whether different formats are used in the same
11
11
  # format string. Do not mix numbered, unnumbered, and named formats in
12
12
  # the same format string.
13
13
  #
@@ -18,12 +18,15 @@ module RuboCop
18
18
  # end
19
19
  #
20
20
  # # bad
21
- # if some_var && true
21
+ # # We're only interested in the left hand side being a truthy literal,
22
+ # # because it affects the evaluation of the &&, whereas the right hand
23
+ # # side will be conditionally executed/called and can be a literal.
24
+ # if true && some_var
22
25
  # do_something
23
26
  # end
24
27
  #
25
28
  # # good
26
- # if some_var && some_condition
29
+ # if some_var
27
30
  # do_something
28
31
  # end
29
32
  #
@@ -34,27 +37,90 @@ module RuboCop
34
37
  # end
35
38
  class LiteralAsCondition < Base
36
39
  include RangeHelp
40
+ extend AutoCorrector
37
41
 
38
42
  MSG = 'Literal `%<literal>s` appeared as a condition.'
39
43
  RESTRICT_ON_SEND = [:!].freeze
40
44
 
45
+ def on_and(node)
46
+ return unless node.lhs.truthy_literal?
47
+
48
+ add_offense(node.lhs) do |corrector|
49
+ corrector.replace(node, node.rhs.source)
50
+ end
51
+ end
52
+
41
53
  def on_if(node)
42
- check_for_literal(node)
54
+ cond = condition(node)
55
+
56
+ if node.unless?
57
+ correct_if_node(node, cond, true) if cond.falsey_literal?
58
+ correct_if_node(node, cond, false) if cond.truthy_literal?
59
+ else
60
+ correct_if_node(node, cond, true) if cond.truthy_literal?
61
+ correct_if_node(node, cond, false) if cond.falsey_literal?
62
+ end
43
63
  end
44
64
 
45
65
  def on_while(node)
46
- return if condition(node).true_type?
66
+ return if node.condition.source == 'true'
47
67
 
48
- check_for_literal(node)
68
+ if node.condition.truthy_literal?
69
+ add_offense(node.condition) do |corrector|
70
+ corrector.replace(node.condition, 'true')
71
+ end
72
+ elsif node.condition.falsey_literal?
73
+ add_offense(node.condition) do |corrector|
74
+ corrector.remove(node)
75
+ end
76
+ end
77
+ end
78
+
79
+ # rubocop:disable Metrics/AbcSize
80
+ def on_while_post(node)
81
+ return if node.condition.source == 'true'
82
+
83
+ if node.condition.truthy_literal?
84
+ add_offense(node.condition) do |corrector|
85
+ corrector.replace(node, node.source.sub(node.condition.source, 'true'))
86
+ end
87
+ elsif node.condition.falsey_literal?
88
+ add_offense(node.condition) do |corrector|
89
+ corrector.replace(node, node.body.child_nodes.map(&:source).join("\n"))
90
+ end
91
+ end
49
92
  end
50
- alias on_while_post on_while
93
+ # rubocop:enable Metrics/AbcSize
51
94
 
52
95
  def on_until(node)
53
- return if condition(node).false_type?
96
+ return if node.condition.source == 'false'
54
97
 
55
- check_for_literal(node)
98
+ if node.condition.falsey_literal?
99
+ add_offense(node.condition) do |corrector|
100
+ corrector.replace(node.condition, 'false')
101
+ end
102
+ elsif node.condition.truthy_literal?
103
+ add_offense(node.condition) do |corrector|
104
+ corrector.remove(node)
105
+ end
106
+ end
107
+ end
108
+
109
+ # rubocop:disable Metrics/AbcSize
110
+ def on_until_post(node)
111
+ return if node.condition.source == 'false'
112
+
113
+ if node.condition.falsey_literal?
114
+ add_offense(node.condition) do |corrector|
115
+ corrector.replace(node, node.source.sub(node.condition.source, 'false'))
116
+ end
117
+ elsif node.condition.truthy_literal?
118
+ add_offense(node.condition) do |corrector|
119
+ corrector.replace(node, node.body.child_nodes.map(&:source).join("\n"))
120
+ end
121
+ end
56
122
  end
57
- alias on_until_post on_until
123
+ # rubocop:enable Metrics/AbcSize
58
124
 
59
125
  def on_case(case_node)
60
126
  if case_node.condition
@@ -130,6 +196,8 @@ module RuboCop
130
196
 
131
197
  def handle_node(node)
132
198
  if node.literal?
199
+ return if node.parent.and_type?
200
+
133
201
  add_offense(node)
134
202
  elsif %i[send and or begin].include?(node.type)
135
203
  check_node(node)
@@ -159,6 +227,28 @@ module RuboCop
159
227
  when_node.conditions.last.source_range.end_pos
160
228
  )
161
229
  end
230
+
231
+ # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
232
+ def correct_if_node(node, cond, result)
233
+ if result
234
+ add_offense(cond) do |corrector|
235
+ corrector.replace(node, node.if_branch.source)
236
+ end
237
+ elsif node.elsif_conditional?
238
+ add_offense(cond) do |corrector|
239
+ corrector.replace(node, "#{node.else_branch.source.sub('elsif', 'if')}\nend")
240
+ end
241
+ elsif node.else? || node.ternary?
242
+ add_offense(cond) do |corrector|
243
+ corrector.replace(node, node.else_branch.source)
244
+ end
245
+ else
246
+ add_offense(cond) do |corrector|
247
+ corrector.remove(node)
248
+ end
249
+ end
250
+ end
251
+ # rubocop:enable Metrics/AbcSize, Metrics/MethodLength
162
252
  end
163
253
  end
164
254
  end
@@ -79,7 +79,7 @@ module RuboCop
79
79
  end
80
80
 
81
81
  def range_pairs(expr)
82
- RuboCop::Cop::Utils::RegexpRanges.new(expr).pairs
82
+ expr.expressions.filter_map { |e| [e.expressions[0], e.expressions[1]] if e.type == :set }
83
83
  end
84
84
 
85
85
  def unsafe_range?(range_start, range_end)
@@ -94,7 +94,7 @@ module RuboCop
94
94
 
95
95
  def skip_range?(range_start, range_end)
96
96
  [range_start, range_end].any? do |bound|
97
- bound.type != :literal
97
+ bound&.type != :literal
98
98
  end
99
99
  end
100
100
 
@@ -17,17 +17,12 @@ module RuboCop
17
17
  # * 2.0+ ... `enumerator`
18
18
  # * 2.1+ ... `thread`
19
19
  # * 2.2+ ... Add `rational` and `complex` above
20
- # * 2.5+ ... Add `pp` above
21
20
  # * 2.7+ ... Add `ruby2_keywords` above
22
21
  # * 3.1+ ... Add `fiber` above
23
22
  # * 3.2+ ... `set`
24
23
  #
25
24
  # This cop target those features.
26
25
  #
27
- # @safety
28
- # This cop's autocorrection is unsafe because if `require 'pp'` is removed from one file,
29
- # `NameError` can be encountered when another file uses `PP.pp`.
30
- #
31
26
  # @example
32
27
  # # bad
33
28
  # require 'unloaded_feature'
@@ -42,10 +37,6 @@ module RuboCop
42
37
  MSG = 'Remove unnecessary `require` statement.'
43
38
  RESTRICT_ON_SEND = %i[require].freeze
44
39
  RUBY_22_LOADED_FEATURES = %w[rational complex].freeze
45
- PRETTY_PRINT_METHODS = %i[
46
- pretty_inspect pretty_print pretty_print_cycle
47
- pretty_print_inspect pretty_print_instance_variables
48
- ].freeze
49
40
 
50
41
  # @!method redundant_require_statement?(node)
51
42
  def_node_matcher :redundant_require_statement?, <<~PATTERN
@@ -53,11 +44,6 @@ module RuboCop
53
44
  (str #redundant_feature?))
54
45
  PATTERN
55
46
 
56
- # @!method pp_const?(node)
57
- def_node_matcher :pp_const?, <<~PATTERN
58
- (const {nil? cbase} :PP)
59
- PATTERN
60
-
61
47
  def on_send(node)
62
48
  return unless redundant_require_statement?(node)
63
49
 
@@ -81,18 +67,11 @@ module RuboCop
81
67
  feature_name == 'enumerator' ||
82
68
  (target_ruby_version >= 2.1 && feature_name == 'thread') ||
83
69
  (target_ruby_version >= 2.2 && RUBY_22_LOADED_FEATURES.include?(feature_name)) ||
84
- (target_ruby_version >= 2.5 && feature_name == 'pp' && !need_to_require_pp?) ||
85
70
  (target_ruby_version >= 2.7 && feature_name == 'ruby2_keywords') ||
86
71
  (target_ruby_version >= 3.1 && feature_name == 'fiber') ||
87
72
  (target_ruby_version >= 3.2 && feature_name == 'set')
88
73
  end
89
74
  # rubocop:enable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity
90
-
91
- def need_to_require_pp?
92
- processed_source.ast.each_descendant(:send).any? do |node|
93
- pp_const?(node.receiver) || PRETTY_PRINT_METHODS.include?(node.method_name)
94
- end
95
- end
96
75
  end
97
76
  end
98
77
  end
@@ -0,0 +1,252 @@
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_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()`, `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_r: 'rational_constructor?',
102
+ to_c: 'complex_constructor?',
103
+ to_a: 'array_constructor?',
104
+ to_h: 'hash_constructor?',
105
+ to_set: 'set_constructor?'
106
+ }.freeze
107
+
108
+ # Methods that already are expected to return a given type, which makes a further
109
+ # conversion redundant.
110
+ TYPED_METHODS = { to_s: %i[inspect] }.freeze
111
+
112
+ CONVERSION_METHODS = Set[*LITERAL_NODE_TYPES.keys].freeze
113
+ RESTRICT_ON_SEND = CONVERSION_METHODS
114
+
115
+ private_constant :LITERAL_NODE_TYPES, :CONSTRUCTOR_MAPPING
116
+
117
+ # @!method type_constructor?(node, type_symbol)
118
+ def_node_matcher :type_constructor?, <<~PATTERN
119
+ (send {nil? (const {cbase nil?} :Kernel)} %1 ...)
120
+ PATTERN
121
+
122
+ # @!method string_constructor?(node)
123
+ def_node_matcher :string_constructor?, <<~PATTERN
124
+ {
125
+ (send (const {cbase nil?} :String) :new ...)
126
+ #type_constructor?(:String)
127
+ }
128
+ PATTERN
129
+
130
+ # @!method integer_constructor?(node)
131
+ def_node_matcher :integer_constructor?, <<~PATTERN
132
+ #type_constructor?(:Integer)
133
+ PATTERN
134
+
135
+ # @!method float_constructor?(node)
136
+ def_node_matcher :float_constructor?, <<~PATTERN
137
+ #type_constructor?(:Float)
138
+ PATTERN
139
+
140
+ # @!method rational_constructor?(node)
141
+ def_node_matcher :rational_constructor?, <<~PATTERN
142
+ #type_constructor?(:Rational)
143
+ PATTERN
144
+
145
+ # @!method complex_constructor?(node)
146
+ def_node_matcher :complex_constructor?, <<~PATTERN
147
+ #type_constructor?(:Complex)
148
+ PATTERN
149
+
150
+ # @!method array_constructor?(node)
151
+ def_node_matcher :array_constructor?, <<~PATTERN
152
+ {
153
+ (send (const {cbase nil?} :Array) {:new :[]} ...)
154
+ #type_constructor?(:Array)
155
+ }
156
+ PATTERN
157
+
158
+ # @!method hash_constructor?(node)
159
+ def_node_matcher :hash_constructor?, <<~PATTERN
160
+ {
161
+ (block (send (const {cbase nil?} :Hash) :new) ...)
162
+ (send (const {cbase nil?} :Hash) {:new :[]} ...)
163
+ (send {nil? (const {cbase nil?} :Kernel)} :Hash ...)
164
+ }
165
+ PATTERN
166
+
167
+ # @!method set_constructor?(node)
168
+ def_node_matcher :set_constructor?, <<~PATTERN
169
+ {
170
+ (send (const {cbase nil?} :Set) {:new :[]} ...)
171
+ }
172
+ PATTERN
173
+
174
+ # @!method exception_false_keyword_argument?(node)
175
+ def_node_matcher :exception_false_keyword_argument?, <<~PATTERN
176
+ (hash (pair (sym :exception) false))
177
+ PATTERN
178
+
179
+ # rubocop:disable Metrics/AbcSize
180
+ def on_send(node)
181
+ return if hash_or_set_with_block?(node)
182
+
183
+ receiver = find_receiver(node)
184
+ return unless literal_receiver?(node, receiver) ||
185
+ constructor?(node, receiver) ||
186
+ chained_conversion?(node, receiver) ||
187
+ chained_to_typed_method?(node, receiver)
188
+
189
+ message = format(MSG, method: node.method_name)
190
+
191
+ add_offense(node.loc.selector, message: message) do |corrector|
192
+ corrector.remove(node.loc.dot.join(node.loc.selector))
193
+ end
194
+ end
195
+ # rubocop:enable Metrics/AbcSize
196
+ alias on_csend on_send
197
+
198
+ private
199
+
200
+ def hash_or_set_with_block?(node)
201
+ return false if !node.method?(:to_h) && !node.method?(:to_set)
202
+
203
+ node.parent&.any_block_type? || node.last_argument&.block_pass_type?
204
+ end
205
+
206
+ def find_receiver(node)
207
+ receiver = node.receiver
208
+ return unless receiver
209
+
210
+ while receiver.begin_type?
211
+ break unless receiver.children.one?
212
+
213
+ receiver = receiver.children.first
214
+ end
215
+
216
+ receiver
217
+ end
218
+
219
+ def literal_receiver?(node, receiver)
220
+ return false unless receiver
221
+
222
+ receiver.type?(*LITERAL_NODE_TYPES[node.method_name])
223
+ end
224
+
225
+ def constructor?(node, receiver)
226
+ matcher = CONSTRUCTOR_MAPPING[node.method_name]
227
+ return false unless matcher
228
+
229
+ public_send(matcher, receiver) && !constructor_suppresses_exceptions?(receiver)
230
+ end
231
+
232
+ def constructor_suppresses_exceptions?(receiver)
233
+ # If the constructor suppresses exceptions with `exception: false`, it is possible
234
+ # it could return `nil`, and therefore a chained conversion is not redundant.
235
+ receiver.arguments.any? { |arg| exception_false_keyword_argument?(arg) }
236
+ end
237
+
238
+ def chained_conversion?(node, receiver)
239
+ return false unless receiver&.call_type?
240
+
241
+ receiver.method?(node.method_name)
242
+ end
243
+
244
+ def chained_to_typed_method?(node, receiver)
245
+ return false unless receiver&.call_type?
246
+
247
+ TYPED_METHODS.fetch(node.method_name, []).include?(receiver.method_name)
248
+ end
249
+ end
250
+ end
251
+ end
252
+ end
@@ -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