Requeue expiration notification (#24311)
parent
c384795731
commit
eb38e9df31
|
@ -141,9 +141,9 @@ class UpdateStatusService < BaseService
|
||||||
poll = @status.preloadable_poll
|
poll = @status.preloadable_poll
|
||||||
|
|
||||||
# If the poll had no expiration date set but now has, or now has a sooner
|
# If the poll had no expiration date set but now has, or now has a sooner
|
||||||
# expiration date, and people have voted, schedule a notification
|
# expiration date, schedule a notification
|
||||||
|
|
||||||
return unless poll.present? && poll.expires_at.present? && poll.votes.exists?
|
return unless poll.present? && poll.expires_at.present?
|
||||||
|
|
||||||
PollExpirationNotifyWorker.remove_from_scheduled(poll.id) if @previous_expires_at.present? && @previous_expires_at > poll.expires_at
|
PollExpirationNotifyWorker.remove_from_scheduled(poll.id) if @previous_expires_at.present? && @previous_expires_at > poll.expires_at
|
||||||
PollExpirationNotifyWorker.perform_at(poll.expires_at + 5.minutes, poll.id)
|
PollExpirationNotifyWorker.perform_at(poll.expires_at + 5.minutes, poll.id)
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
class PollExpirationNotifyWorker
|
class PollExpirationNotifyWorker
|
||||||
include Sidekiq::Worker
|
include Sidekiq::Worker
|
||||||
|
|
||||||
sidekiq_options lock: :until_executed
|
sidekiq_options lock: :until_executing
|
||||||
|
|
||||||
def perform(poll_id)
|
def perform(poll_id)
|
||||||
@poll = Poll.find(poll_id)
|
@poll = Poll.find(poll_id)
|
||||||
|
|
|
@ -120,8 +120,10 @@ RSpec.describe UpdateStatusService, type: :service do
|
||||||
before do
|
before do
|
||||||
status.update(poll: poll)
|
status.update(poll: poll)
|
||||||
VoteService.new.call(voter, poll, [0])
|
VoteService.new.call(voter, poll, [0])
|
||||||
|
Sidekiq::Testing.fake! do
|
||||||
subject.call(status, status.account_id, text: 'Foo', poll: { options: %w(Bar Baz Foo), expires_in: 5.days.to_i })
|
subject.call(status, status.account_id, text: 'Foo', poll: { options: %w(Bar Baz Foo), expires_in: 5.days.to_i })
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
|
||||||
it 'updates poll' do
|
it 'updates poll' do
|
||||||
poll = status.poll.reload
|
poll = status.poll.reload
|
||||||
|
@ -138,6 +140,11 @@ RSpec.describe UpdateStatusService, type: :service do
|
||||||
it 'saves edit history' do
|
it 'saves edit history' do
|
||||||
expect(status.edits.pluck(:poll_options)).to eq [%w(Foo Bar), %w(Bar Baz Foo)]
|
expect(status.edits.pluck(:poll_options)).to eq [%w(Foo Bar), %w(Bar Baz Foo)]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
it 'requeues expiration notification' do
|
||||||
|
poll = status.poll.reload
|
||||||
|
expect(PollExpirationNotifyWorker).to have_enqueued_sidekiq_job(poll.id).at(poll.expires_at + 5.minutes)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
context 'when mentions in text change' do
|
context 'when mentions in text change' do
|
||||||
|
|
|
@ -4,10 +4,69 @@ require 'rails_helper'
|
||||||
|
|
||||||
describe PollExpirationNotifyWorker do
|
describe PollExpirationNotifyWorker do
|
||||||
let(:worker) { described_class.new }
|
let(:worker) { described_class.new }
|
||||||
|
let(:account) { Fabricate(:account, domain: remote? ? 'example.com' : nil) }
|
||||||
|
let(:status) { Fabricate(:status, account: account) }
|
||||||
|
let(:poll) { Fabricate(:poll, status: status, account: account) }
|
||||||
|
let(:remote?) { false }
|
||||||
|
let(:poll_vote) { Fabricate(:poll_vote, poll: poll) }
|
||||||
|
|
||||||
|
describe '#perform' do
|
||||||
|
around do |example|
|
||||||
|
Sidekiq::Testing.fake! do
|
||||||
|
example.run
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
describe 'perform' do
|
|
||||||
it 'runs without error for missing record' do
|
it 'runs without error for missing record' do
|
||||||
expect { worker.perform(nil) }.to_not raise_error
|
expect { worker.perform(nil) }.to_not raise_error
|
||||||
end
|
end
|
||||||
|
|
||||||
|
context 'when poll is not expired' do
|
||||||
|
it 'requeues job' do
|
||||||
|
worker.perform(poll.id)
|
||||||
|
expect(described_class.sidekiq_options_hash['lock']).to be :until_executing
|
||||||
|
expect(described_class).to have_enqueued_sidekiq_job(poll.id).at(poll.expires_at + 5.minutes)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when poll is expired' do
|
||||||
|
before do
|
||||||
|
poll_vote
|
||||||
|
|
||||||
|
travel_to poll.expires_at + 5.minutes
|
||||||
|
|
||||||
|
worker.perform(poll.id)
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when poll is local' do
|
||||||
|
it 'notifies voters' do
|
||||||
|
expect(ActivityPub::DistributePollUpdateWorker).to have_enqueued_sidekiq_job(poll.status.id)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'notifies owner' do
|
||||||
|
expect(LocalNotificationWorker).to have_enqueued_sidekiq_job(poll.account.id, poll.id, 'Poll', 'poll')
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'notifies local voters' do
|
||||||
|
expect(LocalNotificationWorker).to have_enqueued_sidekiq_job(poll_vote.account.id, poll.id, 'Poll', 'poll')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when poll is remote' do
|
||||||
|
let(:remote?) { true }
|
||||||
|
|
||||||
|
it 'does not notify remote voters' do
|
||||||
|
expect(ActivityPub::DistributePollUpdateWorker).to_not have_enqueued_sidekiq_job(poll.status.id)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'does not notify owner' do
|
||||||
|
expect(LocalNotificationWorker).to_not have_enqueued_sidekiq_job(poll.account.id, poll.id, 'Poll', 'poll')
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'notifies local voters' do
|
||||||
|
expect(LocalNotificationWorker).to have_enqueued_sidekiq_job(poll_vote.account.id, poll.id, 'Poll', 'poll')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
Reference in New Issue