Files
argo-cd/util/cache/appstate/cache.go
2025-06-17 11:46:50 -04:00

132 lines
4.1 KiB
Go

package appstate
import (
"context"
"fmt"
"sort"
"time"
"github.com/spf13/cobra"
appv1 "github.com/argoproj/argo-cd/v3/pkg/apis/application/v1alpha1"
cacheutil "github.com/argoproj/argo-cd/v3/util/cache"
"github.com/argoproj/argo-cd/v3/util/env"
)
var (
ErrCacheMiss = cacheutil.ErrCacheMiss
treeShardSize = env.ParseInt64FromEnv("ARGOCD_APPLICATION_TREE_SHARD_SIZE", 0, 0, 1000)
)
const (
clusterInfoCacheExpiration = 10 * time.Minute
)
type Cache struct {
Cache *cacheutil.Cache
appStateCacheExpiration time.Duration
}
func NewCache(cache *cacheutil.Cache, appStateCacheExpiration time.Duration) *Cache {
return &Cache{cache, appStateCacheExpiration}
}
func AddCacheFlagsToCmd(cmd *cobra.Command, opts ...cacheutil.Options) func() (*Cache, error) {
var appStateCacheExpiration time.Duration
cmd.Flags().DurationVar(&appStateCacheExpiration, "app-state-cache-expiration", env.ParseDurationFromEnv("ARGOCD_APP_STATE_CACHE_EXPIRATION", 1*time.Hour, 0, 10*time.Hour), "Cache expiration for app state")
cacheFactory := cacheutil.AddCacheFlagsToCmd(cmd, opts...)
return func() (*Cache, error) {
cache, err := cacheFactory()
if err != nil {
return nil, err
}
return NewCache(cache, appStateCacheExpiration), nil
}
}
func (c *Cache) GetItem(key string, item any) error {
return c.Cache.GetItem(key, item)
}
func (c *Cache) SetItem(key string, item any, expiration time.Duration, deletion bool) error {
return c.Cache.SetItem(key, item, &cacheutil.CacheActionOpts{Expiration: expiration, Delete: deletion})
}
func appManagedResourcesKey(appName string) string {
return "app|managed-resources|" + appName
}
func (c *Cache) GetAppManagedResources(appName string, res *[]*appv1.ResourceDiff) error {
err := c.GetItem(appManagedResourcesKey(appName), &res)
return err
}
func (c *Cache) SetAppManagedResources(appName string, managedResources []*appv1.ResourceDiff) error {
sort.Slice(managedResources, func(i, j int) bool {
return managedResources[i].FullName() < managedResources[j].FullName()
})
return c.SetItem(appManagedResourcesKey(appName), managedResources, c.appStateCacheExpiration, managedResources == nil)
}
func appResourcesTreeKey(appName string, shard int64) string {
key := "app|resources-tree|" + appName
if shard > 0 {
key = fmt.Sprintf("%s|%d", key, shard)
}
return key
}
func clusterInfoKey(server string) string {
return "cluster|info|" + server
}
func (c *Cache) GetAppResourcesTree(appName string, res *appv1.ApplicationTree) error {
err := c.GetItem(appResourcesTreeKey(appName, 0), &res)
if res.ShardsCount > 1 {
for i := int64(1); i < res.ShardsCount; i++ {
var shard appv1.ApplicationTree
err = c.GetItem(appResourcesTreeKey(appName, i), &shard)
if err != nil {
return err
}
res.Merge(&shard)
}
}
return err
}
func (c *Cache) OnAppResourcesTreeChanged(ctx context.Context, appName string, callback func() error) error {
return c.Cache.OnUpdated(ctx, appManagedResourcesKey(appName), callback)
}
func (c *Cache) SetAppResourcesTree(appName string, resourcesTree *appv1.ApplicationTree) error {
if resourcesTree == nil {
if err := c.SetItem(appResourcesTreeKey(appName, 0), resourcesTree, c.appStateCacheExpiration, true); err != nil {
return err
}
} else {
// Splitting resource tree into shards reduces number of Redis SET calls and therefore amount of traffic sent
// from controller to Redis. Controller still stores each shard in cache but util/cache/twolevelclient.go
// forwards request to Redis only if shard actually changes.
for i, shard := range resourcesTree.GetShards(treeShardSize) {
if err := c.SetItem(appResourcesTreeKey(appName, int64(i)), shard, c.appStateCacheExpiration, false); err != nil {
return err
}
}
}
return c.Cache.NotifyUpdated(appManagedResourcesKey(appName))
}
func (c *Cache) SetClusterInfo(server string, info *appv1.ClusterInfo) error {
return c.SetItem(clusterInfoKey(server), info, clusterInfoCacheExpiration, info == nil)
}
func (c *Cache) GetClusterInfo(server string, res *appv1.ClusterInfo) error {
err := c.GetItem(clusterInfoKey(server), &res)
return err
}