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.
- checksums.yaml +4 -4
- data/README.md +3 -3
- data/config/default.yml +55 -13
- data/config/internal_affairs.yml +20 -0
- data/lib/rubocop/cli/command/suggest_extensions.rb +7 -1
- data/lib/rubocop/comment_config.rb +1 -1
- data/lib/rubocop/config.rb +4 -0
- data/lib/rubocop/config_loader.rb +44 -9
- data/lib/rubocop/config_loader_resolver.rb +23 -9
- data/lib/rubocop/config_validator.rb +1 -1
- data/lib/rubocop/cop/internal_affairs/example_description.rb +7 -3
- data/lib/rubocop/cop/internal_affairs/location_exists.rb +116 -0
- data/lib/rubocop/cop/internal_affairs/node_pattern_groups/ast_walker.rb +1 -1
- data/lib/rubocop/cop/internal_affairs/node_type_group.rb +91 -0
- data/lib/rubocop/cop/internal_affairs/plugin.rb +33 -0
- data/lib/rubocop/cop/internal_affairs/undefined_config.rb +7 -1
- data/lib/rubocop/cop/internal_affairs.rb +2 -16
- data/lib/rubocop/cop/layout/block_alignment.rb +2 -0
- data/lib/rubocop/cop/layout/closing_parenthesis_indentation.rb +4 -4
- data/lib/rubocop/cop/layout/else_alignment.rb +1 -1
- data/lib/rubocop/cop/layout/empty_line_after_guard_clause.rb +1 -1
- data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +26 -1
- data/lib/rubocop/cop/layout/empty_lines_around_method_body.rb +22 -2
- data/lib/rubocop/cop/layout/heredoc_argument_closing_parenthesis.rb +1 -1
- data/lib/rubocop/cop/layout/line_length.rb +3 -3
- data/lib/rubocop/cop/layout/multiline_method_call_indentation.rb +2 -2
- data/lib/rubocop/cop/layout/space_around_method_call_operator.rb +1 -1
- data/lib/rubocop/cop/lint/cop_directive_syntax.rb +84 -0
- data/lib/rubocop/cop/lint/duplicate_methods.rb +0 -14
- data/lib/rubocop/cop/lint/empty_conditional_body.rb +14 -64
- data/lib/rubocop/cop/lint/erb_new_arguments.rb +0 -6
- data/lib/rubocop/cop/lint/float_comparison.rb +1 -6
- data/lib/rubocop/cop/lint/format_parameter_mismatch.rb +1 -1
- data/lib/rubocop/cop/lint/literal_as_condition.rb +99 -9
- data/lib/rubocop/cop/lint/mixed_case_range.rb +2 -2
- data/lib/rubocop/cop/lint/redundant_require_statement.rb +0 -21
- data/lib/rubocop/cop/lint/redundant_type_conversion.rb +252 -0
- data/lib/rubocop/cop/lint/suppressed_exception_in_number_conversion.rb +111 -0
- data/lib/rubocop/cop/lint/useless_constant_scoping.rb +80 -0
- data/lib/rubocop/cop/lint/void.rb +6 -0
- data/lib/rubocop/cop/metrics/utils/repeated_attribute_discount.rb +7 -7
- data/lib/rubocop/cop/mixin/alignment.rb +2 -2
- data/lib/rubocop/cop/mixin/allowed_pattern.rb +4 -4
- data/lib/rubocop/cop/mixin/comments_help.rb +1 -1
- data/lib/rubocop/cop/mixin/hash_shorthand_syntax.rb +18 -18
- data/lib/rubocop/cop/mixin/hash_subset.rb +19 -4
- data/lib/rubocop/cop/mixin/hash_transform_method.rb +74 -74
- data/lib/rubocop/cop/mixin/percent_literal.rb +1 -1
- data/lib/rubocop/cop/mixin/range_help.rb +15 -3
- data/lib/rubocop/cop/mixin/string_help.rb +1 -1
- data/lib/rubocop/cop/mixin/trailing_comma.rb +12 -0
- data/lib/rubocop/cop/naming/block_forwarding.rb +3 -3
- data/lib/rubocop/cop/naming/predicate_name.rb +44 -0
- data/lib/rubocop/cop/naming/variable_name.rb +64 -6
- data/lib/rubocop/cop/style/accessor_grouping.rb +19 -5
- data/lib/rubocop/cop/style/arguments_forwarding.rb +3 -3
- data/lib/rubocop/cop/style/commented_keyword.rb +1 -1
- data/lib/rubocop/cop/style/endless_method.rb +163 -18
- data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -7
- data/lib/rubocop/cop/style/inverse_methods.rb +8 -5
- data/lib/rubocop/cop/style/keyword_parameters_order.rb +13 -7
- data/lib/rubocop/cop/style/line_end_concatenation.rb +10 -4
- data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +1 -1
- data/lib/rubocop/cop/style/multiline_block_chain.rb +1 -1
- data/lib/rubocop/cop/style/multiline_method_signature.rb +1 -9
- data/lib/rubocop/cop/style/redundant_condition.rb +45 -0
- data/lib/rubocop/cop/style/redundant_format.rb +250 -0
- data/lib/rubocop/cop/style/redundant_freeze.rb +1 -1
- data/lib/rubocop/cop/style/redundant_parentheses.rb +18 -4
- data/lib/rubocop/cop/style/redundant_self_assignment.rb +1 -1
- data/lib/rubocop/cop/style/single_line_methods.rb +3 -3
- data/lib/rubocop/cop/style/sole_nested_conditional.rb +0 -6
- data/lib/rubocop/cop/style/string_concatenation.rb +1 -1
- data/lib/rubocop/cop/style/trailing_comma_in_array_literal.rb +47 -6
- data/lib/rubocop/cop/style/trailing_comma_in_hash_literal.rb +48 -6
- data/lib/rubocop/cop/style/trivial_accessors.rb +1 -1
- data/lib/rubocop/cop/util.rb +1 -1
- data/lib/rubocop/cop/utils/format_string.rb +10 -5
- data/lib/rubocop/cops_documentation_generator.rb +12 -1
- data/lib/rubocop/directive_comment.rb +35 -2
- data/lib/rubocop/lsp/runtime.rb +2 -0
- data/lib/rubocop/lsp/server.rb +0 -2
- data/lib/rubocop/options.rb +26 -11
- data/lib/rubocop/path_util.rb +4 -0
- data/lib/rubocop/plugin/configuration_integrator.rb +143 -0
- data/lib/rubocop/plugin/load_error.rb +26 -0
- data/lib/rubocop/plugin/loader.rb +100 -0
- data/lib/rubocop/plugin/not_supported_error.rb +29 -0
- data/lib/rubocop/plugin.rb +46 -0
- data/lib/rubocop/rake_task.rb +4 -1
- data/lib/rubocop/rspec/cop_helper.rb +9 -0
- data/lib/rubocop/rspec/shared_contexts.rb +15 -0
- data/lib/rubocop/rspec/support.rb +1 -0
- data/lib/rubocop/server/cache.rb +35 -2
- data/lib/rubocop/server/cli.rb +2 -2
- data/lib/rubocop/version.rb +17 -2
- data/lib/rubocop.rb +5 -1
- data/lib/ruby_lsp/rubocop/addon.rb +7 -10
- data/lib/ruby_lsp/rubocop/{wraps_built_in_lsp_runtime.rb → runtime_adapter.rb} +5 -8
- metadata +36 -10
- 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
|
-
|
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
|
-
#
|
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
|
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
|
-
|
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
|
66
|
+
return if node.condition.source == 'true'
|
47
67
|
|
48
|
-
|
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
|
-
|
93
|
+
# rubocop:enable Metrics/AbcSize
|
51
94
|
|
52
95
|
def on_until(node)
|
53
|
-
return if
|
96
|
+
return if node.condition.source == 'false'
|
54
97
|
|
55
|
-
|
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
|
-
|
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
|
-
|
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
|
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
|