rubocop 1.50.2 → 1.54.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (151) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +3 -3
  3. data/config/default.yml +80 -6
  4. data/lib/rubocop/cli/command/lsp.rb +19 -0
  5. data/lib/rubocop/cli.rb +3 -0
  6. data/lib/rubocop/config.rb +4 -0
  7. data/lib/rubocop/config_loader_resolver.rb +4 -3
  8. data/lib/rubocop/config_obsoletion.rb +2 -2
  9. data/lib/rubocop/cop/base.rb +5 -1
  10. data/lib/rubocop/cop/bundler/gem_comment.rb +1 -1
  11. data/lib/rubocop/cop/bundler/gem_version.rb +2 -2
  12. data/lib/rubocop/cop/correctors/alignment_corrector.rb +1 -1
  13. data/lib/rubocop/cop/gemspec/dependency_version.rb +2 -2
  14. data/lib/rubocop/cop/gemspec/development_dependencies.rb +1 -1
  15. data/lib/rubocop/cop/internal_affairs/cop_description.rb +32 -8
  16. data/lib/rubocop/cop/internal_affairs/location_line_equality_comparison.rb +3 -1
  17. data/lib/rubocop/cop/internal_affairs/node_matcher_directive.rb +5 -5
  18. data/lib/rubocop/cop/layout/class_structure.rb +7 -0
  19. data/lib/rubocop/cop/layout/closing_heredoc_indentation.rb +1 -2
  20. data/lib/rubocop/cop/layout/empty_line_between_defs.rb +27 -4
  21. data/lib/rubocop/cop/layout/empty_lines_around_exception_handling_keywords.rb +2 -0
  22. data/lib/rubocop/cop/layout/heredoc_argument_closing_parenthesis.rb +2 -2
  23. data/lib/rubocop/cop/layout/indentation_style.rb +1 -1
  24. data/lib/rubocop/cop/layout/indentation_width.rb +2 -2
  25. data/lib/rubocop/cop/layout/line_end_string_concatenation_indentation.rb +2 -0
  26. data/lib/rubocop/cop/layout/redundant_line_break.rb +1 -1
  27. data/lib/rubocop/cop/layout/space_after_comma.rb +9 -1
  28. data/lib/rubocop/cop/layout/space_around_operators.rb +3 -1
  29. data/lib/rubocop/cop/layout/space_inside_block_braces.rb +2 -0
  30. data/lib/rubocop/cop/layout/space_inside_range_literal.rb +1 -1
  31. data/lib/rubocop/cop/lint/ambiguous_block_association.rb +13 -1
  32. data/lib/rubocop/cop/lint/debugger.rb +9 -5
  33. data/lib/rubocop/cop/lint/duplicate_hash_key.rb +2 -1
  34. data/lib/rubocop/cop/lint/duplicate_regexp_character_class_element.rb +46 -19
  35. data/lib/rubocop/cop/lint/erb_new_arguments.rb +3 -4
  36. data/lib/rubocop/cop/lint/heredoc_method_call_position.rb +1 -1
  37. data/lib/rubocop/cop/lint/identity_comparison.rb +0 -1
  38. data/lib/rubocop/cop/lint/incompatible_io_select_with_fiber_scheduler.rb +5 -3
  39. data/lib/rubocop/cop/lint/inherit_exception.rb +9 -0
  40. data/lib/rubocop/cop/lint/lambda_without_literal_block.rb +1 -1
  41. data/lib/rubocop/cop/lint/missing_super.rb +34 -5
  42. data/lib/rubocop/cop/lint/mixed_case_range.rb +111 -0
  43. data/lib/rubocop/cop/lint/number_conversion.rb +5 -0
  44. data/lib/rubocop/cop/lint/numbered_parameter_assignment.rb +2 -2
  45. data/lib/rubocop/cop/lint/ordered_magic_comments.rb +0 -1
  46. data/lib/rubocop/cop/lint/out_of_range_regexp_ref.rb +2 -2
  47. data/lib/rubocop/cop/lint/redundant_regexp_quantifiers.rb +130 -0
  48. data/lib/rubocop/cop/lint/redundant_require_statement.rb +8 -3
  49. data/lib/rubocop/cop/lint/redundant_string_coercion.rb +1 -1
  50. data/lib/rubocop/cop/lint/send_with_mixin_argument.rb +1 -2
  51. data/lib/rubocop/cop/lint/shadowed_exception.rb +5 -11
  52. data/lib/rubocop/cop/lint/suppressed_exception.rb +1 -1
  53. data/lib/rubocop/cop/lint/symbol_conversion.rb +1 -1
  54. data/lib/rubocop/cop/lint/top_level_return_with_argument.rb +23 -9
  55. data/lib/rubocop/cop/lint/useless_assignment.rb +59 -1
  56. data/lib/rubocop/cop/lint/void.rb +57 -7
  57. data/lib/rubocop/cop/metrics/utils/abc_size_calculator.rb +1 -2
  58. data/lib/rubocop/cop/metrics/utils/code_length_calculator.rb +30 -2
  59. data/lib/rubocop/cop/migration/department_name.rb +2 -2
  60. data/lib/rubocop/cop/mixin/allowed_receivers.rb +34 -0
  61. data/lib/rubocop/cop/mixin/comments_help.rb +7 -3
  62. data/lib/rubocop/cop/mixin/end_keyword_alignment.rb +1 -1
  63. data/lib/rubocop/cop/mixin/heredoc.rb +6 -2
  64. data/lib/rubocop/cop/mixin/percent_literal.rb +1 -1
  65. data/lib/rubocop/cop/mixin/space_after_punctuation.rb +1 -1
  66. data/lib/rubocop/cop/naming/block_forwarding.rb +1 -1
  67. data/lib/rubocop/cop/naming/constant_name.rb +1 -1
  68. data/lib/rubocop/cop/naming/memoized_instance_variable_name.rb +25 -10
  69. data/lib/rubocop/cop/naming/rescued_exceptions_variable_name.rb +11 -3
  70. data/lib/rubocop/cop/naming/variable_name.rb +6 -1
  71. data/lib/rubocop/cop/style/accessor_grouping.rb +5 -1
  72. data/lib/rubocop/cop/style/attr.rb +11 -1
  73. data/lib/rubocop/cop/style/begin_block.rb +1 -2
  74. data/lib/rubocop/cop/style/block_comments.rb +1 -1
  75. data/lib/rubocop/cop/style/block_delimiters.rb +3 -3
  76. data/lib/rubocop/cop/style/class_and_module_children.rb +1 -1
  77. data/lib/rubocop/cop/style/class_equality_comparison.rb +17 -39
  78. data/lib/rubocop/cop/style/collection_compact.rb +16 -6
  79. data/lib/rubocop/cop/style/colon_method_call.rb +2 -2
  80. data/lib/rubocop/cop/style/combinable_loops.rb +26 -6
  81. data/lib/rubocop/cop/style/conditional_assignment.rb +5 -3
  82. data/lib/rubocop/cop/style/copyright.rb +5 -2
  83. data/lib/rubocop/cop/style/dir.rb +1 -1
  84. data/lib/rubocop/cop/style/dir_empty.rb +8 -14
  85. data/lib/rubocop/cop/style/document_dynamic_eval_definition.rb +1 -1
  86. data/lib/rubocop/cop/style/documentation.rb +1 -1
  87. data/lib/rubocop/cop/style/eval_with_location.rb +5 -5
  88. data/lib/rubocop/cop/style/exact_regexp_match.rb +68 -0
  89. data/lib/rubocop/cop/style/file_read.rb +2 -2
  90. data/lib/rubocop/cop/style/guard_clause.rb +2 -0
  91. data/lib/rubocop/cop/style/hash_each_methods.rb +1 -22
  92. data/lib/rubocop/cop/style/hash_except.rb +19 -8
  93. data/lib/rubocop/cop/style/hash_transform_keys.rb +2 -2
  94. data/lib/rubocop/cop/style/hash_transform_values.rb +2 -2
  95. data/lib/rubocop/cop/style/identical_conditional_branches.rb +6 -2
  96. data/lib/rubocop/cop/style/if_inside_else.rb +6 -0
  97. data/lib/rubocop/cop/style/if_unless_modifier.rb +3 -0
  98. data/lib/rubocop/cop/style/if_with_semicolon.rb +2 -2
  99. data/lib/rubocop/cop/style/invertible_unless_condition.rb +10 -6
  100. data/lib/rubocop/cop/style/lambda.rb +3 -3
  101. data/lib/rubocop/cop/style/method_call_with_args_parentheses/omit_parentheses.rb +3 -4
  102. data/lib/rubocop/cop/style/multiple_comparison.rb +14 -0
  103. data/lib/rubocop/cop/style/numeric_literals.rb +1 -1
  104. data/lib/rubocop/cop/style/percent_literal_delimiters.rb +1 -1
  105. data/lib/rubocop/cop/style/preferred_hash_methods.rb +1 -1
  106. data/lib/rubocop/cop/style/redundant_array_constructor.rb +77 -0
  107. data/lib/rubocop/cop/style/redundant_begin.rb +1 -1
  108. data/lib/rubocop/cop/style/redundant_conditional.rb +1 -1
  109. data/lib/rubocop/cop/style/redundant_current_directory_in_path.rb +38 -0
  110. data/lib/rubocop/cop/style/redundant_filter_chain.rb +101 -0
  111. data/lib/rubocop/cop/style/redundant_line_continuation.rb +7 -3
  112. data/lib/rubocop/cop/style/redundant_parentheses.rb +1 -1
  113. data/lib/rubocop/cop/style/redundant_regexp_argument.rb +100 -0
  114. data/lib/rubocop/cop/style/redundant_regexp_constructor.rb +46 -0
  115. data/lib/rubocop/cop/style/redundant_regexp_escape.rb +2 -1
  116. data/lib/rubocop/cop/style/redundant_self_assignment_branch.rb +3 -1
  117. data/lib/rubocop/cop/style/redundant_sort.rb +1 -1
  118. data/lib/rubocop/cop/style/redundant_string_escape.rb +2 -0
  119. data/lib/rubocop/cop/style/regexp_literal.rb +11 -2
  120. data/lib/rubocop/cop/style/require_order.rb +11 -5
  121. data/lib/rubocop/cop/style/rescue_modifier.rb +1 -3
  122. data/lib/rubocop/cop/style/return_nil_in_predicate_method_definition.rb +81 -0
  123. data/lib/rubocop/cop/style/select_by_regexp.rb +15 -5
  124. data/lib/rubocop/cop/style/semicolon.rb +12 -1
  125. data/lib/rubocop/cop/style/signal_exception.rb +1 -1
  126. data/lib/rubocop/cop/style/single_line_methods.rb +1 -1
  127. data/lib/rubocop/cop/style/sole_nested_conditional.rb +3 -1
  128. data/lib/rubocop/cop/style/special_global_vars.rb +3 -4
  129. data/lib/rubocop/cop/style/yaml_file_read.rb +66 -0
  130. data/lib/rubocop/cop/style/yoda_condition.rb +4 -2
  131. data/lib/rubocop/cop/team.rb +1 -1
  132. data/lib/rubocop/cop/util.rb +1 -1
  133. data/lib/rubocop/cop/utils/regexp_ranges.rb +100 -0
  134. data/lib/rubocop/cop/variable_force/assignment.rb +47 -4
  135. data/lib/rubocop/cop/variable_force/variable_table.rb +2 -2
  136. data/lib/rubocop/cop/variable_force.rb +1 -0
  137. data/lib/rubocop/cops_documentation_generator.rb +1 -1
  138. data/lib/rubocop/ext/regexp_parser.rb +4 -1
  139. data/lib/rubocop/lsp/logger.rb +22 -0
  140. data/lib/rubocop/lsp/routes.rb +231 -0
  141. data/lib/rubocop/lsp/runtime.rb +82 -0
  142. data/lib/rubocop/lsp/server.rb +66 -0
  143. data/lib/rubocop/lsp/severity.rb +27 -0
  144. data/lib/rubocop/options.rb +11 -1
  145. data/lib/rubocop/result_cache.rb +1 -1
  146. data/lib/rubocop/rspec/cop_helper.rb +1 -1
  147. data/lib/rubocop/server/client_command/exec.rb +2 -1
  148. data/lib/rubocop/target_ruby.rb +3 -2
  149. data/lib/rubocop/version.rb +10 -6
  150. data/lib/rubocop.rb +13 -0
  151. metadata +38 -6
@@ -135,7 +135,8 @@ module RuboCop
135
135
  return if nodes.all?(&:single_line?) && cop_config['AllowAdjacentOneLineDefs']
136
136
 
137
137
  correction_node = nodes.last
138
- location = correction_node.loc.keyword.join(correction_node.loc.name)
138
+
139
+ location = def_location(correction_node)
139
140
  add_offense(location, message: message(correction_node, count: count)) do |corrector|
140
141
  autocorrect(corrector, *nodes, count)
141
142
  end
@@ -159,10 +160,28 @@ module RuboCop
159
160
 
160
161
  private
161
162
 
163
+ def def_location(correction_node)
164
+ if correction_node.block_type?
165
+ correction_node.source_range.join(correction_node.children.first.source_range)
166
+ else
167
+ correction_node.loc.keyword.join(correction_node.loc.name)
168
+ end
169
+ end
170
+
162
171
  def candidate?(node)
163
- return unless node
172
+ return false unless node
164
173
 
165
- method_candidate?(node) || class_candidate?(node) || module_candidate?(node)
174
+ method_candidate?(node) || class_candidate?(node) || module_candidate?(node) ||
175
+ macro_candidate?(node)
176
+ end
177
+
178
+ def empty_line_between_macros
179
+ cop_config.fetch('DefLikeMacros', []).map(&:to_sym)
180
+ end
181
+
182
+ def macro_candidate?(node)
183
+ node.block_type? && node.children.first.macro? &&
184
+ empty_line_between_macros.include?(node.children.first.method_name)
166
185
  end
167
186
 
168
187
  def method_candidate?(node)
@@ -226,7 +245,11 @@ module RuboCop
226
245
  end
227
246
 
228
247
  def def_start(node)
229
- node.loc.keyword.line
248
+ if node.block_type? && node.children.first.send_type?
249
+ node.source_range.line
250
+ else
251
+ node.loc.keyword.line
252
+ end
230
253
  end
231
254
 
232
255
  def def_end(node)
@@ -68,6 +68,8 @@ module RuboCop
68
68
  check_body(node.body, node.loc.line)
69
69
  end
70
70
  alias on_defs on_def
71
+ alias on_block on_def
72
+ alias on_numblock on_def
71
73
 
72
74
  def on_kwbegin(node)
73
75
  body, = *node
@@ -228,9 +228,9 @@ module RuboCop
228
228
  end
229
229
 
230
230
  def find_most_bottom_of_heredoc_end(arguments)
231
- arguments.map do |argument|
231
+ arguments.filter_map do |argument|
232
232
  argument.loc.heredoc_end.end_pos if argument.loc.respond_to?(:heredoc_end)
233
- end.compact.max
233
+ end.max
234
234
  end
235
235
 
236
236
  # Internal trailing comma helpers.
@@ -76,7 +76,7 @@ module RuboCop
76
76
 
77
77
  def autocorrect_lambda_for_tabs(corrector, range)
78
78
  spaces = ' ' * configured_indentation_width
79
- corrector.replace(range, range.source.gsub(/\t/, spaces))
79
+ corrector.replace(range, range.source.gsub("\t", spaces))
80
80
  end
81
81
 
82
82
  def autocorrect_lambda_for_spaces(corrector, range)
@@ -366,10 +366,10 @@ module RuboCop
366
366
  end
367
367
 
368
368
  def starts_with_access_modifier?(body_node)
369
- return unless body_node.begin_type?
369
+ return false unless body_node.begin_type?
370
370
 
371
371
  starting_node = body_node.children.first
372
- return unless starting_node
372
+ return false unless starting_node
373
373
 
374
374
  starting_node.send_type? && starting_node.bare_access_modifier?
375
375
  end
@@ -84,6 +84,8 @@ module RuboCop
84
84
  return unless strings_concatenated_with_backslash?(node)
85
85
 
86
86
  children = node.children
87
+ return if children.empty?
88
+
87
89
  if style == :aligned && !always_indented?(node)
88
90
  check_aligned(children, 1)
89
91
  else
@@ -99,7 +99,7 @@ module RuboCop
99
99
  def suitable_as_single_line?(node)
100
100
  !comment_within?(node) &&
101
101
  node.each_descendant(:if, :case, :kwbegin, :def).none? &&
102
- node.each_descendant(:dstr, :str).none?(&:heredoc?) &&
102
+ node.each_descendant(:dstr, :str).none? { |n| n.heredoc? || n.value.include?("\n") } &&
103
103
  node.each_descendant(:begin).none? { |b| !b.single_line? }
104
104
  end
105
105
 
@@ -24,7 +24,15 @@ module RuboCop
24
24
  end
25
25
 
26
26
  def kind(token)
27
- 'comma' if token.comma?
27
+ 'comma' if token.comma? && !before_semicolon?(token)
28
+ end
29
+
30
+ private
31
+
32
+ def before_semicolon?(token)
33
+ tokens = processed_source.tokens
34
+
35
+ tokens[tokens.index(token) + 1].semicolon?
28
36
  end
29
37
  end
30
38
  end
@@ -153,7 +153,9 @@ module RuboCop
153
153
  private
154
154
 
155
155
  def regular_operator?(send_node)
156
- !send_node.unary_operation? && !send_node.dot? && operator_with_regular_syntax?(send_node)
156
+ return false if send_node.unary_operation? || send_node.dot? || send_node.double_colon?
157
+
158
+ operator_with_regular_syntax?(send_node)
157
159
  end
158
160
 
159
161
  def operator_with_regular_syntax?(send_node)
@@ -236,6 +236,8 @@ module RuboCop
236
236
  end
237
237
 
238
238
  def offense(begin_pos, end_pos, msg, style_param = 'EnforcedStyle')
239
+ return if begin_pos > end_pos
240
+
239
241
  range = range_between(begin_pos, end_pos)
240
242
  add_offense(range, message: msg) do |corrector|
241
243
  case range.source
@@ -35,7 +35,7 @@ module RuboCop
35
35
  def check(node)
36
36
  expression = node.source
37
37
  op = node.loc.operator.source
38
- escaped_op = op.gsub(/\./, '\.')
38
+ escaped_op = op.gsub('.', '\.')
39
39
 
40
40
  # account for multiline range literals
41
41
  expression.sub!(/#{escaped_op}\n\s*/, op)
@@ -52,6 +52,8 @@ module RuboCop
52
52
  # expect { do_something }.to not_change { object.attribute }
53
53
  #
54
54
  class AmbiguousBlockAssociation < Base
55
+ extend AutoCorrector
56
+
55
57
  include AllowedMethods
56
58
  include AllowedPattern
57
59
 
@@ -68,7 +70,9 @@ module RuboCop
68
70
 
69
71
  message = message(node)
70
72
 
71
- add_offense(node, message: message)
73
+ add_offense(node, message: message) do |corrector|
74
+ wrap_in_parentheses(corrector, node)
75
+ end
72
76
  end
73
77
  alias on_csend on_send
74
78
 
@@ -89,6 +93,14 @@ module RuboCop
89
93
 
90
94
  format(MSG, param: block_param.source, method: block_param.send_node.source)
91
95
  end
96
+
97
+ def wrap_in_parentheses(corrector, node)
98
+ range = node.loc.selector.end.join(node.first_argument.source_range.begin)
99
+
100
+ corrector.remove(range)
101
+ corrector.insert_before(range, '(')
102
+ corrector.insert_after(node.last_argument, ')')
103
+ end
92
104
  end
93
105
  end
94
106
  end
@@ -68,10 +68,7 @@ module RuboCop
68
68
  MSG = 'Remove debugger entry point `%<source>s`.'
69
69
 
70
70
  def on_send(node)
71
- return unless debugger_method?(node)
72
-
73
- # Basically, debugger methods are not used as a method argument without arguments.
74
- return if node.arguments.empty? && node.each_ancestor(:send, :csend).any?
71
+ return if !debugger_method?(node) || assumed_usage_context?(node)
75
72
 
76
73
  add_offense(node)
77
74
  end
@@ -90,11 +87,18 @@ module RuboCop
90
87
  end
91
88
 
92
89
  def debugger_method?(send_node)
93
- return if send_node.parent&.send_type? && send_node.parent.receiver == send_node
90
+ return false if send_node.parent&.send_type? && send_node.parent.receiver == send_node
94
91
 
95
92
  debugger_methods.include?(chained_method_name(send_node))
96
93
  end
97
94
 
95
+ def assumed_usage_context?(node)
96
+ # Basically, debugger methods are not used as a method argument without arguments.
97
+ return false unless node.arguments.empty? && node.each_ancestor(:send, :csend).any?
98
+
99
+ node.each_ancestor.none?(&:lambda_or_proc?)
100
+ end
101
+
98
102
  def chained_method_name(send_node)
99
103
  chained_method_name = send_node.method_name.to_s
100
104
  receiver = send_node.receiver
@@ -4,6 +4,7 @@ module RuboCop
4
4
  module Cop
5
5
  module Lint
6
6
  # Checks for duplicated keys in hash literals.
7
+ # This cop considers both primitive types and constants for the hash keys.
7
8
  #
8
9
  # This cop mirrors a warning in Ruby 2.2.
9
10
  #
@@ -24,7 +25,7 @@ module RuboCop
24
25
  MSG = 'Duplicated key in hash literal.'
25
26
 
26
27
  def on_hash(node)
27
- keys = node.keys.select(&:recursive_basic_literal?)
28
+ keys = node.keys.select { |key| key.recursive_basic_literal? || key.const_type? }
28
29
 
29
30
  return unless duplicates?(keys)
30
31
 
@@ -24,6 +24,8 @@ module RuboCop
24
24
 
25
25
  MSG_REPEATED_ELEMENT = 'Duplicate element inside regexp character class'
26
26
 
27
+ OCTAL_DIGITS_AFTER_ESCAPE = 2
28
+
27
29
  def on_regexp(node)
28
30
  each_repeated_character_class_element_loc(node) do |loc|
29
31
  add_offense(loc, message: MSG_REPEATED_ELEMENT) do |corrector|
@@ -32,35 +34,57 @@ module RuboCop
32
34
  end
33
35
  end
34
36
 
35
- # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
36
37
  def each_repeated_character_class_element_loc(node)
37
38
  node.parsed_tree&.each_expression do |expr|
38
39
  next if skip_expression?(expr)
39
40
 
40
41
  seen = Set.new
41
- enum = expr.expressions.to_enum
42
- expression_count = expr.expressions.count
42
+ group_expressions(node, expr.expressions) do |group|
43
+ group_source = group.map(&:to_s).join
43
44
 
44
- expression_count.times do |current_number|
45
- current_child = enum.next
46
- next if within_interpolation?(node, current_child)
45
+ yield source_range(group) if seen.include?(group_source)
47
46
 
48
- current_child_source = current_child.to_s
49
- next_child = enum.peek if current_number + 1 < expression_count
47
+ seen << group_source
48
+ end
49
+ end
50
+ end
50
51
 
51
- if seen.include?(current_child_source)
52
- next if start_with_escaped_zero_number?(current_child_source, next_child.to_s)
52
+ private
53
53
 
54
- yield current_child.expression
55
- end
54
+ def group_expressions(node, expressions)
55
+ # Create a mutable list to simplify state tracking while we iterate.
56
+ expressions = expressions.to_a
56
57
 
57
- seen << current_child_source
58
- end
58
+ until expressions.empty?
59
+ # With we may need to compose a group of multiple expressions.
60
+ group = [expressions.shift]
61
+ next if within_interpolation?(node, group.first)
62
+
63
+ # With regexp_parser < 2.7 escaped octal sequences may be up to 3
64
+ # separate expressions ("\\0", "0", "1").
65
+ pop_octal_digits(group, expressions) if escaped_octal?(group.first.to_s)
66
+
67
+ yield(group)
59
68
  end
60
69
  end
61
- # rubocop:enable Metrics/AbcSize, Metrics/MethodLength
62
70
 
63
- private
71
+ def pop_octal_digits(current_child, expressions)
72
+ OCTAL_DIGITS_AFTER_ESCAPE.times do
73
+ next_child = expressions.first
74
+ break unless octal?(next_child.to_s)
75
+
76
+ current_child << expressions.shift
77
+ end
78
+ end
79
+
80
+ def source_range(children)
81
+ return children.first.expression if children.size == 1
82
+
83
+ range_between(
84
+ children.first.expression.begin_pos,
85
+ children.last.expression.begin_pos + children.last.to_s.length
86
+ )
87
+ end
64
88
 
65
89
  def skip_expression?(expr)
66
90
  expr.type != :set || expr.token == :intersection
@@ -75,9 +99,12 @@ module RuboCop
75
99
  interpolation_locs(node).any? { |il| il.overlaps?(parse_tree_child_loc) }
76
100
  end
77
101
 
78
- def start_with_escaped_zero_number?(current_child, next_child)
79
- # Represents escaped code from `"\00"` (`"\u0000"`) to `"\07"` (`"\a"`).
80
- current_child == '\\0' && next_child.match?(/[0-7]/)
102
+ def escaped_octal?(string)
103
+ string.length == 2 && string[0] == '\\' && octal?(string[1])
104
+ end
105
+
106
+ def octal?(char)
107
+ ('0'..'7').cover?(char)
81
108
  end
82
109
 
83
110
  def interpolation_locs(node)
@@ -3,14 +3,13 @@
3
3
  module RuboCop
4
4
  module Cop
5
5
  module Lint
6
- #
7
- # This cop emulates the following Ruby warnings in Ruby 2.6.
6
+ # Emulates the following Ruby warnings in Ruby 2.6.
8
7
  #
9
8
  # [source,console]
10
9
  # ----
11
- # % cat example.rb
10
+ # $ cat example.rb
12
11
  # ERB.new('hi', nil, '-', '@output_buffer')
13
- # % ruby -rerb example.rb
12
+ # $ ruby -rerb example.rb
14
13
  # example.rb:1: warning: Passing safe_level with the 2nd argument of ERB.new is
15
14
  # deprecated. Do not use it, and specify other arguments as keyword arguments.
16
15
  # example.rb:1: warning: Passing trim_mode with the 3rd argument of ERB.new is
@@ -65,7 +65,7 @@ module RuboCop
65
65
  end
66
66
 
67
67
  def send_node?(node)
68
- return nil unless node
68
+ return false unless node
69
69
 
70
70
  node.call_type?
71
71
  end
@@ -3,7 +3,6 @@
3
3
  module RuboCop
4
4
  module Cop
5
5
  module Lint
6
- #
7
6
  # Prefer `equal?` over `==` when comparing `object_id`.
8
7
  #
9
8
  # `Object#equal?` is provided to compare objects for identity, and in contrast
@@ -3,8 +3,10 @@
3
3
  module RuboCop
4
4
  module Cop
5
5
  module Lint
6
+ # Checks for `IO.select` that is incompatible with Fiber Scheduler since Ruby 3.0.
6
7
  #
7
- # This cop checks for `IO.select` that is incompatible with Fiber Scheduler since Ruby 3.0.
8
+ # When an array of IO objects waiting for an exception (the third argument of `IO.select`)
9
+ # is used as an argument, there is no alternative API, so offenses are not registered.
8
10
  #
9
11
  # NOTE: When the method is successful the return value of `IO.select` is `[[IO]]`,
10
12
  # and the return value of `io.wait_readable` and `io.wait_writable` are `self`.
@@ -42,8 +44,8 @@ module RuboCop
42
44
  PATTERN
43
45
 
44
46
  def on_send(node)
45
- read, write, _excepts, timeout = *io_select(node)
46
- return unless read
47
+ read, write, excepts, timeout = *io_select(node)
48
+ return if excepts && !excepts.children.empty?
47
49
  return unless scheduler_compatible?(read, write) || scheduler_compatible?(write, read)
48
50
 
49
51
  preferred = preferred_method(read, write, timeout)
@@ -58,6 +58,7 @@ module RuboCop
58
58
 
59
59
  def on_class(node)
60
60
  return unless node.parent_class && exception_class?(node.parent_class)
61
+ return if inherit_exception_class_with_omitted_namespace?(node)
61
62
 
62
63
  message = message(node.parent_class)
63
64
 
@@ -87,6 +88,14 @@ module RuboCop
87
88
  class_node.const_name == 'Exception'
88
89
  end
89
90
 
91
+ def inherit_exception_class_with_omitted_namespace?(class_node)
92
+ return false if class_node.parent_class.namespace&.cbase_type?
93
+
94
+ class_node.left_siblings.any? do |sibling|
95
+ sibling.respond_to?(:identifier) && exception_class?(sibling.identifier)
96
+ end
97
+ end
98
+
90
99
  def preferred_base_class
91
100
  PREFERRED_BASE_CLASS[style]
92
101
  end
@@ -6,7 +6,7 @@ module RuboCop
6
6
  # Checks uses of lambda without a literal block.
7
7
  # It emulates the following warning in Ruby 3.0:
8
8
  #
9
- # % ruby -vwe 'lambda(&proc {})'
9
+ # $ ruby -vwe 'lambda(&proc {})'
10
10
  # ruby 3.0.0p0 (2020-12-25 revision 95aff21468) [x86_64-darwin19]
11
11
  # -e:1: warning: lambda without a literal block is deprecated; use the proc without
12
12
  # lambda instead
@@ -11,6 +11,16 @@ module RuboCop
11
11
  # missing method. In other cases, the theoretical ideal handling could be
12
12
  # challenging or verbose for no actual gain.
13
13
  #
14
+ # Autocorrection is not supported because the position of `super` cannot be
15
+ # determined automatically.
16
+ #
17
+ # `Object` and `BasicObject` are allowed by this cop because of their
18
+ # stateless nature. However, sometimes you might want to allow other parent
19
+ # classes from this cop, for example in the case of an abstract class that is
20
+ # not meant to be called with `super`. In those cases, you can use the
21
+ # `AllowedParentClasses` option to specify which classes should be allowed
22
+ # *in addition to* `Object` and `BasicObject`.
23
+ #
14
24
  # @example
15
25
  # # bad
16
26
  # class Employee < Person
@@ -57,6 +67,21 @@ module RuboCop
57
67
  # end
58
68
  # end
59
69
  #
70
+ # # good
71
+ # class ClassWithNoParent
72
+ # def initialize
73
+ # do_something
74
+ # end
75
+ # end
76
+ #
77
+ # @example AllowedParentClasses: [MyAbstractClass]
78
+ # # good
79
+ # class MyConcreteClass < MyAbstractClass
80
+ # def initialize
81
+ # do_something
82
+ # end
83
+ # end
84
+ #
60
85
  class MissingSuper < Base
61
86
  CONSTRUCTOR_MSG = 'Call `super` to initialize state of the parent class.'
62
87
  CALLBACK_MSG = 'Call `super` to invoke callback defined in the parent class.'
@@ -100,7 +125,7 @@ module RuboCop
100
125
  end
101
126
 
102
127
  def callback_method_def?(node)
103
- return unless CALLBACKS.include?(node.method_name)
128
+ return false unless CALLBACKS.include?(node.method_name)
104
129
 
105
130
  node.each_ancestor(:class, :sclass, :module).first
106
131
  end
@@ -113,16 +138,20 @@ module RuboCop
113
138
  if (block_node = node.each_ancestor(:block, :numblock).first)
114
139
  return false unless (super_class = class_new_block(block_node))
115
140
 
116
- !stateless_class?(super_class)
141
+ !allowed_class?(super_class)
117
142
  elsif (class_node = node.each_ancestor(:class).first)
118
- class_node.parent_class && !stateless_class?(class_node.parent_class)
143
+ class_node.parent_class && !allowed_class?(class_node.parent_class)
119
144
  else
120
145
  false
121
146
  end
122
147
  end
123
148
 
124
- def stateless_class?(node)
125
- STATELESS_CLASSES.include?(node.const_name)
149
+ def allowed_class?(node)
150
+ allowed_classes.include?(node.const_name)
151
+ end
152
+
153
+ def allowed_classes
154
+ @allowed_classes ||= STATELESS_CLASSES + cop_config.fetch('AllowedParentClasses', [])
126
155
  end
127
156
  end
128
157
  end
@@ -0,0 +1,111 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module Lint
6
+ # Checks for mixed-case character ranges since they include likely unintended characters.
7
+ #
8
+ # Offenses are registered for regexp character classes like `/[A-z]/`
9
+ # as well as range objects like `('A'..'z')`.
10
+ #
11
+ # NOTE: Range objects cannot be autocorrected.
12
+ #
13
+ # @safety
14
+ # The cop autocorrects regexp character classes
15
+ # by replacing one character range with two: `A-z` becomes `A-Za-z`.
16
+ # In most cases this is probably what was originally intended
17
+ # but it changes the regexp to no longer match symbols it used to include.
18
+ # For this reason, this cop's autocorrect is unsafe (it will
19
+ # change the behavior of the code).
20
+ #
21
+ # @example
22
+ #
23
+ # # bad
24
+ # r = /[A-z]/
25
+ #
26
+ # # good
27
+ # r = /[A-Za-z]/
28
+ class MixedCaseRange < Base
29
+ extend AutoCorrector
30
+ include RangeHelp
31
+
32
+ MSG = 'Ranges from upper to lower case ASCII letters may include unintended ' \
33
+ 'characters. Instead of `A-z` (which also includes several symbols) ' \
34
+ 'specify each range individually: `A-Za-z` and individually specify any symbols.'
35
+ RANGES = [('a'..'z').freeze, ('A'..'Z').freeze].freeze
36
+
37
+ def on_irange(node)
38
+ return unless node.children.compact.all?(&:str_type?)
39
+
40
+ range_start, range_end = node.children
41
+
42
+ return if range_start.nil? || range_end.nil?
43
+
44
+ add_offense(node) if unsafe_range?(range_start.value, range_end.value)
45
+ end
46
+ alias on_erange on_irange
47
+
48
+ def on_regexp(node)
49
+ each_unsafe_regexp_range(node) do |loc|
50
+ add_offense(loc) do |corrector|
51
+ corrector.replace(loc, rewrite_regexp_range(loc.source))
52
+ end
53
+ end
54
+ end
55
+
56
+ def each_unsafe_regexp_range(node)
57
+ node.parsed_tree&.each_expression do |expr|
58
+ next if skip_expression?(expr)
59
+
60
+ range_pairs(expr).reject do |range_start, range_end|
61
+ next if skip_range?(range_start, range_end)
62
+
63
+ next unless unsafe_range?(range_start.text, range_end.text)
64
+
65
+ yield(build_source_range(range_start, range_end))
66
+ end
67
+ end
68
+ end
69
+
70
+ private
71
+
72
+ def build_source_range(range_start, range_end)
73
+ range_between(range_start.expression.begin_pos, range_end.expression.end_pos)
74
+ end
75
+
76
+ def range_for(char)
77
+ RANGES.detect do |range|
78
+ range.include?(char)
79
+ end
80
+ end
81
+
82
+ def range_pairs(expr)
83
+ RuboCop::Cop::Utils::RegexpRanges.new(expr).pairs
84
+ end
85
+
86
+ def unsafe_range?(range_start, range_end)
87
+ return false if range_start.length != 1 || range_end.length != 1
88
+
89
+ range_for(range_start) != range_for(range_end)
90
+ end
91
+
92
+ def skip_expression?(expr)
93
+ !(expr.type == :set && expr.token == :character)
94
+ end
95
+
96
+ def skip_range?(range_start, range_end)
97
+ [range_start, range_end].any? do |bound|
98
+ bound.type == :escape
99
+ end
100
+ end
101
+
102
+ def rewrite_regexp_range(source)
103
+ open, close = source.split('-')
104
+ first = [open, range_for(open).end]
105
+ second = [range_for(close).begin, close]
106
+ "#{first.uniq.join('-')}#{second.uniq.join('-')}"
107
+ end
108
+ end
109
+ end
110
+ end
111
+ end
@@ -74,6 +74,7 @@ module RuboCop
74
74
  extend AutoCorrector
75
75
  include AllowedMethods
76
76
  include AllowedPattern
77
+ include IgnoredNode
77
78
 
78
79
  CONVERSION_METHOD_CLASS_MAPPING = {
79
80
  to_i: "#{Integer.name}(%<number_object>s, 10)",
@@ -116,7 +117,11 @@ module RuboCop
116
117
  corrected_method: correct_method(node, receiver)
117
118
  )
118
119
  add_offense(node, message: message) do |corrector|
120
+ next if part_of_ignored_node?(node)
121
+
119
122
  corrector.replace(node, correct_method(node, node.receiver))
123
+
124
+ ignore_node(node)
120
125
  end
121
126
  end
122
127
  end
@@ -6,13 +6,13 @@ module RuboCop
6
6
  # Checks for uses of numbered parameter assignment.
7
7
  # It emulates the following warning in Ruby 2.7:
8
8
  #
9
- # % ruby -ve '_1 = :value'
9
+ # $ ruby -ve '_1 = :value'
10
10
  # ruby 2.7.2p137 (2020-10-01 revision 5445e04352) [x86_64-darwin19]
11
11
  # -e:1: warning: `_1' is reserved for numbered parameter; consider another name
12
12
  #
13
13
  # Assigning to a numbered parameter (from `_1` to `_9`) causes an error in Ruby 3.0.
14
14
  #
15
- # % ruby -ve '_1 = :value'
15
+ # $ ruby -ve '_1 = :value'
16
16
  # ruby 3.0.0p0 (2020-12-25 revision 95aff21468) [x86_64-darwin19]
17
17
  # -e:1: _1 is reserved for numbered parameter
18
18
  #