Skip to content

Commit 215a713

Browse files
committed
Add Rails/FindByOrAssignmentMemoization cop
1 parent b8c4126 commit 215a713

5 files changed

+147
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
* [#1324](https://github.com/rubocop/rubocop-rails/pull/1324): Add `Rails/FindByOrAssignmentMemoization` cop. ([@r7kamura][])

config/default.yml

+6
Original file line numberDiff line numberDiff line change
@@ -502,6 +502,12 @@ Rails/FindById:
502502
Enabled: 'pending'
503503
VersionAdded: '2.7'
504504

505+
Rails/FindByOrAssignmentMemoization:
506+
Description: 'Avoid memoization by `find_by` with `||=`.'
507+
Enabled: pending
508+
Safe: false
509+
VersionAdded: '<<next>>'
510+
505511
Rails/FindEach:
506512
Description: 'Prefer all.find_each over all.each.'
507513
StyleGuide: 'https://rails.rubystyle.guide#find-each'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
# frozen_string_literal: true
2+
3+
module RuboCop
4+
module Cop
5+
module Rails
6+
# Avoid memoization by `find_by` with `||=`.
7+
#
8+
# It is common to see code that attempts to memoize `find_by` result by `||=`,
9+
# but `find_by` may return `nil`, in which case it is not memoized as intended.
10+
#
11+
# @safety
12+
# This cop is unsafe because detected `find_by` may not be activerecord's method,
13+
# or the code may have a different purpose than memoization.
14+
#
15+
# @example
16+
# # bad
17+
# def current_user
18+
# @current_user ||= User.find_by(id: session[:user_id])
19+
# end
20+
#
21+
# # good
22+
# def current_user
23+
# if instance_variable_defined?(:@current_user)
24+
# @current_user
25+
# else
26+
# @current_user = User.find_by(id: session[:user_id])
27+
# end
28+
# end
29+
class FindByOrAssignmentMemoization < Base
30+
extend AutoCorrector
31+
32+
MSG = 'Avoid memoization by `find_by` with `||=`.'
33+
34+
RESTRICT_ON_SEND = %i[find_by].freeze
35+
36+
def_node_matcher :find_by_or_assignment_memoization, <<~PATTERN
37+
(or_asgn
38+
(ivasgn $_)
39+
$(send _ :find_by ...)
40+
)
41+
PATTERN
42+
43+
def on_send(node)
44+
assignment_node = node.parent
45+
find_by_or_assignment_memoization(assignment_node) do |varible_name, find_by|
46+
next if in_condition?(assignment_node)
47+
48+
add_offense(assignment_node) do |corrector|
49+
corrector.replace(
50+
assignment_node,
51+
<<~RUBY.rstrip
52+
if instance_variable_defined?(:#{varible_name})
53+
#{varible_name}
54+
else
55+
#{varible_name} = #{find_by.source}
56+
end
57+
RUBY
58+
)
59+
end
60+
end
61+
end
62+
63+
private
64+
65+
def in_condition?(node)
66+
node.each_ancestor(:if, :unless).any?
67+
end
68+
end
69+
end
70+
end
71+
end

lib/rubocop/cop/rails_cops.rb

+1
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,7 @@
5555
require_relative 'rails/file_path'
5656
require_relative 'rails/find_by'
5757
require_relative 'rails/find_by_id'
58+
require_relative 'rails/find_by_or_assignment_memoization'
5859
require_relative 'rails/find_each'
5960
require_relative 'rails/freeze_time'
6061
require_relative 'rails/has_and_belongs_to_many'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
# frozen_string_literal: true
2+
3+
RSpec.describe RuboCop::Cop::Rails::FindByOrAssignmentMemoization, :config do
4+
context 'when using `find_by` with `||=`' do
5+
it 'registers an offense' do
6+
expect_offense(<<~RUBY)
7+
@current_user ||= User.find_by(id: session[:user_id])
8+
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Avoid memoization by `find_by` with `||=`.
9+
RUBY
10+
11+
expect_correction(<<~RUBY)
12+
if instance_variable_defined?(:@current_user)
13+
@current_user
14+
else
15+
@current_user = User.find_by(id: session[:user_id])
16+
end
17+
RUBY
18+
end
19+
end
20+
21+
context 'with `find_by!`' do
22+
it 'does not register an offense' do
23+
expect_no_offenses(<<~RUBY)
24+
@current_user ||= User.find_by!(id: session[:user_id])
25+
RUBY
26+
end
27+
end
28+
29+
context 'with local variable' do
30+
it 'does not register an offense' do
31+
expect_no_offenses(<<~RUBY)
32+
current_user ||= User.find_by(id: session[:user_id])
33+
RUBY
34+
end
35+
end
36+
37+
context 'with `||`' do
38+
it 'does not register an offense' do
39+
expect_no_offenses(<<~RUBY)
40+
@current_user ||= User.find_by(id: session[:user_id]) || User.anonymous
41+
RUBY
42+
end
43+
end
44+
45+
context 'with ternary operator' do
46+
it 'does not register an offense' do
47+
expect_no_offenses(<<~RUBY)
48+
@current_user ||= session[:user_id] ? User.find_by(id: session[:user_id]) : nil
49+
RUBY
50+
end
51+
end
52+
53+
context 'with `if`' do
54+
it 'does not register an offense' do
55+
expect_no_offenses(<<~RUBY)
56+
@current_user ||= User.find_by(id: session[:user_id]) if session[:user_id]
57+
RUBY
58+
end
59+
end
60+
61+
context 'with `unless`' do
62+
it 'does not register an offense' do
63+
expect_no_offenses(<<~RUBY)
64+
@current_user ||= User.find_by(id: session[:user_id]) unless session[:user_id].nil?
65+
RUBY
66+
end
67+
end
68+
end

0 commit comments

Comments
 (0)