Commit c34532e2 authored by nmilojevic1's avatar nmilojevic1 Committed by Nikola Milojevic

Imporve multistore and specs

- Fix fallback read for multi-store
- Add scard support for multi store
- Use separate config for sessions
- Fix specs for multi-store
- Add additional error layer
parent 3e1462d2
......@@ -3,14 +3,19 @@
module Gitlab
module Redis
class MultiStore
class ReadFromPrimaryError < StandardError
def message
'Value not found on the redis primary store. Read from the redis secondary store successful.'
end
end
class MultiReadError < StandardError
def message
'Value not found, falling back to read from the redis secondary store.'
'Value not found on both primary and secondary store.'
end
end
class MethodMissingError < StandardError
def message
'Method missing. Falling back to execute method on the redis secondary_store.'
'Method missing. Falling back to execute method on the redis secondary store.'
end
end
......@@ -23,6 +28,7 @@ module Gitlab
get
mget
smembers
scard
).freeze
WRITE_COMMANDS = %i(
......@@ -52,67 +58,91 @@ module Gitlab
READ_COMMANDS.each do |name|
define_method(name) do |*args, &block|
if multi_store_enabled?
read_command(name, *args, &block)
else
secondary_store.send(name, *args, &block) # rubocop:disable GitlabSecurity/PublicSend
end
end
end
WRITE_COMMANDS.each do |name|
define_method(name) do |*args, &block|
if multi_store_enabled?
write_command(name, *args, &block)
else
secondary_store.send(name, *args, &block) # rubocop:disable GitlabSecurity/PublicSend
end
end
end
def method_missing(command_name, *args, &block)
if @instance
send_command(@instance, command_name, *args, &block)
else
log_error(MethodMissingError.new, command_name)
increment_method_missing_count(command_name)
secondary_store.send(command_name, *args, &block) # rubocop:disable GitlabSecurity/PublicSend
end
end
def respond_to_missing?(command_name, include_private = false)
true
end
private
def read_command(command_name, *args, &block)
instance = check_redis_store_instance
if instance
send_command(instance, command_name, *args, &block)
if @instance
send_command(@instance, command_name, *args, &block)
else
read_one_with_fallback(command_name, *args, &block)
end
end
def write_command(command_name, *args, &block)
instance = check_redis_store_instance
if instance
send_command(instance, command_name, *args, &block)
if @instance
send_command(@instance, command_name, *args, &block)
else
write_both(command_name, *args, &block)
end
end
def check_redis_store_instance
if multi_store_enabled?
@instance
else
secondary_store # default
end
end
def read_one_with_fallback(command_name, *args, &block)
begin
value = send_command(primary_store, command_name, *args, &block)
rescue StandardError => e
log_error(e, command_name,
multi_store_error_message: FAILED_TO_READ_ERROR_MESSAGE)
ensure
unless value
log_error(MultiReadError.new, command_name)
increment_read_fallback_count(command_name)
end
value ||= fallback_read(command_name, *args, &block)
value
end
def fallback_read(command_name, *args, &block)
value = send_command(secondary_store, command_name, *args, &block)
if value
log_error(ReadFromPrimaryError.new, command_name)
increment_read_fallback_count(command_name)
else
log_error(MultiReadError.new, command_name)
end
value
end
def write_both(command_name, *args, &block)
begin
send_command(primary_store, command_name, *args, &block)
rescue StandardError => e
log_error(e, command_name,
multi_store_error_message: FAILED_TO_WRITE_ERROR_MESSAGE)
ensure
end
send_command(secondary_store, command_name, *args, &block)
end
......@@ -156,17 +186,6 @@ module Gitlab
command_name: command_name,
extra: extra)
end
def method_missing(command_name, *args, &block)
log_error(MethodMissingError.new, command_name)
increment_method_missing_count(command_name)
secondary_store.send(command_name, *args, &block) # rubocop:disable GitlabSecurity/PublicSend
end
def respond_to_missing?(command_name, include_private = false)
true
end
end
end
end
......@@ -38,6 +38,9 @@ sed -i 's|url:.*$|url: redis://redis:6379|g' config/cable.yml
cp config/resque.yml.example config/resque.yml
sed -i 's|url:.*$|url: redis://redis:6379|g' config/resque.yml
cp config/resque.yml.example config/redis.sessions.yml
sed -i 's|url:.*$|url: redis://redis:6379/15|g' config/redis.sessions.yml
if [ "$SETUP_DB" != "false" ]; then
setup_db
elif getent hosts postgres; then
......
......@@ -5,12 +5,17 @@ require 'spec_helper'
RSpec.describe Gitlab::Redis::MultiStore do
using RSpec::Parameterized::TableSyntax
let(:multi_store) { described_class.new(Gitlab::Redis::Sessions.params.merge(serializer: nil), Gitlab::Redis::SharedState.params.merge(serializer: nil))}
let(:primary_store) { multi_store.primary_store }
let(:secondary_store) { multi_store.secondary_store }
let_it_be(:multi_store) { described_class.new(Gitlab::Redis::Sessions.params.merge(serializer: nil), Gitlab::Redis::SharedState.params.merge(serializer: nil))}
let_it_be(:primary_store) { multi_store.primary_store }
let_it_be(:secondary_store) { multi_store.secondary_store }
subject { multi_store.send(name, *args) }
after(:all) do
primary_store.flushdb
secondary_store.flushdb
end
context 'with READ redis commands' do
let_it_be(:key1) { "redis:{1}:key_a" }
let_it_be(:key2) { "redis:{1}:key_b" }
......@@ -26,37 +31,26 @@ RSpec.describe Gitlab::Redis::MultiStore do
'execute :mget command' | :mget | ref(:keys) | ref(:values) | nil
'execute :mget with block' | :mget | ref(:keys) | ref(:values) | ->(value) { value }
'execute :smembers command' | :smembers | ref(:skey) | ref(:svalues) | nil
'execute :scard command' | :scard | ref(:skey) | 2 | nil
end
before(:all) do
redis_shared_state_cleanup!
redis_sessions_cleanup!
Gitlab::Redis::Sessions.with do |redis|
redis.multi do |multi|
primary_store.multi do |multi|
multi.set(key1, value1)
multi.set(key2, value2)
multi.sadd(skey, value1)
multi.sadd(skey, value2)
end
end
Gitlab::Redis::SharedState.with do |redis|
redis.multi do |multi|
secondary_store.multi do |multi|
multi.set(key1, value1)
multi.set(key2, value2)
multi.sadd(skey, value1)
multi.sadd(skey, value2)
end
end
end
after(:all) do
redis_shared_state_cleanup!
redis_sessions_cleanup!
end
RSpec.shared_examples_for 'reads correct value' do |store|
RSpec.shared_examples_for 'reads correct value' do
it 'returns the correct value' do
if value.is_a?(Array)
# :smemebers does not guarantee the order it will return the values (unsorted set)
......@@ -67,6 +61,58 @@ RSpec.describe Gitlab::Redis::MultiStore do
end
end
RSpec.shared_examples_for 'fallback read from the secondary store' do
it 'fallback and execute on secondary instance' do
expect(secondary_store).to receive(name).with(*args).and_call_original
subject
end
it 'logs the ReadFromPrimaryError' do
expect(Gitlab::ErrorTracking).to receive(:log_exception).with(an_instance_of(Gitlab::Redis::MultiStore::ReadFromPrimaryError),
hash_including(command_name: name))
subject
end
it 'increment read fallback count metrics' do
expect(multi_store).to receive(:increment_read_fallback_count).with(name)
subject
end
include_examples 'reads correct value'
context 'when fallback read from the secondary instance raises and exception' do
before do
allow(secondary_store).to receive(name).with(*args).and_raise(StandardError)
end
it 'fails with exception' do
expect { subject }.to raise_error(StandardError)
end
end
context 'when fallback read from the secondary instance returns no value' do
before do
allow(secondary_store).to receive(name).and_return(nil)
end
it 'logs the MultiReadError error' do
expect(Gitlab::ErrorTracking).to receive(:log_exception).with(an_instance_of(Gitlab::Redis::MultiStore::MultiReadError),
hash_including(command_name: name))
subject
end
it 'does not increment read fallback count metrics' do
expect(multi_store).not_to receive(:increment_read_fallback_count)
subject
end
end
end
with_them do
describe "#{name}" do
before do
......@@ -109,13 +155,7 @@ RSpec.describe Gitlab::Redis::MultiStore do
subject
end
it 'fallback and execute on secondary instance' do
expect(secondary_store).to receive(name).with(*args).and_call_original
expect(Gitlab::ErrorTracking).to receive(:log_exception).with(an_instance_of(Gitlab::Redis::MultiStore::MultiReadError),
hash_including(command_name: name))
subject
end
include_examples 'fallback read from the secondary store'
end
context 'when reading from primary instance return no value' do
......@@ -123,24 +163,7 @@ RSpec.describe Gitlab::Redis::MultiStore do
allow(primary_store).to receive(name).and_return(nil)
end
it 'fallback and execute on secondary instance' do
expect(secondary_store).to receive(name).with(*args).and_call_original
subject
end
it 'logs the fallback' do
expect(Gitlab::ErrorTracking).to receive(:log_exception).with(an_instance_of(Gitlab::Redis::MultiStore::MultiReadError),
hash_including(command_name: name))
subject
end
it 'increment metrics' do
expect(multi_store).to receive(:increment_read_fallback_count).with(name)
subject
end
include_examples 'fallback read from the secondary store'
end
context 'when the command is executed within pipelined block' do
......@@ -197,7 +220,7 @@ RSpec.describe Gitlab::Redis::MultiStore do
end
end
context 'with WRITE redis commands', :clean_gitlab_redis_sessions, :clean_gitlab_redis_shared_state do
context 'with WRITE redis commands' do
let_it_be(:key1) { "redis:{1}:key_a" }
let_it_be(:key2) { "redis:{1}:key_b" }
let_it_be(:value1) { "redis_value1"}
......@@ -223,20 +246,19 @@ RSpec.describe Gitlab::Redis::MultiStore do
end
before do
Gitlab::Redis::Sessions.with do |redis|
redis.multi do |multi|
primary_store.flushdb
secondary_store.flushdb
primary_store.multi do |multi|
multi.set(key2, value1)
multi.sadd(skey, value1)
end
end
Gitlab::Redis::SharedState.with do |redis|
redis.multi do |multi|
secondary_store.multi do |multi|
multi.set(key2, value1)
multi.sadd(skey, value1)
end
end
end
RSpec.shared_examples_for 'verify that store contains values' do |store|
it "#{store} redis store contains correct values", :aggregate_errors do
......@@ -296,14 +318,14 @@ RSpec.describe Gitlab::Redis::MultiStore do
include_examples 'verify that store contains values', :secondary_store
end
context 'when the command is executed within pipelined block', :aggregate_errors do
context 'when the command is executed within pipelined block' do
subject do
multi_store.pipelined do
multi_store.send(name, *args)
end
end
it 'is executed only 1 time on each instance' do
it 'is executed only 1 time on each instance', :aggregate_errors do
expect(primary_store).to receive(name).with(*expected_args).once
expect(secondary_store).to receive(name).with(*expected_args).once
......@@ -333,8 +355,13 @@ RSpec.describe Gitlab::Redis::MultiStore do
end
end
context 'with unsupported command', :clean_gitlab_redis_shared_state, :clean_gitlab_redis_sessions do
let_it_be(:key) { "redis:{1}:key_a" }
context 'with unsupported command' do
before do
primary_store.flushdb
secondary_store.flushdb
end
let_it_be(:key) { "redis:counter" }
subject do
multi_store.incr(key)
......@@ -360,12 +387,38 @@ RSpec.describe Gitlab::Redis::MultiStore do
it 'fallback and executes only on the secondary store', :aggregate_errors do
expect(secondary_store).to receive(:incr).with(key).and_call_original
expect(secondary_store).not_to receive(:incr)
expect(primary_store).not_to receive(:incr)
subject
end
it 'correct value is stored on the secondary store', :aggregate_errors do
subject
expect(primary_store.get(key)).to be_nil
expect(secondary_store.get(key)).to eq('1')
end
context 'when the command is executed within pipelined block' do
subject do
multi_store.pipelined do
multi_store.incr(key)
end
end
it 'is executed only 1 time on each instance', :aggregate_errors do
expect(primary_store).to receive(:incr).with(key).once
expect(secondary_store).to receive(:incr).with(key).once
subject
end
it "both redis stores are containing correct values", :aggregate_errors do
subject
expect(primary_store.get(key)).to eq('1')
expect(secondary_store.get(key)).to eq('1')
end
end
end
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