Full Mattermost server source with integrated Community Enterprise features. Includes vendor directory for offline/air-gapped builds. Structure: - enterprise-impl/: Enterprise feature implementations - enterprise-community/: Init files that register implementations - enterprise/: Bridge imports (community_imports.go) - vendor/: All dependencies for offline builds Build (online): go build ./cmd/mattermost Build (offline/air-gapped): go build -mod=vendor ./cmd/mattermost 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
147 lines
4.2 KiB
Go
147 lines
4.2 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package cache
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/mattermost/mattermost/server/public/model"
|
|
"github.com/mattermost/mattermost/server/v8/einterfaces"
|
|
"github.com/redis/rueidis"
|
|
)
|
|
|
|
// CacheOptions contains options for initializing a cache
|
|
type CacheOptions struct {
|
|
Size int
|
|
DefaultExpiry time.Duration
|
|
Name string
|
|
InvalidateClusterEvent model.ClusterEvent
|
|
Striped bool
|
|
// StripedBuckets is used only by LRUStriped and shouldn't be greater than the number
|
|
// of CPUs available on the machine running this cache.
|
|
StripedBuckets int
|
|
}
|
|
|
|
// Provider is a provider for Cache
|
|
type Provider interface {
|
|
// NewCache creates a new cache with given options.
|
|
NewCache(opts *CacheOptions) (Cache, error)
|
|
// Connect opens a new connection to the cache using specific provider parameters.
|
|
// The returned string contains the status of the response from the cache backend.
|
|
Connect() (string, error)
|
|
// SetMetrics
|
|
SetMetrics(metrics einterfaces.MetricsInterface)
|
|
// Close releases any resources used by the cache provider.
|
|
Close() error
|
|
// Type returns what type of cache it generates.
|
|
Type() string
|
|
}
|
|
|
|
type cacheProvider struct {
|
|
}
|
|
|
|
// NewProvider creates a new CacheProvider
|
|
func NewProvider() Provider {
|
|
return &cacheProvider{}
|
|
}
|
|
|
|
// NewCache creates a new cache with given opts
|
|
func (c *cacheProvider) NewCache(opts *CacheOptions) (Cache, error) {
|
|
if opts.Striped {
|
|
return NewLRUStriped(opts)
|
|
}
|
|
return NewLRU(opts), nil
|
|
}
|
|
|
|
// Connect opens a new connection to the cache using specific provider parameters.
|
|
func (c *cacheProvider) Connect() (string, error) {
|
|
return "OK", nil
|
|
}
|
|
|
|
func (c *cacheProvider) SetMetrics(metrics einterfaces.MetricsInterface) {
|
|
}
|
|
|
|
// Close releases any resources used by the cache provider.
|
|
func (c *cacheProvider) Close() error {
|
|
return nil
|
|
}
|
|
|
|
func (c *cacheProvider) Type() string {
|
|
return model.CacheTypeLRU
|
|
}
|
|
|
|
type redisProvider struct {
|
|
client rueidis.Client
|
|
cachePrefix string
|
|
metrics einterfaces.MetricsInterface
|
|
}
|
|
|
|
type RedisOptions struct {
|
|
RedisAddr string
|
|
RedisPassword string
|
|
RedisDB int
|
|
RedisCachePrefix string
|
|
DisableCache bool
|
|
}
|
|
|
|
// NewProvider creates a new CacheProvider
|
|
func NewRedisProvider(opts *RedisOptions) (Provider, error) {
|
|
client, err := rueidis.NewClient(rueidis.ClientOption{
|
|
InitAddress: []string{opts.RedisAddr},
|
|
Password: opts.RedisPassword,
|
|
SelectDB: opts.RedisDB,
|
|
ForceSingleClient: true,
|
|
CacheSizeEachConn: 32 * (1 << 20), // 32MiB local cache size
|
|
DisableCache: opts.DisableCache,
|
|
// This is used to collect more commands before flushing to Redis.
|
|
// This increases latency at the cost of lower CPU usage at Redis.
|
|
// It's a tradeoff we are willing to make because Redis is only
|
|
// meant to be used at very high scales. The docs suggest 20us,
|
|
// but going as high as 250us doesn't make any material difference.
|
|
MaxFlushDelay: 250 * time.Microsecond,
|
|
DisableRetry: true,
|
|
// The default is 10s, which is a bit too high
|
|
ConnWriteTimeout: 5 * time.Second,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &redisProvider{client: client, cachePrefix: opts.RedisCachePrefix}, nil
|
|
}
|
|
|
|
// NewCache creates a new cache with given opts
|
|
func (r *redisProvider) NewCache(opts *CacheOptions) (Cache, error) {
|
|
if r.cachePrefix != "" {
|
|
opts.Name = r.cachePrefix + ":" + opts.Name
|
|
}
|
|
rr, err := NewRedis(opts, r.client)
|
|
rr.metrics = r.metrics
|
|
return rr, err
|
|
}
|
|
|
|
// Connect opens a new connection to the cache using specific provider parameters.
|
|
func (r *redisProvider) Connect() (string, error) {
|
|
res, err := r.client.Do(context.Background(), r.client.B().Ping().Build()).ToString()
|
|
if err != nil {
|
|
return "", fmt.Errorf("unable to establish connection with redis: %v", err)
|
|
}
|
|
return res, nil
|
|
}
|
|
|
|
func (r *redisProvider) SetMetrics(metrics einterfaces.MetricsInterface) {
|
|
r.metrics = metrics
|
|
}
|
|
|
|
func (r *redisProvider) Type() string {
|
|
return model.CacheTypeRedis
|
|
}
|
|
|
|
// Close releases any resources used by the cache provider.
|
|
func (r *redisProvider) Close() error {
|
|
r.client.Close()
|
|
return nil
|
|
}
|