Commit 4cd6c91d authored by Nick Thomas's avatar Nick Thomas

Redis set cache docs and minor cleanup

parent 07323f44
...@@ -35,6 +35,11 @@ module Gitlab ...@@ -35,6 +35,11 @@ module Gitlab
# name - The name of the method to be cached. # name - The name of the method to be cached.
# fallback - A value to fall back to if the repository does not exist, or # fallback - A value to fall back to if the repository does not exist, or
# in case of a Git error. Defaults to nil. # in case of a Git error. Defaults to nil.
#
# It is not safe to use this method prior to the release of 12.3, since
# 12.2 does not correctly invalidate the redis set cache value. A mixed
# code environment containing both 12.2 and 12.3 nodes breaks, while a
# mixed code environment containing both 12.3 and 12.4 nodes will work.
def cache_method_as_redis_set(name, fallback: nil) def cache_method_as_redis_set(name, fallback: nil)
uncached_name = alias_uncached_method(name) uncached_name = alias_uncached_method(name)
...@@ -44,13 +49,20 @@ module Gitlab ...@@ -44,13 +49,20 @@ module Gitlab
end end
end end
# Attempt to determine whether a value is in the set of values being
# cached, by performing a redis SISMEMBERS query if appropriate.
#
# If the full list is already in-memory, we're better using it directly.
#
# If the cache is not yet populated, querying it directly will give the
# wrong answer. We handle that by querying the full list - which fills
# the cache - and using it directly to answer the question.
define_method("#{name}_include?") do |value| define_method("#{name}_include?") do |value|
# If the cache isn't populated, we can't rely on it if strong_memoized?(name) || !redis_set_cache.exist?(name)
return redis_set_cache.include?(name, value) if redis_set_cache.exist?(name) return __send__(name).include?(value) # rubocop:disable GitlabSecurity/PublicSend
end
# Since we have to pull all branch names to populate the cache, use redis_set_cache.include?(name, value)
# the data we already have to answer the query just this once
__send__(name).include?(value) # rubocop:disable GitlabSecurity/PublicSend
end end
end end
......
...@@ -13,7 +13,7 @@ module Gitlab ...@@ -13,7 +13,7 @@ module Gitlab
end end
def cache_key(type) def cache_key(type)
[type, namespace, 'set'].join(':') "#{type}:#{namespace}:set"
end end
def expire(key) def expire(key)
...@@ -37,7 +37,7 @@ module Gitlab ...@@ -37,7 +37,7 @@ module Gitlab
# Splitting into groups of 1000 prevents us from creating a too-long # Splitting into groups of 1000 prevents us from creating a too-long
# Redis command # Redis command
value.in_groups_of(1000, false) { |subset| redis.sadd(full_key, subset) } value.each_slice(1000) { |subset| redis.sadd(full_key, subset) }
redis.expire(full_key, expires_in) redis.expire(full_key, expires_in)
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