mirror of
https://github.com/opencloud-eu/opencloud.git
synced 2026-01-25 06:18:42 -05:00
Bumps [github.com/open-policy-agent/opa](https://github.com/open-policy-agent/opa) from 0.51.0 to 0.59.0. - [Release notes](https://github.com/open-policy-agent/opa/releases) - [Changelog](https://github.com/open-policy-agent/opa/blob/main/CHANGELOG.md) - [Commits](https://github.com/open-policy-agent/opa/compare/v0.51.0...v0.59.0) --- updated-dependencies: - dependency-name: github.com/open-policy-agent/opa dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com>
194 lines
4.5 KiB
Go
194 lines
4.5 KiB
Go
// Copyright 2020 The OPA Authors. All rights reserved.
|
|
// Use of this source code is governed by an Apache2
|
|
// license that can be found in the LICENSE file.
|
|
|
|
// Package cache defines the inter-query cache interface that can cache data across queries
|
|
package cache
|
|
|
|
import (
|
|
"container/list"
|
|
"sync"
|
|
|
|
"github.com/open-policy-agent/opa/ast"
|
|
|
|
"github.com/open-policy-agent/opa/util"
|
|
)
|
|
|
|
const (
|
|
defaultMaxSizeBytes = int64(0) // unlimited
|
|
)
|
|
|
|
// Config represents the configuration of the inter-query cache.
|
|
type Config struct {
|
|
InterQueryBuiltinCache InterQueryBuiltinCacheConfig `json:"inter_query_builtin_cache"`
|
|
}
|
|
|
|
// InterQueryBuiltinCacheConfig represents the configuration of the inter-query cache that built-in functions can utilize.
|
|
type InterQueryBuiltinCacheConfig struct {
|
|
MaxSizeBytes *int64 `json:"max_size_bytes,omitempty"`
|
|
}
|
|
|
|
// ParseCachingConfig returns the config for the inter-query cache.
|
|
func ParseCachingConfig(raw []byte) (*Config, error) {
|
|
if raw == nil {
|
|
maxSize := new(int64)
|
|
*maxSize = defaultMaxSizeBytes
|
|
return &Config{InterQueryBuiltinCache: InterQueryBuiltinCacheConfig{MaxSizeBytes: maxSize}}, nil
|
|
}
|
|
|
|
var config Config
|
|
|
|
if err := util.Unmarshal(raw, &config); err == nil {
|
|
if err = config.validateAndInjectDefaults(); err != nil {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
return nil, err
|
|
}
|
|
|
|
return &config, nil
|
|
}
|
|
|
|
func (c *Config) validateAndInjectDefaults() error {
|
|
if c.InterQueryBuiltinCache.MaxSizeBytes == nil {
|
|
maxSize := new(int64)
|
|
*maxSize = defaultMaxSizeBytes
|
|
c.InterQueryBuiltinCache.MaxSizeBytes = maxSize
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// InterQueryCacheValue defines the interface for the data that the inter-query cache holds.
|
|
type InterQueryCacheValue interface {
|
|
SizeInBytes() int64
|
|
Clone() (InterQueryCacheValue, error)
|
|
}
|
|
|
|
// InterQueryCache defines the interface for the inter-query cache.
|
|
type InterQueryCache interface {
|
|
Get(key ast.Value) (value InterQueryCacheValue, found bool)
|
|
Insert(key ast.Value, value InterQueryCacheValue) int
|
|
Delete(key ast.Value)
|
|
UpdateConfig(config *Config)
|
|
Clone(value InterQueryCacheValue) (InterQueryCacheValue, error)
|
|
}
|
|
|
|
// NewInterQueryCache returns a new inter-query cache.
|
|
func NewInterQueryCache(config *Config) InterQueryCache {
|
|
return &cache{
|
|
items: map[string]cacheItem{},
|
|
usage: 0,
|
|
config: config,
|
|
l: list.New(),
|
|
}
|
|
}
|
|
|
|
type cacheItem struct {
|
|
value InterQueryCacheValue
|
|
keyElement *list.Element
|
|
}
|
|
|
|
type cache struct {
|
|
items map[string]cacheItem
|
|
usage int64
|
|
config *Config
|
|
l *list.List
|
|
mtx sync.Mutex
|
|
}
|
|
|
|
// Insert inserts a key k into the cache with value v.
|
|
func (c *cache) Insert(k ast.Value, v InterQueryCacheValue) (dropped int) {
|
|
c.mtx.Lock()
|
|
defer c.mtx.Unlock()
|
|
return c.unsafeInsert(k, v)
|
|
}
|
|
|
|
// Get returns the value in the cache for k.
|
|
func (c *cache) Get(k ast.Value) (InterQueryCacheValue, bool) {
|
|
c.mtx.Lock()
|
|
defer c.mtx.Unlock()
|
|
cacheItem, ok := c.unsafeGet(k)
|
|
|
|
if ok {
|
|
return cacheItem.value, true
|
|
}
|
|
return nil, false
|
|
}
|
|
|
|
// Delete deletes the value in the cache for k.
|
|
func (c *cache) Delete(k ast.Value) {
|
|
c.mtx.Lock()
|
|
defer c.mtx.Unlock()
|
|
c.unsafeDelete(k)
|
|
}
|
|
|
|
func (c *cache) UpdateConfig(config *Config) {
|
|
if config == nil {
|
|
return
|
|
}
|
|
c.mtx.Lock()
|
|
defer c.mtx.Unlock()
|
|
c.config = config
|
|
}
|
|
|
|
func (c *cache) Clone(value InterQueryCacheValue) (InterQueryCacheValue, error) {
|
|
c.mtx.Lock()
|
|
defer c.mtx.Unlock()
|
|
return c.unsafeClone(value)
|
|
}
|
|
|
|
func (c *cache) unsafeInsert(k ast.Value, v InterQueryCacheValue) (dropped int) {
|
|
size := v.SizeInBytes()
|
|
limit := c.maxSizeBytes()
|
|
|
|
if limit > 0 {
|
|
if size > limit {
|
|
dropped++
|
|
return dropped
|
|
}
|
|
|
|
for key := c.l.Front(); key != nil && (c.usage+size > limit); key = c.l.Front() {
|
|
dropKey := key.Value.(ast.Value)
|
|
c.unsafeDelete(dropKey)
|
|
dropped++
|
|
}
|
|
}
|
|
|
|
// By deleting the old value, if it exists, we ensure the usage variable stays correct
|
|
c.unsafeDelete(k)
|
|
|
|
c.items[k.String()] = cacheItem{
|
|
value: v,
|
|
keyElement: c.l.PushBack(k),
|
|
}
|
|
c.usage += size
|
|
return dropped
|
|
}
|
|
|
|
func (c *cache) unsafeGet(k ast.Value) (cacheItem, bool) {
|
|
value, ok := c.items[k.String()]
|
|
return value, ok
|
|
}
|
|
|
|
func (c *cache) unsafeDelete(k ast.Value) {
|
|
cacheItem, ok := c.unsafeGet(k)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
c.usage -= cacheItem.value.SizeInBytes()
|
|
delete(c.items, k.String())
|
|
c.l.Remove(cacheItem.keyElement)
|
|
}
|
|
|
|
func (c *cache) unsafeClone(value InterQueryCacheValue) (InterQueryCacheValue, error) {
|
|
return value.Clone()
|
|
}
|
|
|
|
func (c *cache) maxSizeBytes() int64 {
|
|
if c.config == nil {
|
|
return defaultMaxSizeBytes
|
|
}
|
|
return *c.config.InterQueryBuiltinCache.MaxSizeBytes
|
|
}
|