From a72d477117410bb2eaa55a55efb7081f4c49aaf7 Mon Sep 17 00:00:00 2001 From: Max Prokopiev Date: Fri, 16 Feb 2024 11:02:58 +0100 Subject: [PATCH] [Fix #1238] Add new `Rails/EnumSyntax` cop Defining enums with keyword arguments is deprecated and will be removed in Rails 8.0. --- .../new_add_new_rails_enum_syntax_cop.md | 1 + config/default.yml | 8 ++ lib/rubocop/cop/rails/enum_syntax.rb | 117 +++++++++++++++++ lib/rubocop/cop/rails_cops.rb | 1 + spec/rubocop/cop/rails/enum_syntax_spec.rb | 120 ++++++++++++++++++ 5 files changed, 247 insertions(+) create mode 100644 changelog/new_add_new_rails_enum_syntax_cop.md create mode 100644 lib/rubocop/cop/rails/enum_syntax.rb create mode 100644 spec/rubocop/cop/rails/enum_syntax_spec.rb diff --git a/changelog/new_add_new_rails_enum_syntax_cop.md b/changelog/new_add_new_rails_enum_syntax_cop.md new file mode 100644 index 0000000000..36166e0851 --- /dev/null +++ b/changelog/new_add_new_rails_enum_syntax_cop.md @@ -0,0 +1 @@ +* [#1238](https://github.com/rubocop/rubocop-rails/issues/1238): Add new `Rails/EnumSyntax` cop. ([@maxprokopiev][]) diff --git a/config/default.yml b/config/default.yml index 1f3b5376dc..f2b87a6367 100644 --- a/config/default.yml +++ b/config/default.yml @@ -424,6 +424,14 @@ Rails/EnumHash: Include: - app/models/**/*.rb +Rails/EnumSyntax: + Description: 'Use positional arguments over keyword arguments when defining enums.' + Enabled: pending + Severity: warning + VersionAdded: '<>' + Include: + - app/models/**/*.rb + Rails/EnumUniqueness: Description: 'Avoid duplicate integers in hash-syntax `enum` declaration.' Enabled: true diff --git a/lib/rubocop/cop/rails/enum_syntax.rb b/lib/rubocop/cop/rails/enum_syntax.rb new file mode 100644 index 0000000000..861bd40497 --- /dev/null +++ b/lib/rubocop/cop/rails/enum_syntax.rb @@ -0,0 +1,117 @@ +# frozen_string_literal: true + +module RuboCop + module Cop + module Rails + # Looks for enums written with keyword arguments syntax. + # + # Defining enums with keyword arguments syntax is deprecated and will be removed in Rails 8.0. + # Positional arguments should be used instead: + # + # @example + # # bad + # enum status: { active: 0, archived: 1 }, _prefix: true + # + # # good + # enum :status, { active: 0, archived: 1 }, prefix: true + # + class EnumSyntax < Base + extend AutoCorrector + extend TargetRailsVersion + + minimum_target_rails_version 7.0 + + MSG_ARGS = <<~MSG.chomp + Enum defined with keyword arguments found in `%s` enum declaration. Use positional arguments instead. + MSG + MSG_OPTS = <<~MSG.chomp + Enum defined with deprecated options found in `%s` enum declaration. Use options without the `_` prefix. + MSG + + RESTRICT_ON_SEND = %i[enum].freeze + + def_node_matcher :enum?, <<~PATTERN + (send nil? :enum (hash $...)) + PATTERN + + def_node_matcher :enum_with_options?, <<~PATTERN + (send nil? :enum $_ ${array hash} $_) + PATTERN + + def_node_matcher :enum_values, <<~PATTERN + (pair $_ ${array hash}) + PATTERN + + def_node_matcher :enum_options, <<~PATTERN + (pair $_ $_) + PATTERN + + def on_send(node) + check_and_correct_keyword_args(node) + check_enum_options(node) + end + + private + + def check_and_correct_keyword_args(node) + enum?(node) do |pairs| + pairs.each do |pair| + key, values = enum_values(pair) + next unless key + + correct_keyword_args(node, key, values, pairs[1..]) + end + end + end + + def correct_keyword_args(node, key, values, options) + add_offense(values, message: format(MSG_ARGS, enum: enum_name(key))) do |corrector| + corrector.replace(node, "enum #{source(key)}, #{values.source}#{correct_options(options)}") + end + end + + def correct_options(options) + corrected_options = options.map do |pair| + name = if pair.key.source[0] == '_' + pair.key.source[1..] + else + pair.key.source + end + + "#{name}: #{pair.value.source}" + end.join(', ') + ", #{corrected_options}" unless corrected_options.empty? + end + + def check_enum_options(node) + enum_with_options?(node) do |key, _, options| + options.children.each do |option| + name, = enum_options(option) + add_offense(name, message: format(MSG_OPTS, enum: enum_name(key))) if name.source[0] == '_' + end + end + end + + def enum_name(key) + case key.type + when :sym, :str + key.value + else + key.source + end + end + + def source(elem) + case elem.type + when :str + elem.value.dump + when :sym + elem.value.inspect + else + elem.source + end + end + end + end + end +end diff --git a/lib/rubocop/cop/rails_cops.rb b/lib/rubocop/cop/rails_cops.rb index e5173baeb0..b7eddc4fe4 100644 --- a/lib/rubocop/cop/rails_cops.rb +++ b/lib/rubocop/cop/rails_cops.rb @@ -46,6 +46,7 @@ require_relative 'rails/dynamic_find_by' require_relative 'rails/eager_evaluation_log_message' require_relative 'rails/enum_hash' +require_relative 'rails/enum_syntax' require_relative 'rails/enum_uniqueness' require_relative 'rails/env_local' require_relative 'rails/environment_comparison' diff --git a/spec/rubocop/cop/rails/enum_syntax_spec.rb b/spec/rubocop/cop/rails/enum_syntax_spec.rb new file mode 100644 index 0000000000..4886059840 --- /dev/null +++ b/spec/rubocop/cop/rails/enum_syntax_spec.rb @@ -0,0 +1,120 @@ +# frozen_string_literal: true + +RSpec.describe RuboCop::Cop::Rails::EnumSyntax, :config do + context 'Rails >= 7.0', :rails70 do + context 'when keyword arguments are used' do + context 'with %i[] syntax' do + it 'registers an offense' do + expect_offense(<<~RUBY) + enum status: { active: 0, archived: 1 }, _prefix: true + ^^^^^^^^^^^^^^^^^^^^^^^^^^ Enum defined with keyword arguments found in `status` enum declaration. Use positional arguments instead. + RUBY + end + end + + context 'with options prefixed with `_`' do + it 'registers an offense' do + expect_offense(<<~RUBY) + enum :status, { active: 0, archived: 1 }, _prefix: true, _suffix: true + ^^^^^^^ Enum defined with deprecated options found in `status` enum declaration. Use options without the `_` prefix. + ^^^^^^^ Enum defined with deprecated options found in `status` enum declaration. Use options without the `_` prefix. + RUBY + end + end + + context 'when the enum name is a string' do + it 'registers an offense' do + expect_offense(<<~RUBY) + enum "status" => { active: 0, archived: 1 } + ^^^^^^^^^^^^^^^^^^^^^^^^^^ Enum defined with keyword arguments found in `status` enum declaration. Use positional arguments instead. + RUBY + end + end + + context 'when the enum name is not a literal' do + it 'registers an offense' do + expect_offense(<<~RUBY) + enum KEY => { active: 0, archived: 1 } + ^^^^^^^^^^^^^^^^^^^^^^^^^^ Enum defined with keyword arguments found in `KEY` enum declaration. Use positional arguments instead. + RUBY + end + end + + it 'autocorrects' do + expect_offense(<<~RUBY) + enum status: { active: 0, archived: 1 } + ^^^^^^^^^^^^^^^^^^^^^^^^^^ Enum defined with keyword arguments found in `status` enum declaration. Use positional arguments instead. + RUBY + + expect_correction(<<~RUBY) + enum :status, { active: 0, archived: 1 } + RUBY + end + + it 'autocorrects options too' do + expect_offense(<<~RUBY) + enum status: { active: 0, archived: 1 }, _prefix: true, _suffix: true + ^^^^^^^^^^^^^^^^^^^^^^^^^^ Enum defined with keyword arguments found in `status` enum declaration. Use positional arguments instead. + RUBY + + expect_correction(<<~RUBY) + enum :status, { active: 0, archived: 1 }, prefix: true, suffix: true + RUBY + end + end + + context 'when positional arguments are used' do + it 'does not register an offense' do + expect_no_offenses('enum :status, { active: 0, archived: 1 }, prefix: true') + end + end + end + + context 'Rails <= 6.1', :rails61 do + context 'when keyword arguments are used' do + context 'with %i[] syntax' do + it 'does not register an offense' do + expect_no_offenses(<<~RUBY) + enum status: { active: 0, archived: 1 }, _prefix: true + RUBY + end + end + + context 'with options prefixed with `_`' do + it 'does not register an offense' do + expect_no_offenses(<<~RUBY) + enum :status, { active: 0, archived: 1 }, _prefix: true, _suffix: true + RUBY + end + end + + context 'when the enum name is a string' do + it 'does not register an offense' do + expect_no_offenses(<<~RUBY) + enum "status" => { active: 0, archived: 1 } + RUBY + end + end + + context 'when the enum name is not a literal' do + it 'does not register an offense' do + expect_no_offenses(<<~RUBY) + enum KEY => { active: 0, archived: 1 } + RUBY + end + end + + it 'autocorrects' do + expect_no_offenses(<<~RUBY) + enum status: { active: 0, archived: 1 } + RUBY + end + + it 'autocorrects options too' do + expect_no_offenses(<<~RUBY) + enum status: { active: 0, archived: 1 }, _prefix: true, _suffix: true + RUBY + end + end + end +end