2022-11-11 12:18:38 +01:00
package result
import (
2022-12-16 23:36:52 +01:00
"context"
2022-11-11 12:18:38 +01:00
"reflect"
"time"
"codeberg.org/gruf/go-cache/v3/ttl"
2022-12-16 23:36:52 +01:00
"codeberg.org/gruf/go-errors/v2"
2022-11-11 12:18:38 +01:00
)
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
2022-12-16 23:36:52 +01:00
// TODO: support toggling case sensitive lookups.
// CaseSensitive bool
2022-11-13 14:02:07 +01:00
}
// 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
2023-04-19 13:46:42 +02:00
invalid func ( Value ) // store unwrapped invalidate callback.
2022-11-11 12:18:38 +01:00
lookups structKeys // pre-determined struct lookups
2022-12-16 23:36:52 +01:00
ignore func ( error ) bool // determines cacheable errors
2022-11-11 12:18:38 +01:00
copy func ( Value ) Value // copies a Value type
next int64 // update key counter
}
2022-12-16 23:36:52 +01:00
// New returns a new initialized Cache, with given lookups, underlying value copy function and provided capacity.
func New [ 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 {
2023-01-06 11:16:09 +01:00
// Create keyed field info for lookup
c . lookups [ i ] = newStructKey ( 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 )
2022-12-16 23:36:52 +01:00
c . IgnoreErrors ( nil )
2022-11-11 12:18:38 +01:00
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-12-16 23:36:52 +01:00
pkeys := key . info . pkeys
delete ( pkeys , key . key )
2022-11-11 12:18:38 +01:00
}
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 ) { }
2023-04-19 13:46:42 +02:00
} // store hook.
c . invalid = hook
2022-11-11 12:18:38 +01:00
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
2022-12-16 23:36:52 +01:00
pkeys := key . info . pkeys
delete ( pkeys , key . key )
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-12-16 23:36:52 +01:00
// IgnoreErrors allows setting a function hook to determine which error types should / not be cached.
func ( c * Cache [ Value ] ) IgnoreErrors ( ignore func ( error ) bool ) {
if ignore == nil {
ignore = func ( err error ) bool {
2023-04-29 18:44:20 +02:00
return errors . Comparable (
2022-12-16 23:36:52 +01:00
err ,
context . Canceled ,
context . DeadlineExceeded ,
)
}
}
c . cache . Lock ( )
c . ignore = ignore
c . cache . Unlock ( )
}
// Load will attempt to load an existing result from the cacche for the given lookup and key parts, else calling the provided load function and caching the 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.
2023-01-06 11:16:09 +01:00
ckey := keyInfo . genKey ( keyParts )
2022-11-11 12:18:38 +01:00
// Acquire cache lock
c . cache . Lock ( )
2022-12-16 23:36:52 +01:00
// Look for primary cache key
2022-11-13 14:02:07 +01:00
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 {
2022-12-16 23:36:52 +01:00
// Generate fresh result.
value , err := load ( )
if err != nil {
if c . ignore ( err ) {
// don't cache this error type
return zero , err
}
// Store error result.
res . Error = err
2022-11-11 12:18:38 +01:00
// This load returned an error, only
// store this item under provided key.
2022-12-16 23:36:52 +01:00
res . Keys = [ ] cacheKey { {
info : keyInfo ,
key : ckey ,
2022-11-11 12:18:38 +01:00
} }
} else {
2022-12-16 23:36:52 +01:00
// Store value result.
res . Value = value
2022-11-11 12:18:38 +01:00
// 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-12-16 23:36:52 +01:00
// Cache result
c . store ( 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-12-16 23:36:52 +01:00
// Cache result
c . store ( result )
2022-11-11 12:18:38 +01:00
2023-04-19 13:46:42 +02:00
// Call invalidate.
c . invalid ( value )
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 ]
2023-01-06 11:16:09 +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.
2023-01-06 11:16:09 +01:00
ckey := keyInfo . 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
2023-01-06 11:16:09 +01:00
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 ( )
// 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 ) {
2023-01-06 11:16:09 +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.
2023-01-06 11:16:09 +01:00
ckey := keyInfo . 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 ( )
2023-01-06 11:16:09 +01:00
pkey , ok := keyInfo . 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-12-16 23:36:52 +01:00
// store will cache this result under all of its required cache keys.
func ( c * Cache [ Value ] ) store ( res result [ Value ] ) {
2022-11-11 12:18:38 +01:00
for _ , key := range res . Keys {
2022-12-16 23:36:52 +01:00
pkeys := key . info . pkeys
2022-11-11 12:18:38 +01:00
// Look for cache primary key
2022-12-16 23:36:52 +01:00
pkey , ok := pkeys [ key . key ]
2022-11-11 12:18:38 +01:00
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.
2022-12-16 23:36:52 +01:00
entry . Value . Keys . drop ( key . info . name )
2022-11-14 10:14:34 +01:00
if len ( entry . Value . Keys ) == 0 {
// We just over-wrote the only lookup key for
2022-12-16 23:36:52 +01:00
// this value, so we drop its primary key too.
2022-11-14 10:14:34 +01:00
c . cache . Cache . Delete ( pkey )
}
2022-11-11 12:18:38 +01:00
}
}
// Get primary key
pkey := c . next
c . next ++
2022-12-16 23:36:52 +01:00
if pkey > c . next {
panic ( "cache primary key overflow" )
}
2022-11-11 12:18:38 +01:00
// Store all primary key lookups
for _ , key := range res . Keys {
2022-12-16 23:36:52 +01:00
pkeys := key . info . pkeys
pkeys [ key . key ] = pkey
2022-11-11 12:18:38 +01:00
}
// 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
}