rubocop 1.72.2 → 1.74.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (75) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +3 -3
  3. data/config/default.yml +56 -15
  4. data/config/internal_affairs.yml +20 -0
  5. data/lib/rubocop/config_loader.rb +0 -1
  6. data/lib/rubocop/config_loader_resolver.rb +4 -3
  7. data/lib/rubocop/config_obsoletion/extracted_cop.rb +4 -3
  8. data/lib/rubocop/config_obsoletion.rb +1 -1
  9. data/lib/rubocop/config_validator.rb +1 -1
  10. data/lib/rubocop/cop/internal_affairs/example_description.rb +3 -1
  11. data/lib/rubocop/cop/internal_affairs/node_type_group.rb +91 -0
  12. data/lib/rubocop/cop/internal_affairs.rb +1 -0
  13. data/lib/rubocop/cop/layout/closing_parenthesis_indentation.rb +4 -4
  14. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +26 -1
  15. data/lib/rubocop/cop/layout/line_length.rb +3 -3
  16. data/lib/rubocop/cop/lint/duplicate_methods.rb +0 -14
  17. data/lib/rubocop/cop/lint/empty_conditional_body.rb +14 -64
  18. data/lib/rubocop/cop/lint/erb_new_arguments.rb +0 -6
  19. data/lib/rubocop/cop/lint/float_comparison.rb +1 -6
  20. data/lib/rubocop/cop/lint/literal_as_condition.rb +103 -9
  21. data/lib/rubocop/cop/lint/mixed_case_range.rb +2 -2
  22. data/lib/rubocop/cop/lint/non_local_exit_from_iterator.rb +2 -2
  23. data/lib/rubocop/cop/lint/redundant_require_statement.rb +0 -21
  24. data/lib/rubocop/cop/lint/redundant_type_conversion.rb +32 -5
  25. data/lib/rubocop/cop/lint/return_in_void_context.rb +4 -11
  26. data/lib/rubocop/cop/lint/shared_mutable_default.rb +12 -1
  27. data/lib/rubocop/cop/lint/useless_constant_scoping.rb +2 -11
  28. data/lib/rubocop/cop/lint/void.rb +6 -0
  29. data/lib/rubocop/cop/mixin/check_single_line_suitability.rb +1 -1
  30. data/lib/rubocop/cop/mixin/hash_subset.rb +19 -4
  31. data/lib/rubocop/cop/mixin/range_help.rb +12 -0
  32. data/lib/rubocop/cop/mixin/target_ruby_version.rb +1 -1
  33. data/lib/rubocop/cop/mixin/trailing_comma.rb +12 -0
  34. data/lib/rubocop/cop/naming/variable_name.rb +64 -6
  35. data/lib/rubocop/cop/style/accessor_grouping.rb +19 -5
  36. data/lib/rubocop/cop/style/class_and_module_children.rb +29 -7
  37. data/lib/rubocop/cop/style/commented_keyword.rb +10 -3
  38. data/lib/rubocop/cop/style/comparable_between.rb +75 -0
  39. data/lib/rubocop/cop/style/double_negation.rb +1 -1
  40. data/lib/rubocop/cop/style/endless_method.rb +163 -18
  41. data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -7
  42. data/lib/rubocop/cop/style/exponential_notation.rb +2 -2
  43. data/lib/rubocop/cop/style/format_string_token.rb +38 -11
  44. data/lib/rubocop/cop/style/if_unless_modifier.rb +2 -2
  45. data/lib/rubocop/cop/style/inverse_methods.rb +8 -5
  46. data/lib/rubocop/cop/style/keyword_parameters_order.rb +13 -7
  47. data/lib/rubocop/cop/style/line_end_concatenation.rb +10 -4
  48. data/lib/rubocop/cop/style/method_call_with_args_parentheses/omit_parentheses.rb +3 -3
  49. data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +1 -1
  50. data/lib/rubocop/cop/style/multiline_block_chain.rb +1 -1
  51. data/lib/rubocop/cop/style/multiline_method_signature.rb +1 -9
  52. data/lib/rubocop/cop/style/redundant_condition.rb +45 -0
  53. data/lib/rubocop/cop/style/redundant_current_directory_in_path.rb +14 -4
  54. data/lib/rubocop/cop/style/redundant_format.rb +23 -11
  55. data/lib/rubocop/cop/style/redundant_freeze.rb +1 -1
  56. data/lib/rubocop/cop/style/redundant_self_assignment.rb +1 -1
  57. data/lib/rubocop/cop/style/rescue_modifier.rb +3 -0
  58. data/lib/rubocop/cop/style/single_line_methods.rb +3 -3
  59. data/lib/rubocop/cop/style/sole_nested_conditional.rb +0 -6
  60. data/lib/rubocop/cop/style/string_concatenation.rb +1 -1
  61. data/lib/rubocop/cop/style/trailing_comma_in_array_literal.rb +47 -6
  62. data/lib/rubocop/cop/style/trailing_comma_in_hash_literal.rb +48 -6
  63. data/lib/rubocop/cop/style/trivial_accessors.rb +1 -1
  64. data/lib/rubocop/cop/utils/format_string.rb +5 -2
  65. data/lib/rubocop/cops_documentation_generator.rb +12 -1
  66. data/lib/rubocop/directive_comment.rb +1 -1
  67. data/lib/rubocop/ext/regexp_node.rb +0 -1
  68. data/lib/rubocop/plugin/load_error.rb +1 -1
  69. data/lib/rubocop/plugin.rb +9 -2
  70. data/lib/rubocop/rspec/shared_contexts.rb +15 -0
  71. data/lib/rubocop/rspec/support.rb +1 -0
  72. data/lib/rubocop/version.rb +1 -1
  73. data/lib/rubocop.rb +1 -1
  74. metadata +6 -5
  75. data/lib/rubocop/cop/utils/regexp_ranges.rb +0 -113
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 4a6ff0f849d961da13d2131ff60e0ed49dce01a4c2d32caa8484a2344344609b
4
- data.tar.gz: 9da9fc43212c62cd6542649c90423e9365f50cb00e82b6337be1a241c6a17283
3
+ metadata.gz: 440bee0e28f294bab2eba4c11e9c681342917c69ad4c00e36eceb88743767c29
4
+ data.tar.gz: 68e9e209a22e891a38b677f97344d2d10e9a8913dd999751472f301deed43b31
5
5
  SHA512:
6
- metadata.gz: 7ff34957554167b3b1e30645e3754094968fae93d0284a3882460b3e24f44fd64bcf358b23fd0e8154489206fd9c95fc6555b551f1af5ac810805fac2b6d27be
7
- data.tar.gz: 48e989049de37523554c740ba1368e32de38bc0f879c1fc717924a758a1ba465ef4413cba80a42be59ccaaf87616b52c0ad437256702bf5fd091c4a5818f428a
6
+ metadata.gz: 70db27a5b7a0e00696672a5b4fcd4e3bf35aa40c85ef65ccb630914ab110a5e5b657927aa1b1c8637be76b7041a90e2ad3c4ed188110266c048d0c4d6a4be0b4
7
+ data.tar.gz: e5599d30a4f776d85d202acad2ac03c7e4568e64a1c4038cb87b8ee3372b2428882924a91ea49853c98cf0614280d002918363eef41fe0990dc8b1895d8e9776
data/README.md CHANGED
@@ -52,7 +52,7 @@ To prevent an unwanted RuboCop update you might want to use a conservative versi
52
52
  in your `Gemfile`:
53
53
 
54
54
  ```rb
55
- gem 'rubocop', '~> 1.72', require: false
55
+ gem 'rubocop', '~> 1.74', require: false
56
56
  ```
57
57
 
58
58
  See [our versioning policy](https://docs.rubocop.org/rubocop/versioning.html) for further details.
@@ -241,9 +241,9 @@ Become a sponsor and get your logo on our README on GitHub with a link to your s
241
241
  <a href="https://opencollective.com/rubocop/organization/28/website" target="_blank"><img src="https://opencollective.com/rubocop/organization/28/avatar.svg"></a>
242
242
  <a href="https://opencollective.com/rubocop/organization/29/website" target="_blank"><img src="https://opencollective.com/rubocop/organization/29/avatar.svg"></a>
243
243
 
244
- ## Changelog
244
+ ## Release Notes
245
245
 
246
- RuboCop's changelog is available [here](CHANGELOG.md).
246
+ RuboCop's release notes are available [here](https://github.com/rubocop/rubocop/releases).
247
247
 
248
248
  ## Copyright
249
249
 
data/config/default.yml CHANGED
@@ -1887,10 +1887,9 @@ Lint/EmptyConditionalBody:
1887
1887
  Description: 'Checks for the presence of `if`, `elsif` and `unless` branches without a body.'
1888
1888
  Enabled: true
1889
1889
  AutoCorrect: contextual
1890
- SafeAutoCorrect: false
1891
1890
  AllowComments: true
1892
1891
  VersionAdded: '0.89'
1893
- VersionChanged: '1.61'
1892
+ VersionChanged: '1.73'
1894
1893
 
1895
1894
  Lint/EmptyEnsure:
1896
1895
  Description: 'Checks for empty ensure block.'
@@ -2046,6 +2045,7 @@ Lint/LambdaWithoutLiteralBlock:
2046
2045
  Lint/LiteralAsCondition:
2047
2046
  Description: 'Checks of literals used in conditions.'
2048
2047
  Enabled: true
2048
+ AutoCorrect: contextual
2049
2049
  VersionAdded: '0.51'
2050
2050
 
2051
2051
  Lint/LiteralAssignmentInCondition:
@@ -2259,9 +2259,8 @@ Lint/RedundantRegexpQuantifiers:
2259
2259
  Lint/RedundantRequireStatement:
2260
2260
  Description: 'Checks for unnecessary `require` statement.'
2261
2261
  Enabled: true
2262
- SafeAutoCorrect: false
2263
2262
  VersionAdded: '0.76'
2264
- VersionChanged: '1.57'
2263
+ VersionChanged: '1.73'
2265
2264
 
2266
2265
  Lint/RedundantSafeNavigation:
2267
2266
  Description: 'Checks for redundant safe navigation calls.'
@@ -3081,13 +3080,15 @@ Naming/VariableName:
3081
3080
  StyleGuide: '#snake-case-symbols-methods-vars'
3082
3081
  Enabled: true
3083
3082
  VersionAdded: '0.50'
3084
- VersionChanged: '1.8'
3083
+ VersionChanged: '1.73'
3085
3084
  EnforcedStyle: snake_case
3086
3085
  SupportedStyles:
3087
3086
  - snake_case
3088
3087
  - camelCase
3089
3088
  AllowedIdentifiers: []
3090
3089
  AllowedPatterns: []
3090
+ ForbiddenIdentifiers: []
3091
+ ForbiddenPatterns: []
3091
3092
 
3092
3093
  Naming/VariableNumber:
3093
3094
  Description: 'Use the configured style when numbering symbols, methods and variables.'
@@ -3103,6 +3104,8 @@ Naming/VariableNumber:
3103
3104
  CheckMethodNames: true
3104
3105
  CheckSymbols: true
3105
3106
  AllowedIdentifiers:
3107
+ - TLS1_1 # OpenSSL::SSL::TLS1_1_VERSION
3108
+ - TLS1_2 # OpenSSL::SSL::TLS1_2_VERSION
3106
3109
  - capture3 # Open3.capture3
3107
3110
  - iso8601 # Time#iso8601
3108
3111
  - rfc1123_date # CGI.rfc1123_date
@@ -3498,6 +3501,7 @@ Style/ClassAndModuleChildren:
3498
3501
  SafeAutoCorrect: false
3499
3502
  Enabled: true
3500
3503
  VersionAdded: '0.19'
3504
+ VersionChanged: '1.74'
3501
3505
  #
3502
3506
  # Basically there are two different styles:
3503
3507
  #
@@ -3513,7 +3517,21 @@ Style/ClassAndModuleChildren:
3513
3517
  #
3514
3518
  # The compact style is only forced, for classes or modules with one child.
3515
3519
  EnforcedStyle: nested
3516
- SupportedStyles:
3520
+ SupportedStyles: &supported_styles
3521
+ - nested
3522
+ - compact
3523
+ # Configure classes separately, if desired. If not set, or set to `nil`,
3524
+ # the `EnforcedStyle` value will be used.
3525
+ EnforcedStyleForClasses: ~
3526
+ SupportedStylesForClasses:
3527
+ - ~
3528
+ - nested
3529
+ - compact
3530
+ # Configure modules separately, if desired. If not set, or set to `nil`,
3531
+ # the `EnforcedStyle` value will be used.
3532
+ EnforcedStyleForModules: ~
3533
+ SupportedStylesForModules:
3534
+ - ~
3517
3535
  - nested
3518
3536
  - compact
3519
3537
 
@@ -3666,6 +3684,12 @@ Style/CommentedKeyword:
3666
3684
  VersionAdded: '0.51'
3667
3685
  VersionChanged: '1.19'
3668
3686
 
3687
+ Style/ComparableBetween:
3688
+ Description: 'Enforces the use of `Comparable#between?` instead of logical comparison.'
3689
+ Enabled: pending
3690
+ VersionAdded: '1.74'
3691
+ StyleGuide: '#ranges-or-between'
3692
+
3669
3693
  Style/ComparableClamp:
3670
3694
  Description: 'Enforces the use of `Comparable#clamp` instead of comparison by minimum and maximum.'
3671
3695
  Enabled: pending
@@ -3922,6 +3946,8 @@ Style/EndlessMethod:
3922
3946
  - allow_single_line
3923
3947
  - allow_always
3924
3948
  - disallow
3949
+ - require_single_line
3950
+ - require_always
3925
3951
 
3926
3952
  Style/EnvHome:
3927
3953
  Description: "Checks for consistent usage of `ENV['HOME']`."
@@ -4066,8 +4092,14 @@ Style/FormatStringToken:
4066
4092
  # style token in a format string to be allowed when enforced style is not
4067
4093
  # `unannotated`.
4068
4094
  MaxUnannotatedPlaceholdersAllowed: 1
4095
+ # The mode the cop operates in. Two values are allowed:
4096
+ # * aggressive (default): all strings are considered
4097
+ # * conservative:
4098
+ # only register offenses for strings given to `printf`, `sprintf`,
4099
+ # format` and `%` methods. Other strings are not considered.
4100
+ Mode: aggressive
4069
4101
  VersionAdded: '0.49'
4070
- VersionChanged: '1.0'
4102
+ VersionChanged: '1.74'
4071
4103
  AllowedMethods: []
4072
4104
  AllowedPatterns: []
4073
4105
 
@@ -5104,6 +5136,9 @@ Style/RedundantCondition:
5104
5136
  Description: 'Checks for unnecessary conditional expressions.'
5105
5137
  Enabled: true
5106
5138
  VersionAdded: '0.76'
5139
+ VersionChanged: '1.73'
5140
+ AllowedMethods:
5141
+ - nonzero?
5107
5142
 
5108
5143
  Style/RedundantConditional:
5109
5144
  Description: "Don't return true/false from a conditional."
@@ -5701,14 +5736,17 @@ Style/TrailingCommaInArrayLiteral:
5701
5736
  StyleGuide: '#no-trailing-array-commas'
5702
5737
  Enabled: true
5703
5738
  VersionAdded: '0.53'
5704
- # If `comma`, the cop requires a comma after the last item in an array,
5705
- # but only when each item is on its own line.
5706
- # If `consistent_comma`, the cop requires a comma after the last item of all
5707
- # non-empty, multiline array literals.
5739
+ # If `comma`, the cop requires a comma after the last item in an array, but only when each item is
5740
+ # on its own line.
5741
+ # If `consistent_comma`, the cop requires a comma after the last item of all non-empty, multiline
5742
+ # array literals.
5743
+ # If `diff_comma`, the cop requires a comma after the last item of all non-empty, multiline array
5744
+ # literals, but only when that last item immediately precedes a newline.
5708
5745
  EnforcedStyleForMultiline: no_comma
5709
5746
  SupportedStylesForMultiline:
5710
5747
  - comma
5711
5748
  - consistent_comma
5749
+ - diff_comma
5712
5750
  - no_comma
5713
5751
 
5714
5752
  Style/TrailingCommaInBlockArgs:
@@ -5720,14 +5758,17 @@ Style/TrailingCommaInBlockArgs:
5720
5758
  Style/TrailingCommaInHashLiteral:
5721
5759
  Description: 'Checks for trailing comma in hash literals.'
5722
5760
  Enabled: true
5723
- # If `comma`, the cop requires a comma after the last item in a hash,
5724
- # but only when each item is on its own line.
5725
- # If `consistent_comma`, the cop requires a comma after the last item of all
5726
- # non-empty, multiline hash literals.
5761
+ # If `comma`, the cop requires a comma after the last item in a hash, but only when each item is
5762
+ # on its own line.
5763
+ # If `consistent_comma`, the cop requires a comma after the last item of all non-empty, multiline
5764
+ # hash literals.
5765
+ # If `diff_comma`, the cop requires a comma after the last item of all non-empty, multiline hash
5766
+ # literals, but only when that last item immediately precedes a newline.
5727
5767
  EnforcedStyleForMultiline: no_comma
5728
5768
  SupportedStylesForMultiline:
5729
5769
  - comma
5730
5770
  - consistent_comma
5771
+ - diff_comma
5731
5772
  - no_comma
5732
5773
  VersionAdded: '0.53'
5733
5774
 
@@ -6,6 +6,26 @@ InternalAffairs/CopDescription:
6
6
  Include:
7
7
  - 'lib/rubocop/cop/**/*.rb'
8
8
 
9
+ InternalAffairs/ExampleHeredocDelimiter:
10
+ Include:
11
+ - 'spec/rubocop/cop/**/*.rb'
12
+
13
+ InternalAffairs/ExampleDescription:
14
+ Include:
15
+ - 'spec/rubocop/cop/**/*.rb'
16
+
17
+ InternalAffairs/NodeTypeGroup:
18
+ Include:
19
+ - 'lib/rubocop/cop/**/*.rb'
20
+
21
+ InternalAffairs/OnSendWithoutOnCSend:
22
+ Include:
23
+ - 'lib/rubocop/cop/**/*.rb'
24
+
25
+ InternalAffairs/UndefinedConfig:
26
+ Include:
27
+ - 'lib/rubocop/cop/**/*.rb'
28
+
9
29
  InternalAffairs/UselessMessageAssertion:
10
30
  Include:
11
31
  - '**/*_spec.rb'
@@ -63,7 +63,6 @@ module RuboCop
63
63
  loaded_features = resolver.resolve_requires(path, hash)
64
64
  add_loaded_features(loaded_features)
65
65
 
66
- resolver.override_department_setting_for_cops({}, hash)
67
66
  resolver.resolve_inheritance_from_gems(hash)
68
67
  resolver.resolve_inheritance(path, hash, file, debug?)
69
68
  hash.delete('inherit_from')
@@ -9,7 +9,8 @@ module RuboCop
9
9
  # @api private
10
10
  class ConfigLoaderResolver # rubocop:disable Metrics/ClassLength
11
11
  def resolve_plugins(rubocop_config, plugins)
12
- return if (plugins = Array(plugins)).empty?
12
+ plugins = Array(plugins) - ConfigLoader.loaded_plugins.map { |plugin| plugin.about.name }
13
+ return if plugins.empty?
13
14
 
14
15
  Plugin.integrate_plugins(rubocop_config, plugins)
15
16
  end
@@ -123,7 +124,7 @@ module RuboCop
123
124
  elsif merge_hashes?(base_hash, derived_hash, key)
124
125
  result[key] = merge(base_hash[key], derived_hash[key], **opts)
125
126
  elsif should_union?(derived_hash, base_hash, opts[:inherit_mode], key)
126
- result[key] = base_hash[key] | derived_hash[key]
127
+ result[key] = Array(base_hash[key]) | Array(derived_hash[key])
127
128
  elsif opts[:debug]
128
129
  warn_on_duplicate_setting(base_hash, derived_hash, key, **opts)
129
130
  end
@@ -205,7 +206,7 @@ module RuboCop
205
206
  end
206
207
 
207
208
  def should_union?(derived_hash, base_hash, root_mode, key)
208
- return false unless base_hash[key].is_a?(Array)
209
+ return false unless base_hash[key].is_a?(Array) || derived_hash[key].is_a?(Array)
209
210
 
210
211
  derived_mode = derived_hash['inherit_mode']
211
212
  return false if should_override?(derived_mode, key)
@@ -15,7 +15,7 @@ module RuboCop
15
15
  end
16
16
 
17
17
  def violated?
18
- return false if feature_loaded?
18
+ return false if plugin_loaded?
19
19
 
20
20
  affected_cops.any?
21
21
  end
@@ -38,8 +38,9 @@ module RuboCop
38
38
  end
39
39
  end
40
40
 
41
- def feature_loaded?
42
- config.loaded_features.include?(gem)
41
+ def plugin_loaded?
42
+ # Plugins loaded via `require` are included in `loaded_features`.
43
+ config.loaded_plugins.include?(gem) || config.loaded_features.include?(gem)
43
44
  end
44
45
  end
45
46
  end
@@ -50,7 +50,7 @@ module RuboCop
50
50
  # Default rules for obsoletions are in config/obsoletion.yml
51
51
  # Additional rules files can be added with `RuboCop::ConfigObsoletion.files << filename`
52
52
  def load_rules # rubocop:disable Metrics/AbcSize
53
- rules = LOAD_RULES_CACHE[self.class.files] ||=
53
+ rules = LOAD_RULES_CACHE[self.class.files.hash] ||=
54
54
  self.class.files.each_with_object({}) do |filename, hash|
55
55
  hash.merge!(YAML.safe_load(File.read(filename)) || {}) do |_key, first, second|
56
56
  case first
@@ -9,7 +9,7 @@ module RuboCop
9
9
 
10
10
  # @api private
11
11
  COMMON_PARAMS = %w[Exclude Include Severity inherit_mode AutoCorrect StyleGuide Details
12
- Enabled].freeze
12
+ Enabled Reference].freeze
13
13
  # @api private
14
14
  INTERNAL_PARAMS = %w[Description StyleGuide
15
15
  VersionAdded VersionChanged VersionRemoved
@@ -50,10 +50,12 @@ module RuboCop
50
50
  }.freeze
51
51
 
52
52
  EXPECT_NO_CORRECTIONS_DESCRIPTION_MAPPING = {
53
- /\A(auto[- ]?)?correct/ => 'does not correct'
53
+ /\A(auto[- ]?)?corrects?/ => 'does not correct',
54
+ /\band (auto[- ]?)?corrects/ => 'but does not correct'
54
55
  }.freeze
55
56
 
56
57
  EXPECT_CORRECTION_DESCRIPTION_MAPPING = {
58
+ /\bbut (does not|doesn't) (auto[- ]?)?correct/ => 'and autocorrects',
57
59
  /\b(does not|doesn't) (auto[- ]?)?correct/ => 'autocorrects'
58
60
  }.freeze
59
61
 
@@ -0,0 +1,91 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module InternalAffairs
6
+ # Checks that node types are checked against their group when all types of a
7
+ # group are checked.
8
+ #
9
+ # @example
10
+ # # bad
11
+ # node.type?(:irange, :erange)
12
+ #
13
+ # # good
14
+ # node.range_type?
15
+ #
16
+ # # bad
17
+ # node.type?(:irange, :erange, :send, :csend)
18
+ #
19
+ # # good
20
+ # node.type?(:range, :call)
21
+ #
22
+ class NodeTypeGroup < Base
23
+ extend AutoCorrector
24
+ include RangeHelp
25
+
26
+ MSG = 'Use `:%<group>s` instead of individually listing group types.'
27
+
28
+ RESTRICT_ON_SEND = %i[type? each_ancestor each_child_node each_descendant each_node].freeze
29
+
30
+ def on_send(node)
31
+ return unless node.receiver
32
+
33
+ symbol_args = node.arguments.select(&:sym_type?)
34
+ return if symbol_args.none?
35
+
36
+ NodePatternGroups::NODE_GROUPS.each do |group_name, group_types|
37
+ next unless group_satisfied?(group_types, symbol_args)
38
+
39
+ offense_range = arguments_range(node)
40
+ add_offense(offense_range, message: format(MSG, group: group_name)) do |corrector|
41
+ autocorrect(corrector, node, symbol_args, group_name, group_types)
42
+ end
43
+ end
44
+ end
45
+ alias on_csend on_send
46
+
47
+ private
48
+
49
+ def arguments_range(node)
50
+ range_between(
51
+ node.first_argument.source_range.begin_pos,
52
+ node.last_argument.source_range.end_pos
53
+ )
54
+ end
55
+
56
+ def group_satisfied?(group_types, symbol_args)
57
+ group_types.all? { |type| symbol_args.any? { |arg| arg.value == type } }
58
+ end
59
+
60
+ def autocorrect(corrector, node, symbol_args, group_name, group_types)
61
+ if node.method?(:type?) && node.arguments.count == group_types.count
62
+ autocorrect_to_explicit_predicate(corrector, node, group_name)
63
+ else
64
+ autocorrect_keep_method(corrector, symbol_args, group_name, group_types)
65
+ end
66
+ end
67
+
68
+ def autocorrect_to_explicit_predicate(corrector, node, group_name)
69
+ corrector.replace(node.selector, "#{group_name}_type?")
70
+ corrector.remove(arguments_range(node))
71
+ end
72
+
73
+ def autocorrect_keep_method(corrector, symbol_args, group_name, group_types)
74
+ first_replaced = false
75
+ symbol_args.each do |arg|
76
+ next unless group_types.include?(arg.value)
77
+
78
+ if first_replaced
79
+ range = range_with_surrounding_space(arg.source_range)
80
+ range = range_with_surrounding_comma(range, :left)
81
+ corrector.remove(range)
82
+ else
83
+ first_replaced = true
84
+ corrector.replace(arg, ":#{group_name}")
85
+ end
86
+ end
87
+ end
88
+ end
89
+ end
90
+ end
91
+ end
@@ -17,6 +17,7 @@ require_relative 'internal_affairs/node_destructuring'
17
17
  require_relative 'internal_affairs/node_first_or_last_argument'
18
18
  require_relative 'internal_affairs/node_matcher_directive'
19
19
  require_relative 'internal_affairs/node_pattern_groups'
20
+ require_relative 'internal_affairs/node_type_group'
20
21
  require_relative 'internal_affairs/node_type_multiple_predicates'
21
22
  require_relative 'internal_affairs/node_type_predicate'
22
23
  require_relative 'internal_affairs/numblock_handler'
@@ -155,10 +155,10 @@ module RuboCop
155
155
  end
156
156
 
157
157
  def all_elements_aligned?(elements)
158
- elements.flat_map do |e|
159
- if e.hash_type?
160
- e.each_child_node.map { |child| child.loc.column }
161
- else
158
+ if elements.first.hash_type?
159
+ elements.first.each_child_node.map { |child| child.loc.column }
160
+ else
161
+ elements.flat_map do |e|
162
162
  e.loc.column
163
163
  end
164
164
  end.uniq.count == 1
@@ -93,7 +93,7 @@ module RuboCop
93
93
  add_offense(node, message: message) do |corrector|
94
94
  line = range_by_whole_lines(node.source_range)
95
95
 
96
- corrector.insert_before(line, "\n") unless previous_line_empty?(node.first_line)
96
+ corrector.insert_before(line, "\n") if should_insert_line_before?(node)
97
97
 
98
98
  correct_next_line_if_denied_style(corrector, node, line)
99
99
  end
@@ -122,6 +122,8 @@ module RuboCop
122
122
  end
123
123
 
124
124
  def correct_next_line_if_denied_style(corrector, node, line)
125
+ return unless should_insert_line_after?(node)
126
+
125
127
  case style
126
128
  when :around
127
129
  corrector.insert_after(line, "\n") unless next_line_empty?(node.last_line)
@@ -205,6 +207,29 @@ module RuboCop
205
207
  format(MSG_BEFORE_FOR_ONLY_BEFORE, modifier: modifier)
206
208
  end
207
209
  end
210
+
211
+ def should_insert_line_before?(node)
212
+ return false if previous_line_empty?(node.first_line)
213
+ return true unless inside_block?(node) && no_empty_lines_around_block_body?
214
+ return true unless node.parent.begin_type?
215
+
216
+ node.parent.children.first != node
217
+ end
218
+
219
+ def should_insert_line_after?(node)
220
+ return true unless inside_block?(node) && no_empty_lines_around_block_body?
221
+
222
+ node.parent.children.last != node
223
+ end
224
+
225
+ def inside_block?(node)
226
+ node.parent.block_type? || (node.parent.begin_type? && node.parent.parent&.block_type?)
227
+ end
228
+
229
+ def no_empty_lines_around_block_body?
230
+ config.for_enabled_cop('Layout/EmptyLinesAroundBlockBody')['EnforcedStyle'] ==
231
+ 'no_empty_lines'
232
+ end
208
233
  end
209
234
  end
210
235
  end
@@ -209,7 +209,7 @@ module RuboCop
209
209
  # are not bisected.
210
210
  # If the string contains spaces, use them to determine a place for a clean break;
211
211
  # otherwise, the string will be broken at the line length limit.
212
- def breakable_string_range(node) # rubocop:disable Metrics/AbcSize
212
+ def breakable_string_range(node)
213
213
  source_range = node.source_range
214
214
  relevant_substr = largest_possible_string(node)
215
215
 
@@ -221,13 +221,13 @@ module RuboCop
221
221
  adjustment = max - source_range.last_column - 3
222
222
  return if adjustment.abs > source_range.size
223
223
 
224
- source_range.adjust(end_pos: max - source_range.last_column - 3)
224
+ source_range.adjust(end_pos: adjustment)
225
225
  end
226
226
  end
227
227
 
228
228
  def breakable_dstr_begin_position(node)
229
229
  source_range = node.source_range
230
- source_range.begin_pos if source_range.begin_pos < max && source_range.end_pos >= max
230
+ source_range.begin_pos if source_range.column < max && source_range.last_column >= max
231
231
  end
232
232
 
233
233
  def breakable_range_by_line_index
@@ -54,14 +54,12 @@ module RuboCop
54
54
  # if a method definition is inside an if, it is very likely
55
55
  # that a different definition is used depending on platform, etc.
56
56
  return if node.each_ancestor.any?(&:if_type?)
57
- return if possible_dsl?(node)
58
57
 
59
58
  found_instance_method(node, node.method_name)
60
59
  end
61
60
 
62
61
  def on_defs(node)
63
62
  return if node.each_ancestor.any?(&:if_type?)
64
- return if possible_dsl?(node)
65
63
 
66
64
  if node.receiver.const_type?
67
65
  _, const_name = *node.receiver
@@ -79,7 +77,6 @@ module RuboCop
79
77
  def on_alias(node)
80
78
  return unless (name = method_alias?(node))
81
79
  return if node.ancestors.any?(&:if_type?)
82
- return if possible_dsl?(node)
83
80
 
84
81
  found_instance_method(node, name)
85
82
  end
@@ -94,7 +91,6 @@ module RuboCop
94
91
  def on_send(node)
95
92
  if (name = alias_method?(node))
96
93
  return if node.ancestors.any?(&:if_type?)
97
- return if possible_dsl?(node)
98
94
 
99
95
  found_instance_method(node, name)
100
96
  elsif (attr = node.attribute_accessor?)
@@ -237,16 +233,6 @@ module RuboCop
237
233
  end
238
234
  end
239
235
 
240
- def possible_dsl?(node)
241
- # DSL methods may evaluate a block in the context of a newly created
242
- # class or module
243
- # Assume that if a method definition is inside any block call which
244
- # we can't identify, it could be a DSL
245
- node.each_ancestor(:block).any? do |ancestor|
246
- !ancestor.method?(:class_eval) && !ancestor.class_constructor?
247
- end
248
- end
249
-
250
236
  def source_location(node)
251
237
  range = node.source_range
252
238
  path = smart_path(range.source_buffer.name)