PERF: Avoid running query unnecessarily when updating bookmark. (#14276)

* Avoid loading an entire ActiveRecord object when saving and updating.
* Avoid running a DB query when `post_id` or `user_id` is not changed.
This commit is contained in:
Alan Guo Xiang Tan 2021-09-09 08:50:26 +08:00 committed by GitHub
parent 13b31def80
commit 1e05175364
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 24 additions and 8 deletions

View File

@ -31,7 +31,10 @@ class Bookmark < ActiveRecord::Base
if: -> { reminder_type.present? } if: -> { reminder_type.present? }
} }
validate :unique_per_post_for_user validate :unique_per_post_for_user,
on: [:create, :update],
if: Proc.new { |b| b.will_save_change_to_post_id? || b.will_save_change_to_user_id? }
validate :ensure_sane_reminder_at_time validate :ensure_sane_reminder_at_time
validate :bookmark_limit_not_reached validate :bookmark_limit_not_reached
validates :name, length: { maximum: 100 } validates :name, length: { maximum: 100 }
@ -47,10 +50,10 @@ class Bookmark < ActiveRecord::Base
end end
def unique_per_post_for_user def unique_per_post_for_user
existing_bookmark = Bookmark.find_by(user_id: user_id, post_id: post_id) if Bookmark.exists?(user_id: user_id, post_id: post_id)
return if existing_bookmark.blank? || existing_bookmark.id == id
self.errors.add(:base, I18n.t("bookmarks.errors.already_bookmarked_post")) self.errors.add(:base, I18n.t("bookmarks.errors.already_bookmarked_post"))
end end
end
def ensure_sane_reminder_at_time def ensure_sane_reminder_at_time
return if reminder_at.blank? return if reminder_at.blank?

View File

@ -3,9 +3,25 @@
require 'rails_helper' require 'rails_helper'
describe Bookmark do describe Bookmark do
fab!(:post) { Fabricate(:post) }
context 'validations' do
it 'does not allow user to bookmark a post twice' do
bookmark = Fabricate(:bookmark, post: post, topic: post.topic)
user = bookmark.user
bookmark_2 = Fabricate.build(:bookmark,
post: post,
topic: post.topic,
user: user
)
expect(bookmark_2.valid?).to eq(false)
end
end
describe "#cleanup!" do describe "#cleanup!" do
it "deletes bookmarks attached to a deleted post which has been deleted for > 3 days" do it "deletes bookmarks attached to a deleted post which has been deleted for > 3 days" do
post = Fabricate(:post)
bookmark = Fabricate(:bookmark, post: post, topic: post.topic) bookmark = Fabricate(:bookmark, post: post, topic: post.topic)
bookmark2 = Fabricate(:bookmark, post: Fabricate(:post, topic: post.topic)) bookmark2 = Fabricate(:bookmark, post: Fabricate(:post, topic: post.topic))
post.trash! post.trash!
@ -16,7 +32,6 @@ describe Bookmark do
end end
it "runs a SyncTopicUserBookmarked job for all deleted bookmark unique topics to make sure topic_user.bookmarked is in sync" do it "runs a SyncTopicUserBookmarked job for all deleted bookmark unique topics to make sure topic_user.bookmarked is in sync" do
post = Fabricate(:post)
post2 = Fabricate(:post) post2 = Fabricate(:post)
bookmark = Fabricate(:bookmark, post: post, topic: post.topic) bookmark = Fabricate(:bookmark, post: post, topic: post.topic)
bookmark2 = Fabricate(:bookmark, post: Fabricate(:post, topic: post.topic)) bookmark2 = Fabricate(:bookmark, post: Fabricate(:post, topic: post.topic))
@ -36,7 +51,6 @@ describe Bookmark do
end end
it "deletes bookmarks attached to a deleted topic which has been deleted for > 3 days" do it "deletes bookmarks attached to a deleted topic which has been deleted for > 3 days" do
post = Fabricate(:post)
bookmark = Fabricate(:bookmark, post: post, topic: post.topic) bookmark = Fabricate(:bookmark, post: post, topic: post.topic)
bookmark2 = Fabricate(:bookmark, topic: post.topic, post: Fabricate(:post, topic: post.topic)) bookmark2 = Fabricate(:bookmark, topic: post.topic, post: Fabricate(:post, topic: post.topic))
bookmark3 = Fabricate(:bookmark) bookmark3 = Fabricate(:bookmark)
@ -49,7 +63,6 @@ describe Bookmark do
end end
it "does not delete bookmarks attached to posts that are not deleted or that have not met the 3 day grace period" do it "does not delete bookmarks attached to posts that are not deleted or that have not met the 3 day grace period" do
post = Fabricate(:post)
bookmark = Fabricate(:bookmark, post: post, topic: post.topic) bookmark = Fabricate(:bookmark, post: post, topic: post.topic)
bookmark2 = Fabricate(:bookmark) bookmark2 = Fabricate(:bookmark)
Bookmark.cleanup! Bookmark.cleanup!