2022-11-11 12:18:38 +01:00
package result
import (
"reflect"
"time"
"codeberg.org/gruf/go-cache/v3/ttl"
)
2022-11-13 14:02:07 +01:00
// Lookup represents a struct object lookup method in the cache.
type Lookup struct {
// Name is a period ('.') separated string
// of struct fields this Key encompasses.
Name string
// AllowZero indicates whether to accept and cache
// under zero value keys, otherwise ignore them.
AllowZero bool
}
// Cache provides a means of caching value structures, along with
// the results of attempting to load them. An example usecase of this
// cache would be in wrapping a database, allowing caching of sql.ErrNoRows.
2022-11-11 12:18:38 +01:00
type Cache [ Value any ] struct {
cache ttl . Cache [ int64 , result [ Value ] ] // underlying result cache
lookups structKeys // pre-determined struct lookups
copy func ( Value ) Value // copies a Value type
next int64 // update key counter
}
// New returns a new initialized Cache, with given lookups and underlying value copy function.
2022-11-13 14:02:07 +01:00
func New [ Value any ] ( lookups [ ] Lookup , copy func ( Value ) Value ) * Cache [ Value ] {
2022-11-11 12:18:38 +01:00
return NewSized ( lookups , copy , 64 )
}
// NewSized returns a new initialized Cache, with given lookups, underlying value copy function and provided capacity.
2022-11-13 14:02:07 +01:00
func NewSized [ Value any ] ( lookups [ ] Lookup , copy func ( Value ) Value , cap int ) * Cache [ Value ] {
2022-11-11 12:18:38 +01:00
var z Value
// Determine generic type
t := reflect . TypeOf ( z )
// Iteratively deref pointer type
for t . Kind ( ) == reflect . Pointer {
t = t . Elem ( )
}
// Ensure that this is a struct type
if t . Kind ( ) != reflect . Struct {
panic ( "generic parameter type must be struct (or ptr to)" )
}
// Allocate new cache object
c := & Cache [ Value ] { copy : copy }
2022-11-13 14:02:07 +01:00
c . lookups = make ( [ ] structKey , len ( lookups ) )
2022-11-11 12:18:38 +01:00
for i , lookup := range lookups {
// Generate keyed field info for lookup
2022-11-13 14:02:07 +01:00
c . lookups [ i ] = genStructKey ( lookup , t )
2022-11-11 12:18:38 +01:00
}
// Create and initialize underlying cache
c . cache . Init ( 0 , cap , 0 )
c . SetEvictionCallback ( nil )
c . SetInvalidateCallback ( nil )
return c
}
// Start will start the cache background eviction routine with given sweep frequency. If already
// running or a freq <= 0 provided, this is a no-op. This will block until eviction routine started.
func ( c * Cache [ Value ] ) Start ( freq time . Duration ) bool {
return c . cache . Start ( freq )
}
// Stop will stop cache background eviction routine. If not running this
// is a no-op. This will block until the eviction routine has stopped.
func ( c * Cache [ Value ] ) Stop ( ) bool {
return c . cache . Stop ( )
}
// SetTTL sets the cache item TTL. Update can be specified to force updates of existing items
// in the cache, this will simply add the change in TTL to their current expiry time.
func ( c * Cache [ Value ] ) SetTTL ( ttl time . Duration , update bool ) {
c . cache . SetTTL ( ttl , update )
}
// SetEvictionCallback sets the eviction callback to the provided hook.
func ( c * Cache [ Value ] ) SetEvictionCallback ( hook func ( Value ) ) {
if hook == nil {
// Ensure non-nil hook.
hook = func ( Value ) { }
}
c . cache . SetEvictionCallback ( func ( item * ttl . Entry [ int64 , result [ Value ] ] ) {
for _ , key := range item . Value . Keys {
// Delete key->pkey lookup
2022-11-13 14:02:07 +01:00
pkeys := key . key . pkeys
2022-11-11 12:18:38 +01:00
delete ( pkeys , key . value )
}
if item . Value . Error != nil {
// Skip error hooks
return
}
// Call user hook.
hook ( item . Value . Value )
} )
}
// SetInvalidateCallback sets the invalidate callback to the provided hook.
func ( c * Cache [ Value ] ) SetInvalidateCallback ( hook func ( Value ) ) {
if hook == nil {
// Ensure non-nil hook.
hook = func ( Value ) { }
}
c . cache . SetInvalidateCallback ( func ( item * ttl . Entry [ int64 , result [ Value ] ] ) {
for _ , key := range item . Value . Keys {
2022-11-13 14:02:07 +01:00
// Delete key->pkey lookup
pkeys := key . key . pkeys
delete ( pkeys , key . value )
2022-11-11 12:18:38 +01:00
}
if item . Value . Error != nil {
// Skip error hooks
return
}
// Call user hook.
hook ( item . Value . Value )
} )
}
2022-11-13 14:02:07 +01:00
// Load will attempt to load an existing result from the cacche for the given lookup and key parts, else calling the load function and caching that result.
2022-11-11 12:18:38 +01:00
func ( c * Cache [ Value ] ) Load ( lookup string , load func ( ) ( Value , error ) , keyParts ... any ) ( Value , error ) {
var (
zero Value
res result [ Value ]
)
2022-11-13 14:02:07 +01:00
// Get lookup key info by name.
keyInfo := c . lookups . get ( lookup )
2022-11-11 12:18:38 +01:00
// Generate cache key string.
2022-11-13 14:02:07 +01:00
ckey := genKey ( keyParts ... )
2022-11-11 12:18:38 +01:00
// Acquire cache lock
c . cache . Lock ( )
2022-11-13 14:02:07 +01:00
// Look for primary key for cache key
pkey , ok := keyInfo . pkeys [ ckey ]
2022-11-11 12:18:38 +01:00
if ok {
// Fetch the result for primary key
entry , _ := c . cache . Cache . Get ( pkey )
res = entry . Value
}
// Done with lock
c . cache . Unlock ( )
if ! ok {
// Generate new result from fresh load.
res . Value , res . Error = load ( )
if res . Error != nil {
// This load returned an error, only
// store this item under provided key.
2022-11-13 14:02:07 +01:00
res . Keys = [ ] cachedKey { {
key : keyInfo ,
value : ckey ,
2022-11-11 12:18:38 +01:00
} }
} else {
// This was a successful load, generate keys.
res . Keys = c . lookups . generate ( res . Value )
}
// Acquire cache lock.
c . cache . Lock ( )
defer c . cache . Unlock ( )
2022-11-14 10:14:34 +01:00
// Cache this result
c . storeResult ( res )
2022-11-11 12:18:38 +01:00
}
// Catch and return error
if res . Error != nil {
return zero , res . Error
}
// Return a copy of value from cache
return c . copy ( res . Value ) , nil
}
2022-11-13 14:02:07 +01:00
// Store will call the given store function, and on success store the value in the cache as a positive result.
2022-11-11 12:18:38 +01:00
func ( c * Cache [ Value ] ) Store ( value Value , store func ( ) error ) error {
// Attempt to store this value.
if err := store ( ) ; err != nil {
return err
}
// Prepare cached result.
result := result [ Value ] {
Keys : c . lookups . generate ( value ) ,
Value : c . copy ( value ) ,
Error : nil ,
}
// Acquire cache lock.
c . cache . Lock ( )
defer c . cache . Unlock ( )
2022-11-14 10:14:34 +01:00
// Cache this result
c . storeResult ( result )
2022-11-11 12:18:38 +01:00
return nil
}
2022-11-13 14:02:07 +01:00
// Has checks the cache for a positive result under the given lookup and key parts.
2022-11-11 12:18:38 +01:00
func ( c * Cache [ Value ] ) Has ( lookup string , keyParts ... any ) bool {
var res result [ Value ]
2022-11-13 14:02:07 +01:00
// Get lookup key type by name.
keyType := c . lookups . get ( lookup )
2022-11-11 12:18:38 +01:00
// Generate cache key string.
2022-11-13 14:02:07 +01:00
ckey := genKey ( keyParts ... )
2022-11-11 12:18:38 +01:00
// Acquire cache lock
c . cache . Lock ( )
2022-11-13 14:02:07 +01:00
// Look for primary key for cache key
pkey , ok := keyType . pkeys [ ckey ]
2022-11-11 12:18:38 +01:00
if ok {
// Fetch the result for primary key
entry , _ := c . cache . Cache . Get ( pkey )
res = entry . Value
}
// Done with lock
c . cache . Unlock ( )
// Check for non-error result.
return ok && ( res . Error == nil )
}
2022-11-13 14:02:07 +01:00
// Invalidate will invalidate any result from the cache found under given lookup and key parts.
2022-11-11 12:18:38 +01:00
func ( c * Cache [ Value ] ) Invalidate ( lookup string , keyParts ... any ) {
2022-11-13 14:02:07 +01:00
// Get lookup key type by name.
keyType := c . lookups . get ( lookup )
2022-11-11 12:18:38 +01:00
// Generate cache key string.
2022-11-13 14:02:07 +01:00
ckey := genKey ( keyParts ... )
2022-11-11 12:18:38 +01:00
2022-11-13 14:02:07 +01:00
// Look for primary key for cache key
2022-11-11 12:18:38 +01:00
c . cache . Lock ( )
2022-11-13 14:02:07 +01:00
pkey , ok := keyType . pkeys [ ckey ]
2022-11-11 12:18:38 +01:00
c . cache . Unlock ( )
if ! ok {
return
}
// Invalid by primary key
c . cache . Invalidate ( pkey )
}
// Clear empties the cache, calling the invalidate callback.
func ( c * Cache [ Value ] ) Clear ( ) {
c . cache . Clear ( )
}
2022-11-13 14:02:07 +01:00
// Len returns the current length of the cache.
2022-11-11 12:18:38 +01:00
func ( c * Cache [ Value ] ) Len ( ) int {
return c . cache . Cache . Len ( )
}
2022-11-13 14:02:07 +01:00
// Cap returns the maximum capacity of this result cache.
2022-11-11 12:18:38 +01:00
func ( c * Cache [ Value ] ) Cap ( ) int {
return c . cache . Cache . Cap ( )
}
2022-11-14 10:14:34 +01:00
func ( c * Cache [ Value ] ) storeResult ( res result [ Value ] ) {
2022-11-11 12:18:38 +01:00
for _ , key := range res . Keys {
2022-11-13 14:02:07 +01:00
pkeys := key . key . pkeys
2022-11-11 12:18:38 +01:00
// Look for cache primary key
pkey , ok := pkeys [ key . value ]
if ok {
2022-11-14 10:14:34 +01:00
// Get the overlapping result with this key.
2022-11-11 12:18:38 +01:00
entry , _ := c . cache . Cache . Get ( pkey )
2022-11-13 14:02:07 +01:00
2022-11-14 10:14:34 +01:00
// From conflicting entry, drop this key, this
// will prevent eviction cleanup key confusion.
entry . Value . Keys . drop ( key . key . name )
if len ( entry . Value . Keys ) == 0 {
// We just over-wrote the only lookup key for
// this value, so we drop its primary key too
c . cache . Cache . Delete ( pkey )
}
2022-11-11 12:18:38 +01:00
}
}
// Get primary key
pkey := c . next
c . next ++
// Store all primary key lookups
for _ , key := range res . Keys {
2022-11-13 14:02:07 +01:00
pkeys := key . key . pkeys
2022-11-11 12:18:38 +01:00
pkeys [ key . value ] = pkey
}
// Store main entry under primary key, using evict hook if needed
c . cache . Cache . SetWithHook ( pkey , & ttl . Entry [ int64 , result [ Value ] ] {
Expiry : time . Now ( ) . Add ( c . cache . TTL ) ,
Key : pkey ,
Value : res ,
} , func ( _ int64 , item * ttl . Entry [ int64 , result [ Value ] ] ) {
c . cache . Evict ( item )
} )
}
type result [ Value any ] struct {
// keys accessible under
2022-11-14 10:14:34 +01:00
Keys cacheKeys
2022-11-11 12:18:38 +01:00
// cached value
Value Value
// cached error
Error error
}