Commit 653778ef authored by Quang-Minh Nguyen's avatar Quang-Minh Nguyen

Integrate size limiter into Sidekiq middleware chain

Isuse https://gitlab.com/gitlab-com/gl-infra/scalability/-/issues/825
parent ef89d2af
......@@ -36,6 +36,8 @@ module Gitlab
chain.add ::Gitlab::SidekiqMiddleware::DuplicateJobs::Client
chain.add ::Gitlab::SidekiqStatus::ClientMiddleware
chain.add ::Gitlab::SidekiqMiddleware::AdminMode::Client
# Size limiter should be placed at the bottom, but before the metrics midleware
chain.add ::Gitlab::SidekiqMiddleware::SizeLimiter::Client
chain.add ::Gitlab::SidekiqMiddleware::ClientMetrics
end
end
......
# frozen_string_literal: true
module Gitlab
module SidekiqMiddleware
module SizeLimiter
# This midleware is inserted into Sidekiq **client** middleware chain. It
# prevents the caller from dispatching a too-large job payload. The job
# payload should be small and simple. Read more at:
# https://github.com/mperham/sidekiq/wiki/Best-Practices#1-make-your-job-parameters-small-and-simple
class Client
def call(worker_class, job, queue, _redis_pool)
Validator.validate!(worker_class, job)
yield
end
end
end
end
end
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Gitlab::SidekiqMiddleware::SizeLimiter::Client, :clean_gitlab_redis_queues do
let(:worker_class) do
Class.new do
def self.name
"TestSizeLimiterWorker"
end
include ApplicationWorker
def perform(*args); end
end
end
before do
stub_const("TestSizeLimiterWorker", worker_class)
end
describe '#call' do
context 'when the validator rejects the job' do
before do
allow(Gitlab::SidekiqMiddleware::SizeLimiter::Validator).to receive(:validate!).and_raise(
Gitlab::SidekiqMiddleware::SizeLimiter::ExceedLimitError.new(
TestSizeLimiterWorker, 500, 300
)
)
end
it 'raises an exception when scheduling job with #perform_at' do
expect do
TestSizeLimiterWorker.perform_at(30.seconds.from_now, 1, 2, 3)
end.to raise_error Gitlab::SidekiqMiddleware::SizeLimiter::ExceedLimitError
end
it 'raises an exception when scheduling job with #perform_async' do
expect do
TestSizeLimiterWorker.perform_async(1, 2, 3)
end.to raise_error Gitlab::SidekiqMiddleware::SizeLimiter::ExceedLimitError
end
it 'raises an exception when scheduling job with #perform_in' do
expect do
TestSizeLimiterWorker.perform_in(3.seconds, 1, 2, 3)
end.to raise_error Gitlab::SidekiqMiddleware::SizeLimiter::ExceedLimitError
end
end
context 'when the validator validates the job suscessfully' do
before do
# Do nothing
allow(Gitlab::SidekiqMiddleware::SizeLimiter::Client).to receive(:validate!)
end
it 'raises an exception when scheduling job with #perform_at' do
expect do
TestSizeLimiterWorker.perform_at(30.seconds.from_now, 1, 2, 3)
end.not_to raise_error
expect(TestSizeLimiterWorker.jobs).to contain_exactly(
a_hash_including(
"class" => "TestSizeLimiterWorker",
"args" => [1, 2, 3],
"at" => be_a(Float)
)
)
end
it 'raises an exception when scheduling job with #perform_async' do
expect do
TestSizeLimiterWorker.perform_async(1, 2, 3)
end.not_to raise_error
expect(TestSizeLimiterWorker.jobs).to contain_exactly(
a_hash_including(
"class" => "TestSizeLimiterWorker",
"args" => [1, 2, 3]
)
)
end
it 'raises an exception when scheduling job with #perform_in' do
expect do
TestSizeLimiterWorker.perform_in(3.seconds, 1, 2, 3)
end.not_to raise_error
expect(TestSizeLimiterWorker.jobs).to contain_exactly(
a_hash_including(
"class" => "TestSizeLimiterWorker",
"args" => [1, 2, 3],
"at" => be_a(Float)
)
)
end
end
end
end
......@@ -177,6 +177,7 @@ RSpec.describe Gitlab::SidekiqMiddleware do
::Gitlab::SidekiqMiddleware::DuplicateJobs::Client,
::Gitlab::SidekiqStatus::ClientMiddleware,
::Gitlab::SidekiqMiddleware::AdminMode::Client,
::Gitlab::SidekiqMiddleware::SizeLimiter::Client,
::Gitlab::SidekiqMiddleware::ClientMetrics
]
end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment