163 lines
5.5 KiB
Ruby
163 lines
5.5 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'spec_helper'
|
|
require 'rspec-parameterized'
|
|
|
|
RSpec.describe Gitlab::Instrumentation::RedisInterceptor, :clean_gitlab_redis_shared_state, :request_store do
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
describe 'read and write' do
|
|
where(:setup, :command, :expect_write, :expect_read) do
|
|
# The response is 'OK', the request size is the combined size of array
|
|
# elements. Exercise counting of a status reply.
|
|
[] | [:set, 'foo', 'bar'] | 3 + 3 + 3 | 2
|
|
|
|
# The response is 1001, so 4 bytes. Exercise counting an integer reply.
|
|
[[:set, 'foobar', 1000]] | [:incr, 'foobar'] | 4 + 6 | 4
|
|
|
|
# Exercise counting empty multi bulk reply
|
|
[] | [:hgetall, 'foobar'] | 7 + 6 | 0
|
|
|
|
# Hgetall response length is combined length of keys and values in the
|
|
# hash. Exercises counting of a multi bulk reply
|
|
[[:hset, 'myhash', 'field', 'hello world']] | [:hgetall, 'myhash'] | 7 + 6 | 5 + 11
|
|
|
|
# Exercise counting of a bulk reply
|
|
[[:set, 'foo', 'bar' * 100]] | [:get, 'foo'] | 3 + 3 | 3 * 100
|
|
|
|
# Nested array response: [['foo', 0], ['bar', 1]]
|
|
[[:zadd, 'myset', 0, 'foo'], [:zadd, 'myset', 1, 'bar']] | [:zrange, 'myset', 0, -1, 'withscores'] | 6 + 5 + 1 + 2 + 10 | 3 + 1 + 3 + 1
|
|
end
|
|
|
|
with_them do
|
|
it 'counts bytes read and written' do
|
|
Gitlab::Redis::SharedState.with do |redis|
|
|
setup.each { |cmd| redis.call(cmd) }
|
|
RequestStore.clear!
|
|
redis.call(command)
|
|
end
|
|
|
|
expect(Gitlab::Instrumentation::Redis.read_bytes).to eq(expect_read)
|
|
expect(Gitlab::Instrumentation::Redis.write_bytes).to eq(expect_write)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'counting' do
|
|
let(:instrumentation_class) { Gitlab::Redis::SharedState.instrumentation_class }
|
|
|
|
it 'counts successful requests' do
|
|
expect(instrumentation_class).to receive(:instance_count_request).with(1).and_call_original
|
|
|
|
Gitlab::Redis::SharedState.with { |redis| redis.call(:get, 'foobar') }
|
|
end
|
|
|
|
it 'counts successful pipelined requests' do
|
|
expect(instrumentation_class).to receive(:instance_count_request).with(2).and_call_original
|
|
|
|
Gitlab::Redis::SharedState.with do |redis|
|
|
redis.pipelined do |pipeline|
|
|
pipeline.call(:get, '{foobar}buz')
|
|
pipeline.call(:get, '{foobar}baz')
|
|
end
|
|
end
|
|
end
|
|
|
|
it 'counts exceptions' do
|
|
expect(instrumentation_class).to receive(:instance_count_exception)
|
|
.with(instance_of(Redis::CommandError)).and_call_original
|
|
expect(instrumentation_class).to receive(:instance_count_request).and_call_original
|
|
|
|
expect do
|
|
Gitlab::Redis::SharedState.with do |redis|
|
|
redis.call(:auth, 'foo', 'bar')
|
|
end
|
|
end.to raise_exception(Redis::CommandError)
|
|
end
|
|
end
|
|
|
|
describe 'latency' do
|
|
let(:instrumentation_class) { Gitlab::Redis::SharedState.instrumentation_class }
|
|
|
|
describe 'commands in the apdex' do
|
|
where(:command) do
|
|
[
|
|
[[:get, 'foobar']],
|
|
[%w[GET foobar]]
|
|
]
|
|
end
|
|
|
|
with_them do
|
|
it 'measures requests we want in the apdex' do
|
|
expect(instrumentation_class).to receive(:instance_observe_duration).with(a_value > 0)
|
|
.and_call_original
|
|
|
|
Gitlab::Redis::SharedState.with { |redis| redis.call(*command) }
|
|
end
|
|
end
|
|
|
|
context 'with pipelined commands' do
|
|
it 'measures requests that do not have blocking commands' do
|
|
expect(instrumentation_class).to receive(:instance_observe_duration).twice.with(a_value > 0)
|
|
.and_call_original
|
|
|
|
Gitlab::Redis::SharedState.with do |redis|
|
|
redis.pipelined do |pipeline|
|
|
pipeline.call(:get, '{foobar}:buz')
|
|
pipeline.call(:get, '{foobar}baz')
|
|
end
|
|
end
|
|
end
|
|
|
|
it 'raises error when keys are not from the same slot' do
|
|
expect do
|
|
Gitlab::Redis::SharedState.with do |redis|
|
|
redis.pipelined do |pipeline|
|
|
pipeline.call(:get, 'foo')
|
|
pipeline.call(:get, 'bar')
|
|
end
|
|
end
|
|
end.to raise_error(instance_of(Gitlab::Instrumentation::RedisClusterValidator::CrossSlotError))
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'commands not in the apdex' do
|
|
where(:command) do
|
|
[
|
|
[%w[brpop foobar 0.01]],
|
|
[%w[blpop foobar 0.01]],
|
|
[%w[brpoplpush foobar bazqux 0.01]],
|
|
[%w[bzpopmin foobar 0.01]],
|
|
[%w[bzpopmax foobar 0.01]],
|
|
[%w[xread block 1 streams mystream 0-0]],
|
|
[%w[xreadgroup group mygroup myconsumer block 1 streams foobar 0-0]]
|
|
]
|
|
end
|
|
|
|
with_them do
|
|
it 'skips requests we do not want in the apdex' do
|
|
expect(instrumentation_class).not_to receive(:instance_observe_duration)
|
|
|
|
begin
|
|
Gitlab::Redis::SharedState.with { |redis| redis.call(*command) }
|
|
rescue Gitlab::Instrumentation::RedisClusterValidator::CrossSlotError, ::Redis::CommandError
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'with pipelined commands' do
|
|
it 'skips requests that have blocking commands', quarantine: 'https://gitlab.com/gitlab-org/gitlab/-/issues/373026' do
|
|
expect(instrumentation_class).not_to receive(:instance_observe_duration)
|
|
|
|
Gitlab::Redis::SharedState.with do |redis|
|
|
redis.pipelined do |pipeline|
|
|
pipeline.call(:get, 'foo')
|
|
pipeline.call(:brpop, 'foobar', '0.01')
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|