2020-03-12 08:16:00 +08:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'rails_helper'
|
|
|
|
|
|
|
|
RSpec.describe BookmarkManager do
|
|
|
|
let(:user) { Fabricate(:user) }
|
|
|
|
|
|
|
|
let(:reminder_type) { 'tomorrow' }
|
2020-03-26 23:32:41 +08:00
|
|
|
let(:reminder_at) { 1.day.from_now }
|
2020-03-12 08:16:00 +08:00
|
|
|
fab!(:post) { Fabricate(:post) }
|
|
|
|
let(:name) { 'Check this out!' }
|
|
|
|
|
|
|
|
subject { described_class.new(user) }
|
|
|
|
|
|
|
|
describe ".create" do
|
|
|
|
it "creates the bookmark for the user" do
|
|
|
|
subject.create(post_id: post.id, name: name)
|
|
|
|
bookmark = Bookmark.find_by(user: user)
|
|
|
|
|
|
|
|
expect(bookmark.post_id).to eq(post.id)
|
|
|
|
expect(bookmark.topic_id).to eq(post.topic_id)
|
|
|
|
end
|
|
|
|
|
2020-09-07 12:52:14 +08:00
|
|
|
it "when topic is deleted it raises invalid access from guardian check" do
|
|
|
|
post.topic.trash!
|
|
|
|
expect { subject.create(post_id: post.id, name: name) }.to raise_error(Discourse::InvalidAccess)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "when post is deleted it raises invalid access from guardian check" do
|
|
|
|
post.trash!
|
|
|
|
expect { subject.create(post_id: post.id, name: name) }.to raise_error(Discourse::InvalidAccess)
|
|
|
|
end
|
|
|
|
|
2020-04-28 14:19:25 +08:00
|
|
|
it "updates the topic user bookmarked column to true if any post is bookmarked" do
|
|
|
|
subject.create(post_id: post.id, name: name, reminder_type: reminder_type, reminder_at: reminder_at)
|
|
|
|
tu = TopicUser.find_by(user: user)
|
|
|
|
expect(tu.bookmarked).to eq(true)
|
|
|
|
tu.update(bookmarked: false)
|
|
|
|
subject.create(post_id: Fabricate(:post, topic: post.topic).id)
|
|
|
|
tu.reload
|
|
|
|
expect(tu.bookmarked).to eq(true)
|
|
|
|
end
|
|
|
|
|
2020-03-12 08:16:00 +08:00
|
|
|
context "when a reminder time + type is provided" do
|
|
|
|
it "saves the values correctly" do
|
|
|
|
subject.create(post_id: post.id, name: name, reminder_type: reminder_type, reminder_at: reminder_at)
|
|
|
|
bookmark = Bookmark.find_by(user: user)
|
|
|
|
|
2020-03-26 23:32:41 +08:00
|
|
|
expect(bookmark.reminder_at).to eq_time(reminder_at)
|
2020-03-12 08:16:00 +08:00
|
|
|
expect(bookmark.reminder_set_at).not_to eq(nil)
|
|
|
|
expect(bookmark.reminder_type).to eq(Bookmark.reminder_types[:tomorrow])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-05-07 11:37:39 +08:00
|
|
|
context "when options are provided" do
|
2020-07-21 08:00:39 +08:00
|
|
|
let(:options) { { auto_delete_preference: Bookmark.auto_delete_preferences[:when_reminder_sent] } }
|
2020-05-07 11:37:39 +08:00
|
|
|
|
|
|
|
it "saves any additional options successfully" do
|
|
|
|
subject.create(post_id: post.id, name: name, options: options)
|
|
|
|
bookmark = Bookmark.find_by(user: user)
|
|
|
|
|
2020-07-21 08:00:39 +08:00
|
|
|
expect(bookmark.auto_delete_preference).to eq(1)
|
2020-05-08 23:24:37 +08:00
|
|
|
end
|
|
|
|
end
|
2020-05-07 11:37:39 +08:00
|
|
|
|
2020-03-12 08:16:00 +08:00
|
|
|
context "when the bookmark already exists for the user & post" do
|
|
|
|
before do
|
|
|
|
Bookmark.create(post: post, user: user, topic: post.topic)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "adds an error to the manager" do
|
|
|
|
subject.create(post_id: post.id)
|
|
|
|
expect(subject.errors.full_messages).to include(I18n.t("bookmarks.errors.already_bookmarked_post"))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-07-08 15:19:01 +08:00
|
|
|
context "when the bookmark name is too long" do
|
|
|
|
it "adds an error to the manager" do
|
|
|
|
subject.create(post_id: post.id, name: "test" * 100)
|
|
|
|
expect(subject.errors.full_messages).to include("Name is too long (maximum is 100 characters)")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-03-12 08:16:00 +08:00
|
|
|
context "when the reminder time is not provided when it needs to be" do
|
|
|
|
let(:reminder_at) { nil }
|
|
|
|
it "adds an error to the manager" do
|
|
|
|
subject.create(post_id: post.id, name: name, reminder_type: reminder_type, reminder_at: reminder_at)
|
|
|
|
expect(subject.errors.full_messages).to include(
|
2020-05-06 13:22:43 +08:00
|
|
|
"Reminder at " + I18n.t("bookmarks.errors.time_must_be_provided")
|
2020-03-12 08:16:00 +08:00
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "when the reminder time is in the past" do
|
2020-03-26 23:32:41 +08:00
|
|
|
let(:reminder_at) { 10.days.ago }
|
|
|
|
|
2020-03-12 08:16:00 +08:00
|
|
|
it "adds an error to the manager" do
|
|
|
|
subject.create(post_id: post.id, name: name, reminder_type: reminder_type, reminder_at: reminder_at)
|
|
|
|
expect(subject.errors.full_messages).to include(I18n.t("bookmarks.errors.cannot_set_past_reminder"))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "when the reminder time is far-flung (> 10 years from now)" do
|
2020-03-26 23:32:41 +08:00
|
|
|
let(:reminder_at) { 11.years.from_now }
|
|
|
|
|
2020-03-12 08:16:00 +08:00
|
|
|
it "adds an error to the manager" do
|
|
|
|
subject.create(post_id: post.id, name: name, reminder_type: reminder_type, reminder_at: reminder_at)
|
|
|
|
expect(subject.errors.full_messages).to include(I18n.t("bookmarks.errors.cannot_set_reminder_in_distant_future"))
|
|
|
|
end
|
|
|
|
end
|
2020-03-23 19:04:39 +08:00
|
|
|
|
2021-05-21 09:43:47 +08:00
|
|
|
context "when the post is inaccessible for the user" do
|
2020-03-23 19:04:39 +08:00
|
|
|
before do
|
|
|
|
post.trash!
|
|
|
|
end
|
|
|
|
it "raises an invalid access error" do
|
|
|
|
expect { subject.create(post_id: post.id, name: name) }.to raise_error(Discourse::InvalidAccess)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-05-21 09:43:47 +08:00
|
|
|
context "when the topic is inaccessible for the user" do
|
2020-03-23 19:04:39 +08:00
|
|
|
before do
|
|
|
|
post.topic.update(category: Fabricate(:private_category, group: Fabricate(:group)))
|
|
|
|
end
|
|
|
|
it "raises an invalid access error" do
|
|
|
|
expect { subject.create(post_id: post.id, name: name) }.to raise_error(Discourse::InvalidAccess)
|
|
|
|
end
|
|
|
|
end
|
2020-03-12 08:16:00 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
describe ".destroy" do
|
|
|
|
let!(:bookmark) { Fabricate(:bookmark, user: user, post: post) }
|
|
|
|
it "deletes the existing bookmark" do
|
2020-04-20 11:30:04 +08:00
|
|
|
result = subject.destroy(bookmark.id)
|
2020-03-12 08:16:00 +08:00
|
|
|
expect(Bookmark.exists?(id: bookmark.id)).to eq(false)
|
2020-04-20 11:30:04 +08:00
|
|
|
expect(result[:topic_bookmarked]).to eq(false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns a value indicating whether there are still other bookmarks in the topic for the user" do
|
|
|
|
Fabricate(:bookmark, user: user, post: Fabricate(:post, topic: post.topic))
|
|
|
|
result = subject.destroy(bookmark.id)
|
|
|
|
expect(result[:topic_bookmarked]).to eq(true)
|
2020-03-12 08:16:00 +08:00
|
|
|
end
|
|
|
|
|
2020-04-28 14:19:25 +08:00
|
|
|
context "if the bookmark is the last one bookmarked in the topic" do
|
|
|
|
it "marks the topic user bookmarked column as false" do
|
|
|
|
TopicUser.create(user: user, topic: bookmark.post.topic, bookmarked: true)
|
|
|
|
subject.destroy(bookmark.id)
|
|
|
|
tu = TopicUser.find_by(user: user)
|
|
|
|
expect(tu.bookmarked).to eq(false)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-03-12 08:16:00 +08:00
|
|
|
context "if the bookmark is belonging to some other user" do
|
|
|
|
let!(:bookmark) { Fabricate(:bookmark, user: Fabricate(:admin), post: post) }
|
|
|
|
it "raises an invalid access error" do
|
|
|
|
expect { subject.destroy(bookmark.id) }.to raise_error(Discourse::InvalidAccess)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "if the bookmark no longer exists" do
|
2021-03-22 07:50:22 +08:00
|
|
|
it "raises a not found error" do
|
2020-03-12 08:16:00 +08:00
|
|
|
expect { subject.destroy(9999) }.to raise_error(Discourse::NotFound)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-04-17 09:08:07 +08:00
|
|
|
describe ".update" do
|
|
|
|
let!(:bookmark) { Fabricate(:bookmark_next_business_day_reminder, user: user, post: post, name: "Old name") }
|
|
|
|
let(:new_name) { "Some new name" }
|
|
|
|
let(:new_reminder_at) { 10.days.from_now }
|
|
|
|
let(:new_reminder_type) { Bookmark.reminder_types[:custom] }
|
2020-05-07 11:37:39 +08:00
|
|
|
let(:options) { {} }
|
2020-04-17 09:08:07 +08:00
|
|
|
|
|
|
|
def update_bookmark
|
|
|
|
subject.update(
|
2020-05-07 11:37:39 +08:00
|
|
|
bookmark_id: bookmark.id,
|
|
|
|
name: new_name,
|
|
|
|
reminder_type: new_reminder_type,
|
|
|
|
reminder_at: new_reminder_at,
|
|
|
|
options: options
|
2020-04-17 09:08:07 +08:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2021-05-21 09:43:47 +08:00
|
|
|
it "saves the time and new reminder type and new name successfully" do
|
2020-04-17 09:08:07 +08:00
|
|
|
update_bookmark
|
2020-04-22 08:44:04 +08:00
|
|
|
bookmark.reload
|
|
|
|
expect(bookmark.name).to eq(new_name)
|
|
|
|
expect(bookmark.reminder_at).to eq_time(new_reminder_at)
|
|
|
|
expect(bookmark.reminder_type).to eq(new_reminder_type)
|
|
|
|
end
|
|
|
|
|
2020-05-07 11:37:39 +08:00
|
|
|
context "when options are provided" do
|
2020-07-21 08:00:39 +08:00
|
|
|
let(:options) { { auto_delete_preference: Bookmark.auto_delete_preferences[:when_reminder_sent] } }
|
2020-05-07 11:37:39 +08:00
|
|
|
|
|
|
|
it "saves any additional options successfully" do
|
|
|
|
update_bookmark
|
|
|
|
bookmark.reload
|
2020-07-21 08:00:39 +08:00
|
|
|
expect(bookmark.auto_delete_preference).to eq(1)
|
2020-05-07 11:37:39 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-04-22 08:44:04 +08:00
|
|
|
context "if the new reminder type is a string" do
|
|
|
|
let(:new_reminder_type) { "custom" }
|
|
|
|
it "is parsed" do
|
|
|
|
update_bookmark
|
|
|
|
bookmark.reload
|
|
|
|
expect(bookmark.reminder_type).to eq(Bookmark.reminder_types[:custom])
|
|
|
|
end
|
2020-04-17 09:08:07 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
context "if the bookmark is belonging to some other user" do
|
|
|
|
let!(:bookmark) { Fabricate(:bookmark, user: Fabricate(:admin), post: post) }
|
|
|
|
it "raises an invalid access error" do
|
|
|
|
expect { update_bookmark }.to raise_error(Discourse::InvalidAccess)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "if the bookmark no longer exists" do
|
|
|
|
before do
|
|
|
|
bookmark.destroy!
|
|
|
|
end
|
2021-03-22 07:50:22 +08:00
|
|
|
it "raises a not found error" do
|
2020-04-17 09:08:07 +08:00
|
|
|
expect { update_bookmark }.to raise_error(Discourse::NotFound)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-03-12 08:16:00 +08:00
|
|
|
describe ".destroy_for_topic" do
|
|
|
|
let!(:topic) { Fabricate(:topic) }
|
2020-03-12 08:52:15 +08:00
|
|
|
let!(:bookmark1) { Fabricate(:bookmark, topic: topic, post: Fabricate(:post, topic: topic), user: user) }
|
|
|
|
let!(:bookmark2) { Fabricate(:bookmark, topic: topic, post: Fabricate(:post, topic: topic), user: user) }
|
2020-03-12 08:16:00 +08:00
|
|
|
|
|
|
|
it "destroys all bookmarks for the topic for the specified user" do
|
|
|
|
subject.destroy_for_topic(topic)
|
|
|
|
expect(Bookmark.where(user: user, topic: topic).length).to eq(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "does not destroy any other user's topic bookmarks" do
|
|
|
|
user2 = Fabricate(:user)
|
|
|
|
Fabricate(:bookmark, topic: topic, post: Fabricate(:post, topic: topic), user: user2)
|
|
|
|
subject.destroy_for_topic(topic)
|
|
|
|
expect(Bookmark.where(user: user2, topic: topic).length).to eq(1)
|
|
|
|
end
|
|
|
|
|
2020-04-16 09:32:21 +08:00
|
|
|
it "updates the topic user bookmarked column to false" do
|
|
|
|
TopicUser.create(user: user, topic: topic, bookmarked: true)
|
|
|
|
subject.destroy_for_topic(topic)
|
|
|
|
tu = TopicUser.find_by(user: user)
|
|
|
|
expect(tu.bookmarked).to eq(false)
|
|
|
|
end
|
2020-03-12 08:16:00 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
describe ".send_reminder_notification" do
|
|
|
|
let(:bookmark) { Fabricate(:bookmark, user: user) }
|
|
|
|
it "clears the reminder_at and sets the reminder_last_sent_at" do
|
|
|
|
expect(bookmark.reminder_last_sent_at).to eq(nil)
|
|
|
|
described_class.send_reminder_notification(bookmark.id)
|
|
|
|
bookmark.reload
|
|
|
|
expect(bookmark.reminder_at).to eq(nil)
|
|
|
|
expect(bookmark.reminder_last_sent_at).not_to eq(nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "creates a notification for the reminder" do
|
|
|
|
described_class.send_reminder_notification(bookmark.id)
|
|
|
|
notif = notifications_for_user.last
|
|
|
|
expect(notif.post_number).to eq(bookmark.post.post_number)
|
|
|
|
end
|
|
|
|
|
|
|
|
context "when the bookmark does no longer exist" do
|
|
|
|
before do
|
|
|
|
bookmark.destroy
|
|
|
|
end
|
|
|
|
it "does not error, and does not create a notification" do
|
|
|
|
described_class.send_reminder_notification(bookmark.id)
|
|
|
|
expect(notifications_for_user.any?).to eq(false)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "if the post has been deleted" do
|
|
|
|
before do
|
|
|
|
bookmark.post.trash!
|
|
|
|
end
|
|
|
|
it "does not error, and does not create a notification, and clears the reminder" do
|
|
|
|
described_class.send_reminder_notification(bookmark.id)
|
|
|
|
bookmark.reload
|
|
|
|
expect(bookmark.reminder_at).to eq(nil)
|
|
|
|
expect(notifications_for_user.any?).to eq(false)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def notifications_for_user
|
|
|
|
Notification.where(notification_type: Notification.types[:bookmark_reminder], user_id: bookmark.user.id)
|
|
|
|
end
|
|
|
|
end
|
2021-03-22 07:50:22 +08:00
|
|
|
|
|
|
|
describe ".toggle_pin" do
|
|
|
|
let!(:bookmark) { Fabricate(:bookmark, user: user) }
|
|
|
|
|
|
|
|
it "sets pinned to false if it is true" do
|
|
|
|
bookmark.update(pinned: true)
|
|
|
|
subject.toggle_pin(bookmark_id: bookmark.id)
|
|
|
|
expect(bookmark.reload.pinned).to eq(false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "sets pinned to true if it is false" do
|
|
|
|
bookmark.update(pinned: false)
|
|
|
|
subject.toggle_pin(bookmark_id: bookmark.id)
|
|
|
|
expect(bookmark.reload.pinned).to eq(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
context "if the bookmark is belonging to some other user" do
|
|
|
|
let!(:bookmark) { Fabricate(:bookmark, user: Fabricate(:admin)) }
|
|
|
|
it "raises an invalid access error" do
|
|
|
|
expect { subject.toggle_pin(bookmark_id: bookmark.id) }.to raise_error(Discourse::InvalidAccess)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "if the bookmark no longer exists" do
|
|
|
|
before do
|
|
|
|
bookmark.destroy!
|
|
|
|
end
|
|
|
|
it "raises a not found error" do
|
|
|
|
expect { subject.toggle_pin(bookmark_id: bookmark.id) }.to raise_error(Discourse::NotFound)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2020-03-12 08:16:00 +08:00
|
|
|
end
|