rbhint 0.85.1.rc2 → 0.87.1.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (148) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +9 -1
  3. data/bin/rubocop-profile +16 -0
  4. data/config/default.yml +118 -10
  5. data/lib/rbhint/version.rb +1 -1
  6. data/lib/rubocop.rb +15 -1
  7. data/lib/rubocop/cli.rb +0 -2
  8. data/lib/rubocop/cli/command/auto_genenerate_config.rb +40 -5
  9. data/lib/rubocop/cli/command/init_dotfile.rb +1 -1
  10. data/lib/rubocop/cli/command/show_cops.rb +1 -1
  11. data/lib/rubocop/config_loader.rb +24 -66
  12. data/lib/rubocop/config_obsoletion.rb +0 -1
  13. data/lib/rubocop/cop/autocorrect_logic.rb +14 -24
  14. data/lib/rubocop/cop/badge.rb +1 -1
  15. data/lib/rubocop/cop/base.rb +407 -0
  16. data/lib/rubocop/cop/bundler/insecure_protocol_source.rb +10 -20
  17. data/lib/rubocop/cop/commissioner.rb +48 -50
  18. data/lib/rubocop/cop/cop.rb +85 -236
  19. data/lib/rubocop/cop/corrector.rb +38 -115
  20. data/lib/rubocop/cop/correctors/multiline_literal_brace_corrector.rb +26 -0
  21. data/lib/rubocop/cop/gemspec/required_ruby_version.rb +6 -1
  22. data/lib/rubocop/cop/gemspec/ruby_version_globals_usage.rb +1 -1
  23. data/lib/rubocop/cop/generator.rb +1 -1
  24. data/lib/rubocop/cop/internal_affairs/node_type_predicate.rb +11 -14
  25. data/lib/rubocop/cop/layout/case_indentation.rb +18 -19
  26. data/lib/rubocop/cop/layout/class_structure.rb +2 -37
  27. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +1 -0
  28. data/lib/rubocop/cop/layout/empty_lines_around_attribute_accessor.rb +1 -8
  29. data/lib/rubocop/cop/layout/end_alignment.rb +3 -2
  30. data/lib/rubocop/cop/layout/first_argument_indentation.rb +4 -0
  31. data/lib/rubocop/cop/layout/hash_alignment.rb +1 -2
  32. data/lib/rubocop/cop/layout/multiline_block_layout.rb +17 -7
  33. data/lib/rubocop/cop/layout/space_around_block_parameters.rb +22 -27
  34. data/lib/rubocop/cop/layout/space_around_method_call_operator.rb +27 -68
  35. data/lib/rubocop/cop/layout/space_inside_array_literal_brackets.rb +3 -2
  36. data/lib/rubocop/cop/legacy/corrections_proxy.rb +49 -0
  37. data/lib/rubocop/cop/legacy/corrector.rb +29 -0
  38. data/lib/rubocop/cop/lint/deprecated_open_ssl_constant.rb +4 -4
  39. data/lib/rubocop/cop/lint/duplicate_elsif_condition.rb +39 -0
  40. data/lib/rubocop/cop/lint/duplicate_methods.rb +2 -2
  41. data/lib/rubocop/cop/lint/implicit_string_concatenation.rb +3 -2
  42. data/lib/rubocop/cop/lint/interpolation_check.rb +13 -0
  43. data/lib/rubocop/cop/lint/literal_as_condition.rb +11 -1
  44. data/lib/rubocop/cop/lint/nested_method_definition.rb +14 -20
  45. data/lib/rubocop/cop/lint/non_deterministic_require_order.rb +2 -2
  46. data/lib/rubocop/cop/lint/parentheses_as_grouped_expression.rb +8 -3
  47. data/lib/rubocop/cop/lint/raise_exception.rb +8 -0
  48. data/lib/rubocop/cop/lint/rand_one.rb +1 -1
  49. data/lib/rubocop/cop/lint/redundant_cop_disable_directive.rb +27 -23
  50. data/lib/rubocop/cop/lint/redundant_splat_expansion.rb +2 -2
  51. data/lib/rubocop/cop/lint/regexp_as_condition.rb +6 -0
  52. data/lib/rubocop/cop/lint/safe_navigation_with_empty.rb +8 -0
  53. data/lib/rubocop/cop/lint/syntax.rb +11 -26
  54. data/lib/rubocop/cop/lint/unused_method_argument.rb +1 -1
  55. data/lib/rubocop/cop/lint/useless_access_modifier.rb +1 -1
  56. data/lib/rubocop/cop/metrics/block_length.rb +24 -2
  57. data/lib/rubocop/cop/metrics/block_nesting.rb +1 -1
  58. data/lib/rubocop/cop/metrics/class_length.rb +26 -3
  59. data/lib/rubocop/cop/metrics/cyclomatic_complexity.rb +1 -1
  60. data/lib/rubocop/cop/metrics/method_length.rb +24 -1
  61. data/lib/rubocop/cop/metrics/module_length.rb +26 -3
  62. data/lib/rubocop/cop/metrics/parameter_lists.rb +1 -1
  63. data/lib/rubocop/cop/metrics/perceived_complexity.rb +3 -3
  64. data/lib/rubocop/cop/metrics/utils/code_length_calculator.rb +129 -0
  65. data/lib/rubocop/cop/mixin/allowed_methods.rb +19 -0
  66. data/lib/rubocop/cop/mixin/auto_corrector.rb +12 -0
  67. data/lib/rubocop/cop/mixin/code_length.rb +4 -0
  68. data/lib/rubocop/cop/mixin/configurable_formatting.rb +1 -1
  69. data/lib/rubocop/cop/mixin/enforce_superclass.rb +3 -1
  70. data/lib/rubocop/cop/mixin/nil_methods.rb +3 -5
  71. data/lib/rubocop/cop/mixin/ordered_gem_node.rb +6 -1
  72. data/lib/rubocop/cop/mixin/statement_modifier.rb +3 -3
  73. data/lib/rubocop/cop/mixin/surrounding_space.rb +7 -2
  74. data/lib/rubocop/cop/mixin/too_many_lines.rb +3 -13
  75. data/lib/rubocop/cop/mixin/uncommunicative_name.rb +4 -2
  76. data/lib/rubocop/cop/mixin/visibility_help.rb +50 -0
  77. data/lib/rubocop/cop/naming/ascii_identifiers.rb +27 -4
  78. data/lib/rubocop/cop/naming/binary_operator_parameter_name.rb +2 -2
  79. data/lib/rubocop/cop/naming/method_name.rb +1 -1
  80. data/lib/rubocop/cop/naming/method_parameter_name.rb +1 -1
  81. data/lib/rubocop/cop/naming/predicate_name.rb +3 -5
  82. data/lib/rubocop/cop/naming/variable_name.rb +1 -1
  83. data/lib/rubocop/cop/naming/variable_number.rb +1 -1
  84. data/lib/rubocop/cop/offense.rb +16 -2
  85. data/lib/rubocop/cop/style/accessor_grouping.rb +147 -0
  86. data/lib/rubocop/cop/style/array_coercion.rb +63 -0
  87. data/lib/rubocop/cop/style/auto_resource_cleanup.rb +3 -2
  88. data/lib/rubocop/cop/style/bisected_attr_accessor.rb +146 -0
  89. data/lib/rubocop/cop/style/case_like_if.rb +217 -0
  90. data/lib/rubocop/cop/style/class_vars.rb +21 -0
  91. data/lib/rubocop/cop/style/commented_keyword.rb +5 -2
  92. data/lib/rubocop/cop/style/conditional_assignment.rb +1 -1
  93. data/lib/rubocop/cop/style/date_time.rb +1 -1
  94. data/lib/rubocop/cop/style/dir.rb +2 -2
  95. data/lib/rubocop/cop/style/empty_literal.rb +5 -5
  96. data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -2
  97. data/lib/rubocop/cop/style/exponential_notation.rb +6 -8
  98. data/lib/rubocop/cop/style/float_division.rb +7 -10
  99. data/lib/rubocop/cop/style/format_string_token.rb +5 -5
  100. data/lib/rubocop/cop/style/hash_as_last_array_item.rb +62 -0
  101. data/lib/rubocop/cop/style/hash_like_case.rb +76 -0
  102. data/lib/rubocop/cop/style/if_unless_modifier.rb +11 -11
  103. data/lib/rubocop/cop/style/if_unless_modifier_of_if_unless.rb +12 -0
  104. data/lib/rubocop/cop/style/missing_else.rb +1 -11
  105. data/lib/rubocop/cop/style/multiline_block_chain.rb +10 -1
  106. data/lib/rubocop/cop/style/mutable_constant.rb +4 -4
  107. data/lib/rubocop/cop/style/nested_parenthesized_calls.rb +2 -5
  108. data/lib/rubocop/cop/style/numeric_predicate.rb +3 -4
  109. data/lib/rubocop/cop/style/parallel_assignment.rb +3 -3
  110. data/lib/rubocop/cop/style/proc.rb +1 -1
  111. data/lib/rubocop/cop/style/random_with_offset.rb +4 -10
  112. data/lib/rubocop/cop/style/redundant_assignment.rb +117 -0
  113. data/lib/rubocop/cop/style/redundant_exception.rb +14 -10
  114. data/lib/rubocop/cop/style/redundant_fetch_block.rb +122 -0
  115. data/lib/rubocop/cop/style/redundant_file_extension_in_require.rb +50 -0
  116. data/lib/rubocop/cop/style/redundant_freeze.rb +1 -1
  117. data/lib/rubocop/cop/style/redundant_parentheses.rb +7 -1
  118. data/lib/rubocop/cop/style/redundant_regexp_character_class.rb +2 -1
  119. data/lib/rubocop/cop/style/redundant_regexp_escape.rb +2 -2
  120. data/lib/rubocop/cop/style/redundant_sort.rb +3 -2
  121. data/lib/rubocop/cop/style/rescue_standard_error.rb +1 -1
  122. data/lib/rubocop/cop/style/signal_exception.rb +1 -1
  123. data/lib/rubocop/cop/style/stabby_lambda_parentheses.rb +3 -2
  124. data/lib/rubocop/cop/style/stderr_puts.rb +1 -1
  125. data/lib/rubocop/cop/style/struct_inheritance.rb +2 -2
  126. data/lib/rubocop/cop/style/symbol_proc.rb +1 -1
  127. data/lib/rubocop/cop/style/trailing_method_end_statement.rb +9 -32
  128. data/lib/rubocop/cop/style/trivial_accessors.rb +8 -7
  129. data/lib/rubocop/cop/style/yoda_condition.rb +18 -1
  130. data/lib/rubocop/cop/style/zero_length_predicate.rb +2 -2
  131. data/lib/rubocop/cop/team.rb +97 -81
  132. data/lib/rubocop/cop/utils/format_string.rb +1 -2
  133. data/lib/rubocop/cop/variable_force/variable.rb +5 -3
  134. data/lib/rubocop/file_finder.rb +4 -4
  135. data/lib/rubocop/formatter/disabled_config_formatter.rb +1 -1
  136. data/lib/rubocop/name_similarity.rb +1 -3
  137. data/lib/rubocop/options.rb +15 -8
  138. data/lib/rubocop/path_util.rb +2 -17
  139. data/lib/rubocop/rake_task.rb +6 -9
  140. data/lib/rubocop/result_cache.rb +9 -5
  141. data/lib/rubocop/rspec/cop_helper.rb +4 -4
  142. data/lib/rubocop/rspec/expect_offense.rb +52 -22
  143. data/lib/rubocop/rspec/shared_contexts.rb +8 -8
  144. data/lib/rubocop/runner.rb +33 -32
  145. data/lib/rubocop/target_ruby.rb +1 -1
  146. data/lib/rubocop/version.rb +1 -1
  147. metadata +25 -8
  148. data/lib/rubocop/cop/mixin/classish_length.rb +0 -37
@@ -30,7 +30,7 @@ module RuboCop
30
30
  def on_masgn(node)
31
31
  lhs, rhs = *node
32
32
  lhs_elements = *lhs
33
- rhs_elements = [*rhs].compact # edge case for one constant
33
+ rhs_elements = Array(rhs).compact # edge case for one constant
34
34
 
35
35
  return if allowed_lhs?(lhs) || allowed_rhs?(rhs) ||
36
36
  allowed_masign?(lhs_elements, rhs_elements)
@@ -42,7 +42,7 @@ module RuboCop
42
42
  lambda do |corrector|
43
43
  left, right = *node
44
44
  left_elements = *left
45
- right_elements = [*right].compact
45
+ right_elements = Array(right).compact
46
46
  order = find_valid_order(left_elements, right_elements)
47
47
  correction = assignment_corrector(node, order)
48
48
 
@@ -69,7 +69,7 @@ module RuboCop
69
69
 
70
70
  def allowed_rhs?(node)
71
71
  # Edge case for one constant
72
- elements = [*node].compact
72
+ elements = Array(node).compact
73
73
 
74
74
  # Account for edge case of `Constant::CONSTANT`
75
75
  !node.array_type? ||
@@ -17,7 +17,7 @@ module RuboCop
17
17
  MSG = 'Use `proc` instead of `Proc.new`.'
18
18
 
19
19
  def_node_matcher :proc_new?,
20
- '(block $(send (const nil? :Proc) :new) ...)'
20
+ '(block $(send (const {nil? cbase} :Proc) :new) ...)'
21
21
 
22
22
  def on_block(node)
23
23
  proc_new?(node) do |block_method|
@@ -31,7 +31,7 @@ module RuboCop
31
31
  (send
32
32
  int {:+ :-}
33
33
  (send
34
- {nil? (const nil? :Random) (const nil? :Kernel)}
34
+ {nil? (const {nil? cbase} :Random) (const {nil? cbase} :Kernel)}
35
35
  :rand
36
36
  {int irange erange}))
37
37
  PATTERN
@@ -39,7 +39,7 @@ module RuboCop
39
39
  def_node_matcher :rand_op_integer?, <<~PATTERN
40
40
  (send
41
41
  (send
42
- {nil? (const nil? :Random) (const nil? :Kernel)}
42
+ {nil? (const {nil? cbase} :Random) (const {nil? cbase} :Kernel)}
43
43
  :rand
44
44
  {int irange erange})
45
45
  {:+ :-}
@@ -49,7 +49,7 @@ module RuboCop
49
49
  def_node_matcher :rand_modified?, <<~PATTERN
50
50
  (send
51
51
  (send
52
- {nil? (const nil? :Random) (const nil? :Kernel)}
52
+ {nil? (const {nil? cbase} :Random) (const {nil? cbase} :Kernel)}
53
53
  :rand
54
54
  {int irange erange})
55
55
  {:succ :pred :next})
@@ -128,14 +128,8 @@ module RuboCop
128
128
  end
129
129
  end
130
130
 
131
- def_node_matcher :namespace, <<~PATTERN
132
- {$nil? (const nil? $_)}
133
- PATTERN
134
-
135
131
  def prefix_from_prefix_node(node)
136
- namespace(node) do |namespace|
137
- [namespace, 'rand'].compact.join('.')
138
- end
132
+ [node&.source, 'rand'].compact.join('.')
139
133
  end
140
134
 
141
135
  def boundaries_from_random_node(random_node)
@@ -0,0 +1,117 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Style
6
+ # This cop checks for redundant assignment before returning.
7
+ #
8
+ # @example
9
+ # # bad
10
+ # def test
11
+ # x = foo
12
+ # x
13
+ # end
14
+ #
15
+ # # bad
16
+ # def test
17
+ # if x
18
+ # z = foo
19
+ # z
20
+ # elsif y
21
+ # z = bar
22
+ # z
23
+ # end
24
+ # end
25
+ #
26
+ # # good
27
+ # def test
28
+ # foo
29
+ # end
30
+ #
31
+ # # good
32
+ # def test
33
+ # if x
34
+ # foo
35
+ # elsif y
36
+ # bar
37
+ # end
38
+ # end
39
+ #
40
+ class RedundantAssignment < Cop
41
+ MSG = 'Redundant assignment before returning detected.'
42
+
43
+ def_node_matcher :redundant_assignment?, <<~PATTERN
44
+ (... $(lvasgn _name _expression) (lvar _name))
45
+ PATTERN
46
+
47
+ def on_def(node)
48
+ check_branch(node.body)
49
+ end
50
+ alias on_defs on_def
51
+
52
+ def autocorrect(node)
53
+ lambda do |corrector|
54
+ expression = node.children[1]
55
+ corrector.replace(node, expression.source)
56
+ corrector.remove(right_sibling_of(node))
57
+ end
58
+ end
59
+
60
+ private
61
+
62
+ def check_branch(node)
63
+ return unless node
64
+
65
+ case node.type
66
+ when :case then check_case_node(node)
67
+ when :if then check_if_node(node)
68
+ when :rescue, :resbody
69
+ check_rescue_node(node)
70
+ when :ensure then check_ensure_node(node)
71
+ when :begin, :kwbegin
72
+ check_begin_node(node)
73
+ end
74
+ end
75
+
76
+ def check_case_node(node)
77
+ node.when_branches.each { |when_node| check_branch(when_node.body) }
78
+ check_branch(node.else_branch)
79
+ end
80
+
81
+ def check_if_node(node)
82
+ return if node.modifier_form? || node.ternary?
83
+
84
+ check_branch(node.if_branch)
85
+ check_branch(node.else_branch)
86
+ end
87
+
88
+ def check_rescue_node(node)
89
+ node.child_nodes.each do |child_node|
90
+ check_branch(child_node)
91
+ end
92
+ end
93
+
94
+ def check_ensure_node(node)
95
+ check_branch(node.body)
96
+ end
97
+
98
+ def check_begin_node(node)
99
+ if (assignment = redundant_assignment?(node))
100
+ add_offense(assignment)
101
+ else
102
+ last_expr = node.children.last
103
+ check_branch(last_expr)
104
+ end
105
+ end
106
+
107
+ def right_sibling_of(node)
108
+ siblings_of(node)[node.sibling_index + 1]
109
+ end
110
+
111
+ def siblings_of(node)
112
+ node.parent.children
113
+ end
114
+ end
115
+ end
116
+ end
117
+ end
@@ -16,21 +16,22 @@ module RuboCop
16
16
  #
17
17
  # # Good
18
18
  # raise 'message'
19
- class RedundantException < Cop
19
+ class RedundantException < Base
20
+ extend AutoCorrector
21
+
20
22
  MSG_1 = 'Redundant `RuntimeError` argument can be removed.'
21
23
  MSG_2 = 'Redundant `RuntimeError.new` call can be replaced with ' \
22
24
  'just the message.'
23
25
 
26
+ # Switch `raise RuntimeError, 'message'` to `raise 'message'`, and
27
+ # `raise RuntimeError.new('message')` to `raise 'message'`.
24
28
  def on_send(node)
25
- exploded?(node) { return add_offense(node, message: MSG_1) }
26
- compact?(node) { add_offense(node, message: MSG_2) }
29
+ fix_exploded(node) || fix_compact(node)
27
30
  end
28
31
 
29
- # Switch `raise RuntimeError, 'message'` to `raise 'message'`, and
30
- # `raise RuntimeError.new('message')` to `raise 'message'`.
31
- def autocorrect(node) # rubocop:disable Metrics/MethodLength
32
+ def fix_exploded(node)
32
33
  exploded?(node) do |command, message|
33
- return lambda do |corrector|
34
+ add_offense(node, message: MSG_1) do |corrector|
34
35
  if node.parenthesized?
35
36
  corrector.replace(node,
36
37
  "#{command}(#{message.source})")
@@ -40,19 +41,22 @@ module RuboCop
40
41
  end
41
42
  end
42
43
  end
44
+ end
45
+
46
+ def fix_compact(node)
43
47
  compact?(node) do |new_call, message|
44
- lambda do |corrector|
48
+ add_offense(node, message: MSG_2) do |corrector|
45
49
  corrector.replace(new_call, message.source)
46
50
  end
47
51
  end
48
52
  end
49
53
 
50
54
  def_node_matcher :exploded?, <<~PATTERN
51
- (send nil? ${:raise :fail} (const nil? :RuntimeError) $_)
55
+ (send nil? ${:raise :fail} (const {nil? cbase} :RuntimeError) $_)
52
56
  PATTERN
53
57
 
54
58
  def_node_matcher :compact?, <<~PATTERN
55
- (send nil? {:raise :fail} $(send (const nil? :RuntimeError) :new $_))
59
+ (send nil? {:raise :fail} $(send (const {nil? cbase} :RuntimeError) :new $_))
56
60
  PATTERN
57
61
  end
58
62
  end
@@ -0,0 +1,122 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Style
6
+ # This cop identifies places where `fetch(key) { value }`
7
+ # can be replaced by `fetch(key, value)`.
8
+ #
9
+ # In such cases `fetch(key, value)` method is faster
10
+ # than `fetch(key) { value }`.
11
+ #
12
+ # @example SafeForConstants: false (default)
13
+ # # bad
14
+ # hash.fetch(:key) { 5 }
15
+ # hash.fetch(:key) { true }
16
+ # hash.fetch(:key) { nil }
17
+ # array.fetch(5) { :value }
18
+ # ENV.fetch(:key) { 'value' }
19
+ #
20
+ # # good
21
+ # hash.fetch(:key, 5)
22
+ # hash.fetch(:key, true)
23
+ # hash.fetch(:key, nil)
24
+ # array.fetch(5, :value)
25
+ # ENV.fetch(:key, 'value')
26
+ #
27
+ # @example SafeForConstants: true
28
+ # # bad
29
+ # ENV.fetch(:key) { VALUE }
30
+ #
31
+ # # good
32
+ # ENV.fetch(:key, VALUE)
33
+ #
34
+ class RedundantFetchBlock < Cop
35
+ include FrozenStringLiteral
36
+ include RangeHelp
37
+
38
+ MSG = 'Use `%<good>s` instead of `%<bad>s`.'
39
+
40
+ def_node_matcher :redundant_fetch_block_candidate?, <<~PATTERN
41
+ (block
42
+ $(send _ :fetch _)
43
+ (args)
44
+ ${nil? #basic_literal? #const_type?})
45
+ PATTERN
46
+
47
+ def on_block(node)
48
+ redundant_fetch_block_candidate?(node) do |send, body|
49
+ return if should_not_check?(send, body)
50
+
51
+ range = fetch_range(send, node)
52
+ good = build_good_method(send, body)
53
+ bad = build_bad_method(send, body)
54
+
55
+ add_offense(
56
+ node,
57
+ location: range,
58
+ message: format(MSG, good: good, bad: bad)
59
+ )
60
+ end
61
+ end
62
+
63
+ def autocorrect(node)
64
+ redundant_fetch_block_candidate?(node) do |send, body|
65
+ lambda do |corrector|
66
+ receiver, _, key = send.children
67
+ default_value = body ? body.source : 'nil'
68
+
69
+ corrector.replace(node, "#{receiver.source}.fetch(#{key.source}, #{default_value})")
70
+ end
71
+ end
72
+ end
73
+
74
+ private
75
+
76
+ def basic_literal?(node)
77
+ node&.basic_literal?
78
+ end
79
+
80
+ def const_type?(node)
81
+ node&.const_type?
82
+ end
83
+
84
+ def should_not_check?(send, body)
85
+ (body&.const_type? && !check_for_constant?) ||
86
+ (body&.str_type? && !check_for_string?) ||
87
+ rails_cache?(send.receiver)
88
+ end
89
+
90
+ def_node_matcher :rails_cache?, <<~PATTERN
91
+ (send (const _ :Rails) :cache)
92
+ PATTERN
93
+
94
+ def fetch_range(send, node)
95
+ range_between(send.loc.selector.begin_pos, node.loc.end.end_pos)
96
+ end
97
+
98
+ def build_good_method(send, body)
99
+ key = send.children[2].source
100
+ default_value = body ? body.source : 'nil'
101
+
102
+ "fetch(#{key}, #{default_value})"
103
+ end
104
+
105
+ def build_bad_method(send, body)
106
+ key = send.children[2].source
107
+ block = body ? "{ #{body.source} }" : '{}'
108
+
109
+ "fetch(#{key}) #{block}"
110
+ end
111
+
112
+ def check_for_constant?
113
+ cop_config['SafeForConstants']
114
+ end
115
+
116
+ def check_for_string?
117
+ frozen_string_literals_enabled?
118
+ end
119
+ end
120
+ end
121
+ end
122
+ end
@@ -0,0 +1,50 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Style
6
+ # This cop checks for the presence of superfluous `.rb` extension in
7
+ # the filename provided to `require` and `require_relative`.
8
+ #
9
+ # Note: If the extension is omitted, Ruby tries adding '.rb', '.so',
10
+ # and so on to the name until found. If the file named cannot be found,
11
+ # a `LoadError` will be raised.
12
+ # There is an edge case where `foo.so` file is loaded instead of a `LoadError`
13
+ # if `foo.so` file exists when `require 'foo.rb'` will be changed to `require 'foo'`,
14
+ # but that seems harmless.
15
+ #
16
+ # @example
17
+ # # bad
18
+ # require 'foo.rb'
19
+ # require_relative '../foo.rb'
20
+ #
21
+ # # good
22
+ # require 'foo'
23
+ # require 'foo.so'
24
+ # require_relative '../foo'
25
+ # require_relative '../foo.so'
26
+ #
27
+ class RedundantFileExtensionInRequire < Cop
28
+ MSG = 'Redundant `.rb` file extension detected.'
29
+
30
+ def_node_matcher :require_call?, <<~PATTERN
31
+ (send nil? {:require :require_relative} $str_type?)
32
+ PATTERN
33
+
34
+ def on_send(node)
35
+ require_call?(node) do |name_node|
36
+ add_offense(name_node) if name_node.value.end_with?('.rb')
37
+ end
38
+ end
39
+
40
+ def autocorrect(node)
41
+ correction = node.value.sub(/\.rb\z/, '')
42
+
43
+ lambda do |corrector|
44
+ corrector.replace(node, "'#{correction}'")
45
+ end
46
+ end
47
+ end
48
+ end
49
+ end
50
+ end
@@ -56,7 +56,7 @@ module RuboCop
56
56
  (begin (send {float int} {:+ :- :* :** :/ :% :<<} _))
57
57
  (begin (send !(str _) {:+ :- :* :** :/ :%} {float int}))
58
58
  (begin (send _ {:== :=== :!= :<= :>= :< :>} _))
59
- (send (const nil? :ENV) :[] _)
59
+ (send (const {nil? cbase} :ENV) :[] _)
60
60
  (send _ {:count :length :size} ...)
61
61
  (block (send _ {:count :length :size} ...) ...)
62
62
  }
@@ -205,7 +205,9 @@ module RuboCop
205
205
  end
206
206
 
207
207
  def first_argument?(node)
208
- first_send_argument?(node) || first_super_argument?(node)
208
+ first_send_argument?(node) ||
209
+ first_super_argument?(node) ||
210
+ first_yield_argument?(node)
209
211
  end
210
212
 
211
213
  def_node_matcher :first_send_argument?, <<~PATTERN
@@ -216,6 +218,10 @@ module RuboCop
216
218
  ^(super equal?(%0) ...)
217
219
  PATTERN
218
220
 
221
+ def_node_matcher :first_yield_argument?, <<~PATTERN
222
+ ^(yield equal?(%0) ...)
223
+ PATTERN
224
+
219
225
  def call_chain_starts_with_int?(begin_node, send_node)
220
226
  recv = first_part_of_call_chain(send_node)
221
227
  recv&.int_type? && (parent = begin_node.parent) &&