2021-04-17 20:07:23 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'spec_helper'
|
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
RSpec.describe Gitlab::SidekiqMiddleware::SizeLimiter::Validator, :aggregate_failures do
|
2021-09-04 01:27:46 +05:30
|
|
|
let(:base_payload) do
|
|
|
|
{
|
|
|
|
"class" => "ARandomWorker",
|
|
|
|
"queue" => "a_worker",
|
|
|
|
"retry" => true,
|
|
|
|
"jid" => "d774900367dc8b2962b2479c",
|
|
|
|
"created_at" => 1234567890,
|
|
|
|
"enqueued_at" => 1234567890
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def job_payload(args = {})
|
|
|
|
base_payload.merge('args' => args)
|
|
|
|
end
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
let(:worker_class) do
|
|
|
|
Class.new do
|
|
|
|
def self.name
|
|
|
|
"TestSizeLimiterWorker"
|
|
|
|
end
|
|
|
|
|
|
|
|
include ApplicationWorker
|
|
|
|
|
|
|
|
def perform(*args); end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
2021-11-11 11:23:49 +05:30
|
|
|
# Settings aren't in the database in specs, but stored in memory, this is fine
|
|
|
|
# for these tests.
|
|
|
|
allow(Gitlab::CurrentSettings).to receive(:current_application_settings?).and_return(true)
|
2021-04-17 20:07:23 +05:30
|
|
|
stub_const("TestSizeLimiterWorker", worker_class)
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#initialize' do
|
2021-11-11 11:23:49 +05:30
|
|
|
context 'configuration from application settings' do
|
|
|
|
let(:validator) { described_class.new(worker_class, job_payload) }
|
|
|
|
|
|
|
|
it 'has the right defaults' do
|
|
|
|
expect(validator.mode).to eq(described_class::COMPRESS_MODE)
|
|
|
|
expect(validator.compression_threshold).to eq(described_class::DEFAULT_COMPRESSION_THRESHOLD_BYTES)
|
|
|
|
expect(validator.size_limit).to eq(described_class::DEFAULT_SIZE_LIMIT)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'allows configuration through application settings' do
|
|
|
|
stub_application_setting(
|
|
|
|
sidekiq_job_limiter_mode: 'track',
|
|
|
|
sidekiq_job_limiter_compression_threshold_bytes: 1,
|
|
|
|
sidekiq_job_limiter_limit_bytes: 2
|
|
|
|
)
|
|
|
|
|
|
|
|
expect(validator.mode).to eq(described_class::TRACK_MODE)
|
|
|
|
expect(validator.compression_threshold).to eq(1)
|
|
|
|
expect(validator.size_limit).to eq(2)
|
|
|
|
end
|
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
shared_examples 'validate limit job payload size' do
|
|
|
|
context 'in track mode' do
|
2021-09-04 01:27:46 +05:30
|
|
|
let(:compression_threshold) { nil }
|
2021-04-17 20:07:23 +05:30
|
|
|
let(:mode) { 'track' }
|
|
|
|
|
|
|
|
context 'when size limit is 0' do
|
|
|
|
let(:size_limit) { 0 }
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:job) { job_payload(a: 'a' * 300) }
|
2021-04-17 20:07:23 +05:30
|
|
|
|
|
|
|
it 'does not track jobs' do
|
|
|
|
expect(Gitlab::ErrorTracking).not_to receive(:track_exception)
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not raise exception' do
|
2021-09-04 01:27:46 +05:30
|
|
|
expect do
|
2021-11-18 22:05:49 +05:30
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
2021-09-04 01:27:46 +05:30
|
|
|
end.not_to raise_error
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
it 'marks the job as validated' do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
|
|
|
|
expect(job['size_limiter']).to eq('validated')
|
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'when job size is bigger than size limit' do
|
|
|
|
let(:size_limit) { 50 }
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:job) { job_payload(a: 'a' * 300) }
|
2021-04-17 20:07:23 +05:30
|
|
|
|
|
|
|
it 'tracks job' do
|
|
|
|
expect(Gitlab::ErrorTracking).to receive(:track_exception).with(
|
|
|
|
be_a(Gitlab::SidekiqMiddleware::SizeLimiter::ExceedLimitError)
|
|
|
|
)
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not raise an exception' do
|
2021-09-04 01:27:46 +05:30
|
|
|
expect do
|
2021-11-18 22:05:49 +05:30
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
2021-09-04 01:27:46 +05:30
|
|
|
end.not_to raise_error
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
it 'marks the job as tracked' do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
|
|
|
|
expect(job['size_limiter']).to eq('tracked')
|
|
|
|
end
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
context 'when the worker has big_payload attribute' do
|
|
|
|
before do
|
|
|
|
worker_class.big_payload!
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not track jobs' do
|
|
|
|
expect(Gitlab::ErrorTracking).not_to receive(:track_exception)
|
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
validate.call(TestSizeLimiterWorker, job_payload(a: 'a' * 300))
|
|
|
|
validate.call('TestSizeLimiterWorker', job_payload(a: 'a' * 300))
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not raise an exception' do
|
2021-09-04 01:27:46 +05:30
|
|
|
expect do
|
|
|
|
validate.call(TestSizeLimiterWorker, job_payload(a: 'a' * 300))
|
|
|
|
end.not_to raise_error
|
|
|
|
expect do
|
|
|
|
validate.call('TestSizeLimiterWorker', job_payload(a: 'a' * 300))
|
|
|
|
end.not_to raise_error
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
it 'marks the job as validated' do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
|
|
|
|
expect(job['size_limiter']).to eq('validated')
|
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when job size is less than size limit' do
|
|
|
|
let(:size_limit) { 50 }
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:job) { job_payload(a: 'a') }
|
2021-04-17 20:07:23 +05:30
|
|
|
|
|
|
|
it 'does not track job' do
|
|
|
|
expect(Gitlab::ErrorTracking).not_to receive(:track_exception)
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not raise an exception' do
|
2021-11-18 22:05:49 +05:30
|
|
|
expect { validate.call(TestSizeLimiterWorker, job) }.not_to raise_error
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'marks the job as validated' do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
|
|
|
|
expect(job['size_limiter']).to eq('validated')
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
context 'in compress mode' do
|
2021-09-30 23:02:18 +05:30
|
|
|
let(:size_limit) { 50 }
|
|
|
|
let(:compression_threshold) { 30 }
|
2021-09-04 01:27:46 +05:30
|
|
|
let(:mode) { 'compress' }
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
context 'when job size is less than compression threshold' do
|
|
|
|
let(:job) { job_payload(a: 'a' * 10) }
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
it 'does not raise an exception' do
|
|
|
|
expect(::Gitlab::SidekiqMiddleware::SizeLimiter::Compressor).not_to receive(:compress)
|
2021-11-18 22:05:49 +05:30
|
|
|
expect { validate.call(TestSizeLimiterWorker, job) }.not_to raise_error
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'marks the job as validated' do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
|
|
|
|
expect(job['size_limiter']).to eq('validated')
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
context 'when job size is bigger than compression threshold and less than size limit after compressed' do
|
|
|
|
let(:args) { { a: 'a' * 300 } }
|
|
|
|
let(:job) { job_payload(args) }
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
it 'does not raise an exception' do
|
|
|
|
expect(::Gitlab::SidekiqMiddleware::SizeLimiter::Compressor).to receive(:compress).with(
|
|
|
|
job, Sidekiq.dump_json(args)
|
|
|
|
).and_return('a' * 40)
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
expect do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
end.not_to raise_error
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
it 'marks the job as validated' do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
|
|
|
|
expect(job['size_limiter']).to eq('validated')
|
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
context 'when job size is bigger than compression threshold and size limit is 0' do
|
|
|
|
let(:size_limit) { 0 }
|
|
|
|
let(:args) { { a: 'a' * 300 } }
|
|
|
|
let(:job) { job_payload(args) }
|
|
|
|
|
|
|
|
it 'does not raise an exception and compresses the arguments' do
|
|
|
|
expect(::Gitlab::SidekiqMiddleware::SizeLimiter::Compressor).to receive(:compress).with(
|
|
|
|
job, Sidekiq.dump_json(args)
|
|
|
|
).and_return('a' * 40)
|
|
|
|
|
|
|
|
expect do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
end.not_to raise_error
|
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
it 'marks the job as validated' do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
|
|
|
|
expect(job['size_limiter']).to eq('validated')
|
|
|
|
end
|
2021-11-11 11:23:49 +05:30
|
|
|
end
|
|
|
|
|
2021-09-30 23:02:18 +05:30
|
|
|
context 'when the job was already compressed' do
|
|
|
|
let(:job) do
|
|
|
|
job_payload({ a: 'a' * 10 })
|
|
|
|
.merge(Gitlab::SidekiqMiddleware::SizeLimiter::Compressor::COMPRESSED_KEY => true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not compress the arguments again' do
|
|
|
|
expect(Gitlab::SidekiqMiddleware::SizeLimiter::Compressor).not_to receive(:compress)
|
|
|
|
|
|
|
|
expect { validate.call(TestSizeLimiterWorker, job) }.not_to raise_error
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
context 'when job size is bigger than compression threshold and bigger than size limit after compressed' do
|
|
|
|
let(:args) { { a: 'a' * 3000 } }
|
|
|
|
let(:job) { job_payload(args) }
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
it 'raises an exception' do
|
2021-09-04 01:27:46 +05:30
|
|
|
expect(::Gitlab::SidekiqMiddleware::SizeLimiter::Compressor).to receive(:compress).with(
|
|
|
|
job, Sidekiq.dump_json(args)
|
|
|
|
).and_return('a' * 60)
|
|
|
|
|
|
|
|
expect do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
end.to raise_error(Gitlab::SidekiqMiddleware::SizeLimiter::ExceedLimitError)
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
expect(job['size_limiter']).to eq(nil)
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
2021-11-11 11:23:49 +05:30
|
|
|
|
|
|
|
it 'does not raise an exception when the worker allows big payloads' do
|
|
|
|
worker_class.big_payload!
|
|
|
|
|
|
|
|
expect(::Gitlab::SidekiqMiddleware::SizeLimiter::Compressor).to receive(:compress).with(
|
|
|
|
job, Sidekiq.dump_json(args)
|
|
|
|
).and_return('a' * 60)
|
|
|
|
|
|
|
|
expect do
|
|
|
|
validate.call(TestSizeLimiterWorker, job)
|
|
|
|
end.not_to raise_error
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
expect(job['size_limiter']).to eq('validated')
|
2021-11-11 11:23:49 +05:30
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
describe '.validate!' do
|
|
|
|
let(:validate) { ->(worker_class, job) { described_class.validate!(worker_class, job) } }
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
it_behaves_like 'validate limit job payload size' do
|
2021-04-17 20:07:23 +05:30
|
|
|
before do
|
2021-11-11 11:23:49 +05:30
|
|
|
stub_application_setting(
|
|
|
|
sidekiq_job_limiter_mode: mode,
|
|
|
|
sidekiq_job_limiter_compression_threshold_bytes: compression_threshold,
|
|
|
|
sidekiq_job_limiter_limit_bytes: size_limit
|
|
|
|
)
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
2021-11-11 11:23:49 +05:30
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
it "skips background migrations" do
|
|
|
|
expect(described_class).not_to receive(:new)
|
|
|
|
|
|
|
|
described_class::EXEMPT_WORKER_NAMES.each do |class_name|
|
|
|
|
validate.call(class_name.constantize, job_payload)
|
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
it "skips jobs that are already validated" do
|
|
|
|
expect(described_class).to receive(:new).once.and_call_original
|
|
|
|
|
|
|
|
job = job_payload
|
|
|
|
|
|
|
|
described_class.validate!(TestSizeLimiterWorker, job)
|
|
|
|
described_class.validate!(TestSizeLimiterWorker, job)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.validated?' do
|
|
|
|
let(:job) { job_payload }
|
|
|
|
|
|
|
|
it 'returns true when the job is already validated' do
|
|
|
|
described_class.validate!(TestSizeLimiterWorker, job)
|
|
|
|
|
|
|
|
expect(described_class.validated?(job)).to eq(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false when job is not yet validated' do
|
|
|
|
expect(described_class.validated?(job)).to eq(false)
|
|
|
|
end
|
2021-11-11 11:23:49 +05:30
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
describe '#validate!' do
|
2021-12-11 22:18:48 +05:30
|
|
|
let(:validate) do
|
|
|
|
->(worker_class, job) do
|
|
|
|
described_class.new(worker_class, job).validate!
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-12-11 22:18:48 +05:30
|
|
|
before do
|
|
|
|
stub_application_setting(
|
|
|
|
sidekiq_job_limiter_mode: mode,
|
|
|
|
sidekiq_job_limiter_compression_threshold_bytes: compression_threshold,
|
|
|
|
sidekiq_job_limiter_limit_bytes: size_limit
|
|
|
|
)
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
2021-12-11 22:18:48 +05:30
|
|
|
|
|
|
|
it_behaves_like 'validate limit job payload size'
|
2021-04-17 20:07:23 +05:30
|
|
|
end
|
|
|
|
end
|