Commit 3e1462d2 authored by nmilojevic1's avatar nmilojevic1 Committed by Nikola Milojevic

Improve multi_store implementation

- Remove use_primary_store ff
- Fix specs for smembers
- Add support for the feature flags
- Check all write and read commands
- Fixed kwargs arguments deprecation error
- Add error handling and logging.
parent 7e778e8f
---
name: use_multi_store
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/73660
rollout_issue_url:
milestone: '14.5'
type: development
group: group::memory
default_enabled: false
...@@ -3,8 +3,22 @@ ...@@ -3,8 +3,22 @@
module Gitlab module Gitlab
module Redis module Redis
class MultiStore class MultiStore
class MultiReadError < StandardError
def message
'Value not found, falling back to read from the redis secondary store.'
end
end
class MethodMissingError < StandardError
def message
'Method missing. Falling back to execute method on the redis secondary_store.'
end
end
attr_reader :primary_store, :secondary_store attr_reader :primary_store, :secondary_store
FAILED_TO_READ_ERROR_MESSAGE = 'Failed to read from the redis primary_store.'
FAILED_TO_WRITE_ERROR_MESSAGE = 'Failed to write to the redis primary_store.'
READ_COMMANDS = %i( READ_COMMANDS = %i(
get get
mget mget
...@@ -12,6 +26,7 @@ module Gitlab ...@@ -12,6 +26,7 @@ module Gitlab
).freeze ).freeze
WRITE_COMMANDS = %i( WRITE_COMMANDS = %i(
set
setnx setnx
setex setex
sadd sadd
...@@ -35,60 +50,88 @@ module Gitlab ...@@ -35,60 +50,88 @@ module Gitlab
super(klass) super(klass)
end end
# TODO: Add Feature flags, by default read only from the secondary store,
# by enabling the FF, read from the primary, and fallback to read from the secondary.
READ_COMMANDS.each do |name| READ_COMMANDS.each do |name|
define_method(name) do |*args, **kwargs, &block| define_method(name) do |*args, &block|
if @instance read_command(name, *args, &block)
send_command(@instance, name, *args, **kwargs, &block)
else
value = send_command(primary_store, name, *args, **kwargs, &block)
# TODO: Add logger to detect for which key we fallback to shared_steate_store
value ||= send_command(secondary_store, name, *args, **kwargs, &block)
value
end
end end
end end
# TODO: Add proper error handling if primary store fails, ensuring that we execute at least on secondary store
# TODO: Add Feature flags, by default write only on the secondary store (SharedState), by enabling the FF, write to the primary as well.
WRITE_COMMANDS.each do |name| WRITE_COMMANDS.each do |name|
define_method(name) do |*args, **kwargs, &block| define_method(name) do |*args, &block|
if @instance write_command(name, *args, &block)
send_command(@instance, name, *args, **kwargs, &block)
else
send_command(primary_store, name, *args, **kwargs, &block)
send_command(secondary_store, name, *args, **kwargs, &block)
end
end end
end end
# TEST: try to avoid Deprecation Toolkit issues private
# See the pipelines of the POC for the example
# We call set there https://github.com/redis-store/redis-rack/blob/v2.1.3/lib/rack/session/redis.rb#L49 def read_command(command_name, *args, &block)
# With the meta-definition like there, we have hash <-> kwargs Ruby 2.7 issues as the downstream is defined like: instance = check_redis_store_instance
# https://github.com/redis/redis-rb/blob/master/lib/redis.rb#L846
def set(...) if instance
if @instance send_command(instance, command_name, *args, &block)
send_command(@instance, :set, ...)
else else
send_command(primary_store, :set, ...) read_one_with_fallback(command_name, *args, &block)
send_command(secondary_store, :set, ...)
end end
end end
private def write_command(command_name, *args, &block)
instance = check_redis_store_instance
def send_command(redis_instance, name, *args, **kwargs, &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)
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)
value = send_command(secondary_store, command_name, *args, &block)
end
value
end
def write_both(command_name, *args, &block)
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
send_command(secondary_store, command_name, *args, &block)
end
def multi_store_enabled?
Feature.enabled?(:use_multi_store, default_enabled: :yaml)
end
# rubocop:disable GitlabSecurity/PublicSend
def send_command(redis_instance, command_name, *args, &block)
if block_given? if block_given?
redis_instance.send(name, *args, **kwargs) do |*args| # rubocop:disable GitlabSecurity/PublicSend # Make sure that block is wrapped and executed only on the redis instance that is executing the block
redis_instance.send(command_name, *args) do |*args|
with_instance(redis_instance, *args, &block) with_instance(redis_instance, *args, &block)
end end
else else
redis_instance.send(name, *args, **kwargs) # rubocop:disable GitlabSecurity/PublicSend redis_instance.send(command_name, *args)
end end
end end
# rubocop:enable GitlabSecurity/PublicSend
def with_instance(instance, *args) def with_instance(instance, *args)
@instance = instance @instance = instance
...@@ -97,12 +140,31 @@ module Gitlab ...@@ -97,12 +140,31 @@ module Gitlab
@instance = nil @instance = nil
end end
def method_missing(...) def increment_read_fallback_count(command_name)
# TODO: Add logger here to log for which key and command we did fallback to the shared_state_store @read_fallback_counter ||= Gitlab::Metrics.counter(:gitlab_redis_multi_store_read_fallback_total, 'Client side Redis MultiStore reading fallback')
secondary_store.send(...) # rubocop:disable GitlabSecurity/PublicSend @read_fallback_counter.increment(command: command_name)
end
def increment_method_missing_count(command_name)
@method_missing_counter ||= Gitlab::Metrics.counter(:gitlab_redis_multi_store_method_missing_total, 'Client side Redis MultiStore method missing')
@method_missing_counter.increment(command: command_name)
end
def log_error(exception, command_name, extra = {})
Gitlab::ErrorTracking.log_exception(
exception,
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 end
def respond_to_missing?(method_name, include_private = false) def respond_to_missing?(command_name, include_private = false)
true true
end end
end end
......
This diff is collapsed.
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