From f229951133d774af8992c5a462c70a214ac6ef37 Mon Sep 17 00:00:00 2001 From: Koichi ITO Date: Sat, 23 Apr 2022 15:12:57 +0900 Subject: [PATCH] Fix a false negative for `Rails/TransactionExitStatement` Follow up https://github.com/rubocop/rubocop-rails/pull/674#issuecomment-1107330877 Revert "[Fix #669] Fix a false positive for `Rails/TransactionExitStatement`" This reverts commit d9ec02de75d95337e7dc2a8f876ba8a4874276c1 and tweak a spec. So this PR fixes a false negative for `Rails/TransactionExitStatement` when `return` is used in `rescue`. --- ...tive_for_rails_transaction_exit_statement.md | 1 + .../cop/rails/transaction_exit_statement.rb | 8 +------- .../rails/transaction_exit_statement_spec.rb | 17 +++++++++-------- 3 files changed, 11 insertions(+), 15 deletions(-) create mode 100644 changelog/fix_a_false_negative_for_rails_transaction_exit_statement.md diff --git a/changelog/fix_a_false_negative_for_rails_transaction_exit_statement.md b/changelog/fix_a_false_negative_for_rails_transaction_exit_statement.md new file mode 100644 index 0000000000..b9e1bec01b --- /dev/null +++ b/changelog/fix_a_false_negative_for_rails_transaction_exit_statement.md @@ -0,0 +1 @@ +* [#696](https://github.com/rubocop/rubocop-rails/pull/696): Fix a false negative for `Rails/TransactionExitStatement` when `return` is used in `rescue`. ([@koic][]) diff --git a/lib/rubocop/cop/rails/transaction_exit_statement.rb b/lib/rubocop/cop/rails/transaction_exit_statement.rb index 373d9403d9..d2ec6c6ed0 100644 --- a/lib/rubocop/cop/rails/transaction_exit_statement.rb +++ b/lib/rubocop/cop/rails/transaction_exit_statement.rb @@ -58,7 +58,7 @@ def on_send(node) return unless parent.block_type? && parent.body exit_statements(parent.body).each do |statement_node| - next if in_rescue?(statement_node) || nested_block?(statement_node) + next if statement_node.break_type? && nested_block?(statement_node) statement = statement(statement_node) message = format(MSG, statement: statement) @@ -79,13 +79,7 @@ def statement(statement_node) end end - def in_rescue?(statement_node) - statement_node.ancestors.find(&:rescue_type?) - end - def nested_block?(statement_node) - return false unless statement_node.break_type? - !statement_node.ancestors.find(&:block_type?).method?(:transaction) end end diff --git a/spec/rubocop/cop/rails/transaction_exit_statement_spec.rb b/spec/rubocop/cop/rails/transaction_exit_statement_spec.rb index 953e19a2d2..2a0c0f3aa5 100644 --- a/spec/rubocop/cop/rails/transaction_exit_statement_spec.rb +++ b/spec/rubocop/cop/rails/transaction_exit_statement_spec.rb @@ -66,21 +66,22 @@ RUBY end - it 'does not register an offense when `break` is used in `loop` in transactions' do - expect_no_offenses(<<~RUBY) + it 'registers an offense when `return` is used in `rescue`' do + expect_offense(<<~RUBY) ApplicationRecord.transaction do - loop do - break if condition - end + rescue + return do_something + ^^^^^^^^^^^^^^^^^^^ Exit statement `return` is not allowed. Use `raise` (rollback) or `next` (commit). end RUBY end - it 'does not register an offense when `return` is used in `rescue`' do + it 'does not register an offense when `break` is used in `loop` in transactions' do expect_no_offenses(<<~RUBY) ApplicationRecord.transaction do - rescue - return do_something + loop do + break if condition + end end RUBY end