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>
168 lines
6.0 KiB
Go
168 lines
6.0 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package localcachelayer
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/mattermost/mattermost/server/public/model"
|
|
"github.com/mattermost/mattermost/server/public/shared/request"
|
|
"github.com/mattermost/mattermost/server/v8/channels/store"
|
|
)
|
|
|
|
type LocalCachePostStore struct {
|
|
store.PostStore
|
|
rootStore *LocalCacheStore
|
|
}
|
|
|
|
func (s *LocalCachePostStore) handleClusterInvalidateLastPostTime(msg *model.ClusterMessage) {
|
|
if bytes.Equal(msg.Data, clearCacheMessageData) {
|
|
s.rootStore.lastPostTimeCache.Purge()
|
|
} else {
|
|
s.rootStore.lastPostTimeCache.Remove(string(msg.Data))
|
|
}
|
|
}
|
|
|
|
func (s *LocalCachePostStore) handleClusterInvalidateLastPosts(msg *model.ClusterMessage) {
|
|
if bytes.Equal(msg.Data, clearCacheMessageData) {
|
|
s.rootStore.postLastPostsCache.Purge()
|
|
} else {
|
|
s.rootStore.postLastPostsCache.Remove(string(msg.Data))
|
|
}
|
|
}
|
|
|
|
func (s *LocalCachePostStore) handleClusterInvalidatePostsUsage(msg *model.ClusterMessage) {
|
|
if bytes.Equal(msg.Data, clearCacheMessageData) {
|
|
s.rootStore.postsUsageCache.Purge()
|
|
} else {
|
|
s.rootStore.postsUsageCache.Remove(string(msg.Data))
|
|
}
|
|
}
|
|
|
|
func (s LocalCachePostStore) ClearCaches() {
|
|
s.rootStore.doClearCacheCluster(s.rootStore.lastPostTimeCache)
|
|
s.rootStore.doClearCacheCluster(s.rootStore.postLastPostsCache)
|
|
s.rootStore.doClearCacheCluster(s.rootStore.postsUsageCache)
|
|
s.PostStore.ClearCaches()
|
|
|
|
if s.rootStore.metrics != nil {
|
|
s.rootStore.metrics.IncrementMemCacheInvalidationCounter(s.rootStore.lastPostTimeCache.Name())
|
|
s.rootStore.metrics.IncrementMemCacheInvalidationCounter(s.rootStore.postLastPostsCache.Name())
|
|
s.rootStore.metrics.IncrementMemCacheInvalidationCounter(s.rootStore.postsUsageCache.Name())
|
|
}
|
|
}
|
|
|
|
func (s LocalCachePostStore) InvalidateLastPostTimeCache(channelId string) {
|
|
s.rootStore.doInvalidateCacheCluster(s.rootStore.lastPostTimeCache, channelId, nil)
|
|
|
|
// Keys are "{channelid}{limit}" and caching only occurs on limits of 30 and 60
|
|
s.rootStore.doInvalidateCacheCluster(s.rootStore.postLastPostsCache, channelId+"30", nil)
|
|
s.rootStore.doInvalidateCacheCluster(s.rootStore.postLastPostsCache, channelId+"60", nil)
|
|
|
|
s.PostStore.InvalidateLastPostTimeCache(channelId)
|
|
|
|
if s.rootStore.metrics != nil {
|
|
s.rootStore.metrics.IncrementMemCacheInvalidationCounter(s.rootStore.lastPostTimeCache.Name())
|
|
s.rootStore.metrics.IncrementMemCacheInvalidationCounter(s.rootStore.postLastPostsCache.Name())
|
|
}
|
|
}
|
|
|
|
func (s LocalCachePostStore) GetEtag(channelId string, allowFromCache, collapsedThreads bool) string {
|
|
if allowFromCache {
|
|
var lastTime int64
|
|
if err := s.rootStore.doStandardReadCache(s.rootStore.lastPostTimeCache, channelId, &lastTime); err == nil {
|
|
return fmt.Sprintf("%v.%v", model.CurrentVersion, lastTime)
|
|
}
|
|
}
|
|
|
|
result := s.PostStore.GetEtag(channelId, allowFromCache, collapsedThreads)
|
|
|
|
splittedResult := strings.Split(result, ".")
|
|
|
|
lastTime, _ := strconv.ParseInt((splittedResult[len(splittedResult)-1]), 10, 64)
|
|
|
|
s.rootStore.doStandardAddToCache(s.rootStore.lastPostTimeCache, channelId, lastTime)
|
|
|
|
return result
|
|
}
|
|
|
|
func (s LocalCachePostStore) GetPostsSince(rctx request.CTX, options model.GetPostsSinceOptions, allowFromCache bool, sanitizeOptions map[string]bool) (*model.PostList, error) {
|
|
if allowFromCache {
|
|
// If the last post in the channel's time is less than or equal to the time we are getting posts since,
|
|
// we can safely return no posts.
|
|
var lastTime int64
|
|
if err := s.rootStore.doStandardReadCache(s.rootStore.lastPostTimeCache, options.ChannelId, &lastTime); err == nil && lastTime <= options.Time {
|
|
list := model.NewPostList()
|
|
return list, nil
|
|
}
|
|
}
|
|
|
|
list, err := s.PostStore.GetPostsSince(rctx, options, allowFromCache, sanitizeOptions)
|
|
|
|
latestUpdate := options.Time
|
|
if err == nil {
|
|
for _, p := range list.ToSlice() {
|
|
if latestUpdate < p.UpdateAt {
|
|
latestUpdate = p.UpdateAt
|
|
}
|
|
}
|
|
s.rootStore.doStandardAddToCache(s.rootStore.lastPostTimeCache, options.ChannelId, latestUpdate)
|
|
}
|
|
|
|
return list, err
|
|
}
|
|
|
|
func (s LocalCachePostStore) GetPosts(rctx request.CTX, options model.GetPostsOptions, allowFromCache bool, sanitizeOptions map[string]bool) (*model.PostList, error) {
|
|
if !allowFromCache {
|
|
return s.PostStore.GetPosts(rctx, options, allowFromCache, sanitizeOptions)
|
|
}
|
|
|
|
offset := options.PerPage * options.Page
|
|
// Caching only occurs on limits of 30 and 60, the common limits requested by MM clients
|
|
if offset == 0 && (options.PerPage == 60 || options.PerPage == 30) {
|
|
var cacheItem *model.PostList
|
|
if err := s.rootStore.doStandardReadCache(s.rootStore.postLastPostsCache, fmt.Sprintf("%s%v", options.ChannelId, options.PerPage), &cacheItem); err == nil {
|
|
return cacheItem, nil
|
|
}
|
|
}
|
|
|
|
list, err := s.PostStore.GetPosts(rctx, options, false, sanitizeOptions)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Caching only occurs on limits of 30 and 60, the common limits requested by MM clients
|
|
if offset == 0 && (options.PerPage == 60 || options.PerPage == 30) {
|
|
s.rootStore.doStandardAddToCache(s.rootStore.postLastPostsCache, fmt.Sprintf("%s%v", options.ChannelId, options.PerPage), list)
|
|
}
|
|
|
|
return list, err
|
|
}
|
|
|
|
// AnalyticsPostCount looks up cache only when ExcludeDeleted and UsersPostsOnly are true and rest are falsy.
|
|
func (s LocalCachePostStore) AnalyticsPostCount(options *model.PostCountOptions) (int64, error) {
|
|
if !options.AllowFromCache || options.MustHaveFile || options.MustHaveHashtag || !options.UsersPostsOnly || !options.ExcludeDeleted || options.TeamId != "" {
|
|
return s.PostStore.AnalyticsPostCount(options)
|
|
}
|
|
|
|
// Currently cache only for app > usage > GetPostsUsage()
|
|
// Other filter combinations can be cached if required
|
|
cacheKey := "posts_usage"
|
|
var count int64
|
|
if err := s.rootStore.doStandardReadCache(s.rootStore.postsUsageCache, cacheKey, &count); err == nil {
|
|
return count, nil
|
|
}
|
|
|
|
count, err := s.PostStore.AnalyticsPostCount(options)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
s.rootStore.doStandardAddToCache(s.rootStore.postsUsageCache, cacheKey, count)
|
|
return count, nil
|
|
}
|