mirror of
https://github.com/zitadel/zitadel.git
synced 2025-01-10 13:13:46 +00:00
250f2344c8
# Which Problems Are Solved Add a cache implementation using Redis single mode. This does not add support for Redis Cluster or sentinel. # How the Problems Are Solved Added the `internal/cache/redis` package. All operations occur atomically, including setting of secondary indexes, using LUA scripts where needed. The [`miniredis`](https://github.com/alicebob/miniredis) package is used to run unit tests. # Additional Changes - Move connector code to `internal/cache/connector/...` and remove duplicate code from `query` and `command` packages. - Fix a missed invalidation on the restrictions projection # Additional Context Closes #8130
173 lines
4.7 KiB
Go
173 lines
4.7 KiB
Go
package redis
|
|
|
|
import (
|
|
"context"
|
|
_ "embed"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"log/slog"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/redis/go-redis/v9"
|
|
|
|
"github.com/zitadel/zitadel/internal/cache"
|
|
"github.com/zitadel/zitadel/internal/telemetry/tracing"
|
|
)
|
|
|
|
var (
|
|
//go:embed _select.lua
|
|
selectComponent string
|
|
//go:embed _util.lua
|
|
utilComponent string
|
|
//go:embed _remove.lua
|
|
removeComponent string
|
|
//go:embed set.lua
|
|
setScript string
|
|
//go:embed get.lua
|
|
getScript string
|
|
//go:embed invalidate.lua
|
|
invalidateScript string
|
|
|
|
// Don't mind the creative "import"
|
|
setParsed = redis.NewScript(strings.Join([]string{selectComponent, utilComponent, setScript}, "\n"))
|
|
getParsed = redis.NewScript(strings.Join([]string{selectComponent, utilComponent, removeComponent, getScript}, "\n"))
|
|
invalidateParsed = redis.NewScript(strings.Join([]string{selectComponent, utilComponent, removeComponent, invalidateScript}, "\n"))
|
|
)
|
|
|
|
type redisCache[I, K comparable, V cache.Entry[I, K]] struct {
|
|
db int
|
|
config *cache.Config
|
|
indices []I
|
|
connector *Connector
|
|
logger *slog.Logger
|
|
}
|
|
|
|
// NewCache returns a cache that stores and retrieves object using single Redis.
|
|
func NewCache[I, K comparable, V cache.Entry[I, K]](config cache.Config, client *Connector, db int, indices []I) cache.Cache[I, K, V] {
|
|
return &redisCache[I, K, V]{
|
|
config: &config,
|
|
db: db,
|
|
indices: indices,
|
|
connector: client,
|
|
logger: config.Log.Slog(),
|
|
}
|
|
}
|
|
|
|
func (c *redisCache[I, K, V]) Set(ctx context.Context, value V) {
|
|
if _, err := c.set(ctx, value); err != nil {
|
|
c.logger.ErrorContext(ctx, "redis cache set", "err", err)
|
|
}
|
|
}
|
|
|
|
func (c *redisCache[I, K, V]) set(ctx context.Context, value V) (objectID string, err error) {
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
defer func() { span.EndWithError(err) }()
|
|
|
|
// Internal ID used for the object
|
|
objectID = uuid.NewString()
|
|
keys := []string{objectID}
|
|
// flatten the secondary keys
|
|
for _, index := range c.indices {
|
|
keys = append(keys, c.redisIndexKeys(index, value.Keys(index)...)...)
|
|
}
|
|
var buf strings.Builder
|
|
err = json.NewEncoder(&buf).Encode(value)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
err = setParsed.Run(ctx, c.connector, keys,
|
|
c.db, // DB namespace
|
|
buf.String(), // object
|
|
int64(c.config.LastUseAge/time.Second), // usage_lifetime
|
|
int64(c.config.MaxAge/time.Second), // max_age,
|
|
).Err()
|
|
// redis.Nil is always returned because the script doesn't have a return value.
|
|
if err != nil && !errors.Is(err, redis.Nil) {
|
|
return "", err
|
|
}
|
|
return objectID, nil
|
|
}
|
|
|
|
func (c *redisCache[I, K, V]) Get(ctx context.Context, index I, key K) (value V, ok bool) {
|
|
var (
|
|
obj any
|
|
err error
|
|
)
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
defer func() {
|
|
if errors.Is(err, redis.Nil) {
|
|
err = nil
|
|
}
|
|
span.EndWithError(err)
|
|
}()
|
|
|
|
logger := c.logger.With("index", index, "key", key)
|
|
obj, err = getParsed.Run(ctx, c.connector, c.redisIndexKeys(index, key), c.db).Result()
|
|
if err != nil && !errors.Is(err, redis.Nil) {
|
|
logger.ErrorContext(ctx, "redis cache get", "err", err)
|
|
return value, false
|
|
}
|
|
data, ok := obj.(string)
|
|
if !ok {
|
|
logger.With("err", err).InfoContext(ctx, "redis cache miss")
|
|
return value, false
|
|
}
|
|
err = json.NewDecoder(strings.NewReader(data)).Decode(&value)
|
|
if err != nil {
|
|
logger.ErrorContext(ctx, "redis cache get", "err", fmt.Errorf("decode: %w", err))
|
|
return value, false
|
|
}
|
|
return value, true
|
|
}
|
|
|
|
func (c *redisCache[I, K, V]) Invalidate(ctx context.Context, index I, key ...K) (err error) {
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
defer func() { span.EndWithError(err) }()
|
|
|
|
if len(key) == 0 {
|
|
return nil
|
|
}
|
|
err = invalidateParsed.Run(ctx, c.connector, c.redisIndexKeys(index, key...), c.db).Err()
|
|
// redis.Nil is always returned because the script doesn't have a return value.
|
|
if err != nil && !errors.Is(err, redis.Nil) {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *redisCache[I, K, V]) Delete(ctx context.Context, index I, key ...K) (err error) {
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
defer func() { span.EndWithError(err) }()
|
|
|
|
if len(key) == 0 {
|
|
return nil
|
|
}
|
|
pipe := c.connector.Pipeline()
|
|
pipe.Select(ctx, c.db)
|
|
pipe.Del(ctx, c.redisIndexKeys(index, key...)...)
|
|
_, err = pipe.Exec(ctx)
|
|
return err
|
|
}
|
|
|
|
func (c *redisCache[I, K, V]) Truncate(ctx context.Context) (err error) {
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
defer func() { span.EndWithError(err) }()
|
|
|
|
pipe := c.connector.Pipeline()
|
|
pipe.Select(ctx, c.db)
|
|
pipe.FlushDB(ctx)
|
|
_, err = pipe.Exec(ctx)
|
|
return err
|
|
}
|
|
|
|
func (c *redisCache[I, K, V]) redisIndexKeys(index I, keys ...K) []string {
|
|
out := make([]string, len(keys))
|
|
for i, k := range keys {
|
|
out[i] = fmt.Sprintf("%v:%v", index, k)
|
|
}
|
|
return out
|
|
}
|