2019-04-29 20:27:42 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2015-10-11 05:41:23 -04:00
|
|
|
require 'rails_helper'
|
2013-07-03 21:47:12 -04:00
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
RSpec.describe TopicTimer, type: :model do
|
2020-04-21 00:38:35 -04:00
|
|
|
fab!(:topic_timer) { Fabricate(:topic_timer) }
|
2019-05-06 23:12:20 -04:00
|
|
|
fab!(:topic) { Fabricate(:topic) }
|
|
|
|
fab!(:admin) { Fabricate(:admin) }
|
2015-12-08 07:43:23 -05:00
|
|
|
|
2020-04-21 00:38:35 -04:00
|
|
|
before { freeze_time }
|
2018-06-05 03:29:17 -04:00
|
|
|
|
2017-03-21 23:12:02 -04:00
|
|
|
context "validations" do
|
|
|
|
describe '#status_type' do
|
2017-05-16 14:49:42 -04:00
|
|
|
it 'should ensure that only one active public topic status update exists' do
|
2017-05-11 18:23:18 -04:00
|
|
|
topic_timer.update!(topic: topic)
|
|
|
|
Fabricate(:topic_timer, deleted_at: Time.zone.now, topic: topic)
|
2014-10-29 16:26:32 -04:00
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
expect { Fabricate(:topic_timer, topic: topic) }
|
2017-03-21 23:12:02 -04:00
|
|
|
.to raise_error(ActiveRecord::RecordInvalid)
|
|
|
|
end
|
2017-05-16 14:49:42 -04:00
|
|
|
|
|
|
|
it 'should ensure that only one active private topic timer exists per user' do
|
|
|
|
Fabricate(:topic_timer, topic: topic, user: admin, status_type: TopicTimer.types[:reminder])
|
|
|
|
|
|
|
|
expect { Fabricate(:topic_timer, topic: topic, user: admin, status_type: TopicTimer.types[:reminder]) }
|
|
|
|
.to raise_error(ActiveRecord::RecordInvalid)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should allow users to have their own private topic timer' do
|
2017-05-16 21:37:11 -04:00
|
|
|
expect do
|
|
|
|
Fabricate(:topic_timer,
|
|
|
|
topic: topic,
|
|
|
|
user: Fabricate(:admin),
|
|
|
|
status_type: TopicTimer.types[:reminder]
|
|
|
|
)
|
|
|
|
end.to_not raise_error
|
2017-05-16 14:49:42 -04:00
|
|
|
end
|
2017-03-21 23:12:02 -04:00
|
|
|
end
|
2014-10-29 16:26:32 -04:00
|
|
|
|
2017-03-21 23:12:02 -04:00
|
|
|
describe '#execute_at' do
|
|
|
|
describe 'when #execute_at is greater than #created_at' do
|
|
|
|
it 'should be valid' do
|
2017-05-11 18:23:18 -04:00
|
|
|
topic_timer = Fabricate.build(:topic_timer,
|
2017-03-21 23:12:02 -04:00
|
|
|
execute_at: Time.zone.now + 1.hour,
|
|
|
|
user: Fabricate(:user),
|
|
|
|
topic: Fabricate(:topic)
|
|
|
|
)
|
2013-07-03 21:47:12 -04:00
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
expect(topic_timer).to be_valid
|
2017-03-21 23:12:02 -04:00
|
|
|
end
|
|
|
|
end
|
2013-07-03 21:47:12 -04:00
|
|
|
|
2017-03-21 23:12:02 -04:00
|
|
|
describe 'when #execute_at is smaller than #created_at' do
|
|
|
|
it 'should not be valid' do
|
2017-05-11 18:23:18 -04:00
|
|
|
topic_timer = Fabricate.build(:topic_timer,
|
2017-03-21 23:12:02 -04:00
|
|
|
execute_at: Time.zone.now - 1.hour,
|
|
|
|
created_at: Time.zone.now,
|
|
|
|
user: Fabricate(:user),
|
|
|
|
topic: Fabricate(:topic)
|
|
|
|
)
|
2013-07-03 21:47:12 -04:00
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
expect(topic_timer).to_not be_valid
|
2017-03-21 23:12:02 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2017-04-07 03:26:15 -04:00
|
|
|
|
|
|
|
describe '#category_id' do
|
|
|
|
describe 'when #status_type is publish_to_category' do
|
|
|
|
describe 'when #category_id is not present' do
|
|
|
|
it 'should not be valid' do
|
2017-05-11 18:23:18 -04:00
|
|
|
topic_timer = Fabricate.build(:topic_timer,
|
|
|
|
status_type: TopicTimer.types[:publish_to_category]
|
2017-04-07 03:26:15 -04:00
|
|
|
)
|
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
expect(topic_timer).to_not be_valid
|
2019-04-30 02:58:18 -04:00
|
|
|
expect(topic_timer.errors).to include(:category_id)
|
2017-04-07 03:26:15 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'when #category_id is present' do
|
|
|
|
it 'should be valid' do
|
2017-05-11 18:23:18 -04:00
|
|
|
topic_timer = Fabricate.build(:topic_timer,
|
|
|
|
status_type: TopicTimer.types[:publish_to_category],
|
2017-04-07 03:26:15 -04:00
|
|
|
category_id: Fabricate(:category).id,
|
|
|
|
user: Fabricate(:user),
|
|
|
|
topic: Fabricate(:topic)
|
|
|
|
)
|
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
expect(topic_timer).to be_valid
|
2017-04-07 03:26:15 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2013-07-03 21:47:12 -04:00
|
|
|
end
|
2014-10-29 16:26:32 -04:00
|
|
|
|
2017-03-21 23:12:02 -04:00
|
|
|
context 'callbacks' do
|
|
|
|
describe 'when #execute_at and #user_id are not changed' do
|
|
|
|
it 'should not schedule another to update topic' do
|
2020-04-21 00:38:35 -04:00
|
|
|
Jobs.expects(:enqueue_at).never
|
2017-03-21 23:12:02 -04:00
|
|
|
Jobs.expects(:cancel_scheduled_job).never
|
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
topic_timer.update!(topic: Fabricate(:topic))
|
2017-03-21 23:12:02 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'when #execute_at value is changed' do
|
|
|
|
it 'reschedules the job' do
|
2017-07-24 09:17:42 -04:00
|
|
|
Jobs.expects(:cancel_scheduled_job).with(
|
|
|
|
:toggle_topic_closed, topic_timer_id: topic_timer.id
|
|
|
|
)
|
2017-03-21 23:12:02 -04:00
|
|
|
|
2020-07-24 05:16:52 -04:00
|
|
|
expect_enqueued_with(job: :toggle_topic_closed, args: { topic_timer_id: topic_timer.id, state: true }, at: 3.days.from_now) do
|
|
|
|
topic_timer.update!(execute_at: 3.days.from_now, created_at: Time.zone.now)
|
|
|
|
end
|
2017-03-21 23:12:02 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe 'when execute_at is smaller than the current time' do
|
|
|
|
it 'should enqueue the job immediately' do
|
2020-07-24 05:16:52 -04:00
|
|
|
expect_enqueued_with(job: :toggle_topic_closed, args: { topic_timer_id: topic_timer.id, state: true }, at: Time.zone.now) do
|
|
|
|
topic_timer.update!(
|
|
|
|
execute_at: Time.zone.now - 1.hour,
|
|
|
|
created_at: Time.zone.now - 2.hour
|
|
|
|
)
|
|
|
|
end
|
2017-03-21 23:12:02 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'when user is changed' do
|
|
|
|
it 'should update the job' do
|
2017-07-24 09:17:42 -04:00
|
|
|
Jobs.expects(:cancel_scheduled_job).with(
|
|
|
|
:toggle_topic_closed, topic_timer_id: topic_timer.id
|
|
|
|
)
|
2017-03-21 23:12:02 -04:00
|
|
|
|
2020-07-24 05:16:52 -04:00
|
|
|
expect_enqueued_with(job: :toggle_topic_closed, args: { topic_timer_id: topic_timer.id, state: true }, at: topic_timer.execute_at) do
|
|
|
|
topic_timer.update!(user: admin)
|
|
|
|
end
|
2017-03-21 23:12:02 -04:00
|
|
|
end
|
|
|
|
end
|
2017-04-02 21:06:15 -04:00
|
|
|
|
2020-08-25 21:17:12 -04:00
|
|
|
describe 'when a topic has been deleted' do
|
2019-05-06 23:12:20 -04:00
|
|
|
fab!(:topic) { Fabricate(:topic, closed: false) }
|
2020-04-21 00:38:35 -04:00
|
|
|
fab!(:topic_timer) do
|
2017-05-11 18:23:18 -04:00
|
|
|
Fabricate(:topic_timer,
|
2017-04-02 21:06:15 -04:00
|
|
|
status_type: described_class.types[:open],
|
|
|
|
topic: topic
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2018-05-31 03:53:49 -04:00
|
|
|
before do
|
2019-03-14 10:47:38 -04:00
|
|
|
Jobs.run_immediately!
|
2018-05-31 03:53:49 -04:00
|
|
|
end
|
|
|
|
|
2020-08-25 21:17:12 -04:00
|
|
|
it 'should not queue the job' do
|
|
|
|
topic.trash!
|
2017-05-11 18:23:18 -04:00
|
|
|
topic_timer
|
2017-04-02 21:06:15 -04:00
|
|
|
|
2020-08-25 21:17:12 -04:00
|
|
|
expect(Jobs::ToggleTopicClosed.jobs).to eq([])
|
2017-04-02 21:06:15 -04:00
|
|
|
end
|
|
|
|
end
|
2017-05-16 14:49:42 -04:00
|
|
|
|
2017-05-16 21:37:11 -04:00
|
|
|
describe '#public_type' do
|
2017-05-16 14:49:42 -04:00
|
|
|
[:close, :open, :delete].each do |public_type|
|
|
|
|
it "is true for #{public_type}" do
|
|
|
|
timer = Fabricate(:topic_timer, status_type: described_class.types[public_type])
|
|
|
|
expect(timer.public_type).to eq(true)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it "is true for publish_to_category" do
|
|
|
|
timer = Fabricate(:topic_timer, status_type: described_class.types[:publish_to_category], category: Fabricate(:category))
|
|
|
|
expect(timer.public_type).to eq(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
described_class.private_types.keys.each do |private_type|
|
|
|
|
it "is false for #{private_type}" do
|
|
|
|
timer = Fabricate(:topic_timer, status_type: described_class.types[private_type])
|
|
|
|
expect(timer.public_type).to be_falsey
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2014-10-29 16:26:32 -04:00
|
|
|
end
|
|
|
|
|
2017-03-21 23:12:02 -04:00
|
|
|
describe '.ensure_consistency!' do
|
|
|
|
it 'should enqueue jobs that have been missed' do
|
2017-05-11 18:23:18 -04:00
|
|
|
close_topic_timer = Fabricate(:topic_timer,
|
2017-03-21 23:12:02 -04:00
|
|
|
execute_at: Time.zone.now - 1.hour,
|
|
|
|
created_at: Time.zone.now - 2.hour
|
|
|
|
)
|
2014-10-29 16:26:32 -04:00
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
open_topic_timer = Fabricate(:topic_timer,
|
2017-03-21 23:12:02 -04:00
|
|
|
status_type: described_class.types[:open],
|
|
|
|
execute_at: Time.zone.now - 1.hour,
|
2017-06-21 02:31:15 -04:00
|
|
|
created_at: Time.zone.now - 2.hour,
|
|
|
|
topic: Fabricate(:topic, closed: true)
|
2017-03-21 23:12:02 -04:00
|
|
|
)
|
|
|
|
|
2018-08-14 06:35:20 -04:00
|
|
|
Fabricate(:topic_timer, execute_at: Time.zone.now + 1.hour)
|
2017-03-21 23:12:02 -04:00
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
Fabricate(:topic_timer,
|
2017-04-13 00:02:35 -04:00
|
|
|
execute_at: Time.zone.now - 1.hour,
|
|
|
|
created_at: Time.zone.now - 2.hour
|
|
|
|
).topic.trash!
|
|
|
|
|
2018-08-14 06:35:20 -04:00
|
|
|
# creating topic timers already enqueues jobs
|
|
|
|
# let's delete them to test ensure_consistency!
|
|
|
|
Sidekiq::Worker.clear_all
|
|
|
|
|
2017-03-21 23:12:02 -04:00
|
|
|
expect { described_class.ensure_consistency! }
|
|
|
|
.to change { Jobs::ToggleTopicClosed.jobs.count }.by(2)
|
|
|
|
|
|
|
|
job_args = Jobs::ToggleTopicClosed.jobs.first["args"].first
|
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
expect(job_args["topic_timer_id"]).to eq(close_topic_timer.id)
|
2017-03-21 23:12:02 -04:00
|
|
|
expect(job_args["state"]).to eq(true)
|
|
|
|
|
|
|
|
job_args = Jobs::ToggleTopicClosed.jobs.last["args"].first
|
|
|
|
|
2017-05-11 18:23:18 -04:00
|
|
|
expect(job_args["topic_timer_id"]).to eq(open_topic_timer.id)
|
2017-03-21 23:12:02 -04:00
|
|
|
expect(job_args["state"]).to eq(false)
|
|
|
|
end
|
2017-05-16 14:49:42 -04:00
|
|
|
|
2018-08-14 06:35:20 -04:00
|
|
|
it "should enqueue remind me jobs that have been missed" do
|
|
|
|
reminder = Fabricate(:topic_timer,
|
|
|
|
status_type: described_class.types[:reminder],
|
|
|
|
execute_at: Time.zone.now - 1.hour,
|
|
|
|
created_at: Time.zone.now - 2.hour
|
|
|
|
)
|
2017-05-17 12:46:41 -04:00
|
|
|
|
2018-08-14 06:35:20 -04:00
|
|
|
# creating topic timers already enqueues jobs
|
|
|
|
# let's delete them to test ensure_consistency!
|
|
|
|
Sidekiq::Worker.clear_all
|
2017-05-17 12:46:41 -04:00
|
|
|
|
2018-08-14 06:35:20 -04:00
|
|
|
expect { described_class.ensure_consistency! }
|
|
|
|
.to change { Jobs::TopicReminder.jobs.count }.by(1)
|
|
|
|
|
|
|
|
job_args = Jobs::TopicReminder.jobs.first["args"].first
|
|
|
|
expect(job_args["topic_timer_id"]).to eq(reminder.id)
|
|
|
|
end
|
2014-10-29 16:26:32 -04:00
|
|
|
end
|
2013-07-03 21:47:12 -04:00
|
|
|
end
|