add new send workers
This commit is contained in:
parent
b1d30aa9cc
commit
7596a49b46
6 changed files with 160 additions and 0 deletions
29
app/workers/send_base.rb
Normal file
29
app/workers/send_base.rb
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
module Workers
|
||||||
|
class SendBase < Base
|
||||||
|
sidekiq_options queue: :http, retry: 0
|
||||||
|
|
||||||
|
MAX_RETRIES = AppConfig.environment.sidekiq.retry.get.to_i
|
||||||
|
|
||||||
|
protected
|
||||||
|
|
||||||
|
def schedule_retry(retry_count, sender_id, obj_str, failed_urls)
|
||||||
|
if retry_count < MAX_RETRIES
|
||||||
|
yield(seconds_to_delay(retry_count), retry_count)
|
||||||
|
else
|
||||||
|
logger.warn "status=abandon sender=#{sender_id} obj=#{obj_str} failed_urls='[#{failed_urls.join(', ')}]'"
|
||||||
|
raise MaxRetriesReached
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
# based on Sidekiq::Middleware::Server::RetryJobs#seconds_to_delay
|
||||||
|
def seconds_to_delay(count)
|
||||||
|
((count + 3)**4) + (rand(30) * (count + 1))
|
||||||
|
end
|
||||||
|
|
||||||
|
# send job to the dead job queue
|
||||||
|
class MaxRetriesReached < RuntimeError
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
13
app/workers/send_private.rb
Normal file
13
app/workers/send_private.rb
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
module Workers
|
||||||
|
class SendPrivate < SendBase
|
||||||
|
def perform(sender_id, obj_str, targets, retry_count=0)
|
||||||
|
targets_to_retry = DiasporaFederation::Federation::Sender.private(sender_id, obj_str, targets)
|
||||||
|
|
||||||
|
return if targets_to_retry.empty?
|
||||||
|
|
||||||
|
schedule_retry(retry_count + 1, sender_id, obj_str, targets_to_retry.keys) do |delay, new_retry_count|
|
||||||
|
Workers::SendPrivate.perform_in(delay, sender_id, obj_str, targets_to_retry, new_retry_count)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
13
app/workers/send_public.rb
Normal file
13
app/workers/send_public.rb
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
module Workers
|
||||||
|
class SendPublic < SendBase
|
||||||
|
def perform(sender_id, obj_str, urls, xml, retry_count=0)
|
||||||
|
urls_to_retry = DiasporaFederation::Federation::Sender.public(sender_id, obj_str, urls, xml)
|
||||||
|
|
||||||
|
return if urls_to_retry.empty?
|
||||||
|
|
||||||
|
schedule_retry(retry_count + 1, sender_id, obj_str, urls_to_retry) do |delay, new_retry_count|
|
||||||
|
Workers::SendPublic.perform_in(delay, sender_id, obj_str, urls_to_retry, xml, new_retry_count)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
20
spec/workers/send_base_spec.rb
Normal file
20
spec/workers/send_base_spec.rb
Normal file
|
|
@ -0,0 +1,20 @@
|
||||||
|
require "spec_helper"
|
||||||
|
|
||||||
|
describe Workers::SendBase do
|
||||||
|
it "retries first time after at least 256 seconds" do
|
||||||
|
retry_delay = Workers::SendBase.new.send(:seconds_to_delay, 1)
|
||||||
|
expect(retry_delay).to be >= 256
|
||||||
|
expect(retry_delay).to be < 316
|
||||||
|
end
|
||||||
|
|
||||||
|
it "increases the interval for each retry" do
|
||||||
|
expect(Workers::SendBase.new.send(:seconds_to_delay, 2)).to be >= 625
|
||||||
|
expect(Workers::SendBase.new.send(:seconds_to_delay, 3)).to be >= 1_296
|
||||||
|
expect(Workers::SendBase.new.send(:seconds_to_delay, 4)).to be >= 2_401
|
||||||
|
expect(Workers::SendBase.new.send(:seconds_to_delay, 5)).to be >= 4_096
|
||||||
|
expect(Workers::SendBase.new.send(:seconds_to_delay, 6)).to be >= 6_561
|
||||||
|
expect(Workers::SendBase.new.send(:seconds_to_delay, 7)).to be >= 10_000
|
||||||
|
expect(Workers::SendBase.new.send(:seconds_to_delay, 8)).to be >= 14_641
|
||||||
|
expect(Workers::SendBase.new.send(:seconds_to_delay, 9)).to be >= 20_736
|
||||||
|
end
|
||||||
|
end
|
||||||
44
spec/workers/send_private_spec.rb
Normal file
44
spec/workers/send_private_spec.rb
Normal file
|
|
@ -0,0 +1,44 @@
|
||||||
|
require "spec_helper"
|
||||||
|
|
||||||
|
describe Workers::SendPrivate do
|
||||||
|
let(:sender_id) { "any_user@example.org" }
|
||||||
|
let(:obj_str) { "status_message@guid" }
|
||||||
|
let(:targets) {
|
||||||
|
{
|
||||||
|
"https://example.org/receive/user/guid" => "<xml>post</xml>",
|
||||||
|
"https://example.com/receive/user/guid" => "<xml>post2</xml>"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let(:failing_targets) { {"https://example.org/receive/user/guid" => "<xml>post</xml>"} }
|
||||||
|
|
||||||
|
it "succeeds if all urls were successful" do
|
||||||
|
expect(DiasporaFederation::Federation::Sender).to receive(:private).with(
|
||||||
|
sender_id, obj_str, targets
|
||||||
|
).and_return({})
|
||||||
|
expect(Workers::SendPrivate).not_to receive(:perform_in)
|
||||||
|
|
||||||
|
Workers::SendPrivate.new.perform(sender_id, obj_str, targets)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "retries failing urls" do
|
||||||
|
expect(DiasporaFederation::Federation::Sender).to receive(:private).with(
|
||||||
|
sender_id, obj_str, targets
|
||||||
|
).and_return(failing_targets)
|
||||||
|
expect(Workers::SendPrivate).to receive(:perform_in).with(
|
||||||
|
kind_of(Fixnum), sender_id, obj_str, failing_targets, 1
|
||||||
|
)
|
||||||
|
|
||||||
|
Workers::SendPrivate.new.perform(sender_id, obj_str, targets)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "does not retry failing urls if max retries is reached" do
|
||||||
|
expect(DiasporaFederation::Federation::Sender).to receive(:private).with(
|
||||||
|
sender_id, obj_str, targets
|
||||||
|
).and_return(failing_targets)
|
||||||
|
expect(Workers::SendPrivate).not_to receive(:perform_in)
|
||||||
|
|
||||||
|
expect {
|
||||||
|
Workers::SendPrivate.new.perform(sender_id, obj_str, targets, 9)
|
||||||
|
}.to raise_error Workers::SendBase::MaxRetriesReached
|
||||||
|
end
|
||||||
|
end
|
||||||
41
spec/workers/send_public_spec.rb
Normal file
41
spec/workers/send_public_spec.rb
Normal file
|
|
@ -0,0 +1,41 @@
|
||||||
|
require "spec_helper"
|
||||||
|
|
||||||
|
describe Workers::SendPublic do
|
||||||
|
let(:sender_id) { "any_user@example.org" }
|
||||||
|
let(:obj_str) { "status_message@guid" }
|
||||||
|
let(:urls) { ["https://example.org/receive/public", "https://example.com/receive/public"] }
|
||||||
|
let(:xml) { "<xml>post</xml>" }
|
||||||
|
|
||||||
|
it "succeeds if all urls were successful" do
|
||||||
|
expect(DiasporaFederation::Federation::Sender).to receive(:public).with(
|
||||||
|
sender_id, obj_str, urls, xml
|
||||||
|
).and_return([])
|
||||||
|
expect(Workers::SendPublic).not_to receive(:perform_in)
|
||||||
|
|
||||||
|
Workers::SendPublic.new.perform(sender_id, obj_str, urls, xml)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "retries failing urls" do
|
||||||
|
failing_urls = [urls.at(0)]
|
||||||
|
expect(DiasporaFederation::Federation::Sender).to receive(:public).with(
|
||||||
|
sender_id, obj_str, urls, xml
|
||||||
|
).and_return(failing_urls)
|
||||||
|
expect(Workers::SendPublic).to receive(:perform_in).with(
|
||||||
|
kind_of(Fixnum), sender_id, obj_str, failing_urls, xml, 1
|
||||||
|
)
|
||||||
|
|
||||||
|
Workers::SendPublic.new.perform(sender_id, obj_str, urls, xml)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "does not retry failing urls if max retries is reached" do
|
||||||
|
failing_urls = [urls.at(0)]
|
||||||
|
expect(DiasporaFederation::Federation::Sender).to receive(:public).with(
|
||||||
|
sender_id, obj_str, urls, xml
|
||||||
|
).and_return(failing_urls)
|
||||||
|
expect(Workers::SendPublic).not_to receive(:perform_in)
|
||||||
|
|
||||||
|
expect {
|
||||||
|
Workers::SendPublic.new.perform(sender_id, obj_str, urls, xml, 9)
|
||||||
|
}.to raise_error Workers::SendBase::MaxRetriesReached
|
||||||
|
end
|
||||||
|
end
|
||||||
Loading…
Reference in a new issue