mirror of
https://github.com/netbirdio/netbird.git
synced 2025-01-18 20:08:28 +01:00
301 lines
7.9 KiB
Go
301 lines
7.9 KiB
Go
package refcounter
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"runtime"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/hashicorp/go-multierror"
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
nberrors "github.com/netbirdio/netbird/client/errors"
|
|
)
|
|
|
|
const logLevel = log.TraceLevel
|
|
|
|
// ErrIgnore can be returned by AddFunc to indicate that the counter should not be incremented for the given key.
|
|
var ErrIgnore = errors.New("ignore")
|
|
|
|
// Ref holds the reference count and associated data for a key.
|
|
type Ref[O any] struct {
|
|
Count int
|
|
Out O
|
|
}
|
|
|
|
// AddFunc is the function type for adding a new key.
|
|
// Key is the type of the key (e.g., netip.Prefix).
|
|
type AddFunc[Key, I, O any] func(key Key, in I) (out O, err error)
|
|
|
|
// RemoveFunc is the function type for removing a key.
|
|
type RemoveFunc[Key, O any] func(key Key, out O) error
|
|
|
|
// Counter is a generic reference counter for managing keys and their associated data.
|
|
// Key: The type of the key (e.g., netip.Prefix, string).
|
|
//
|
|
// I: The input type for the AddFunc. It is the input type for additional data needed
|
|
// when adding a key, it is passed as the second argument to AddFunc.
|
|
//
|
|
// O: The output type for the AddFunc and RemoveFunc. This is the output returned by AddFunc.
|
|
// It is stored and passed to RemoveFunc when the reference count reaches 0.
|
|
//
|
|
// The types can be aliased to a specific type using the following syntax:
|
|
//
|
|
// type RouteRefCounter = Counter[netip.Prefix, any, any]
|
|
type Counter[Key comparable, I, O any] struct {
|
|
// refCountMap keeps track of the reference Ref for keys
|
|
refCountMap map[Key]Ref[O]
|
|
mu sync.Mutex
|
|
// idMap keeps track of the keys associated with an ID for removal
|
|
idMap map[string][]Key
|
|
add AddFunc[Key, I, O]
|
|
remove RemoveFunc[Key, O]
|
|
}
|
|
|
|
// New creates a new Counter instance.
|
|
// Usage example:
|
|
//
|
|
// counter := New[netip.Prefix, string, string](
|
|
// func(key netip.Prefix, in string) (out string, err error) { ... },
|
|
// func(key netip.Prefix, out string) error { ... },`
|
|
// )
|
|
func New[Key comparable, I, O any](add AddFunc[Key, I, O], remove RemoveFunc[Key, O]) *Counter[Key, I, O] {
|
|
return &Counter[Key, I, O]{
|
|
refCountMap: map[Key]Ref[O]{},
|
|
idMap: map[string][]Key{},
|
|
add: add,
|
|
remove: remove,
|
|
}
|
|
}
|
|
|
|
// LoadData loads the data from the existing counter
|
|
// The passed counter should not be used any longer after calling this function.
|
|
func (rm *Counter[Key, I, O]) LoadData(
|
|
existingCounter *Counter[Key, I, O],
|
|
) {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
existingCounter.mu.Lock()
|
|
defer existingCounter.mu.Unlock()
|
|
|
|
rm.refCountMap = existingCounter.refCountMap
|
|
rm.idMap = existingCounter.idMap
|
|
}
|
|
|
|
// Get retrieves the current reference count and associated data for a key.
|
|
// If the key doesn't exist, it returns a zero value Ref and false.
|
|
func (rm *Counter[Key, I, O]) Get(key Key) (Ref[O], bool) {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
|
|
ref, ok := rm.refCountMap[key]
|
|
return ref, ok
|
|
}
|
|
|
|
// Increment increments the reference count for the given key.
|
|
// If this is the first reference to the key, the AddFunc is called.
|
|
func (rm *Counter[Key, I, O]) Increment(key Key, in I) (Ref[O], error) {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
|
|
return rm.increment(key, in)
|
|
}
|
|
|
|
func (rm *Counter[Key, I, O]) increment(key Key, in I) (Ref[O], error) {
|
|
ref := rm.refCountMap[key]
|
|
logCallerF("Increasing ref count [%d -> %d] for key %v with In [%v] Out [%v]", ref.Count, ref.Count+1, key, in, ref.Out)
|
|
|
|
// Call AddFunc only if it's a new key
|
|
if ref.Count == 0 {
|
|
logCallerF("Calling add for key %v", key)
|
|
out, err := rm.add(key, in)
|
|
|
|
if errors.Is(err, ErrIgnore) {
|
|
return ref, nil
|
|
}
|
|
if err != nil {
|
|
return ref, fmt.Errorf("failed to add for key %v: %w", key, err)
|
|
}
|
|
ref.Out = out
|
|
}
|
|
|
|
ref.Count++
|
|
rm.refCountMap[key] = ref
|
|
|
|
return ref, nil
|
|
}
|
|
|
|
// IncrementWithID increments the reference count for the given key and groups it under the given ID.
|
|
// If this is the first reference to the key, the AddFunc is called.
|
|
func (rm *Counter[Key, I, O]) IncrementWithID(id string, key Key, in I) (Ref[O], error) {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
|
|
ref, err := rm.increment(key, in)
|
|
if err != nil {
|
|
return ref, fmt.Errorf("with ID: %w", err)
|
|
}
|
|
rm.idMap[id] = append(rm.idMap[id], key)
|
|
|
|
return ref, nil
|
|
}
|
|
|
|
// Decrement decrements the reference count for the given key.
|
|
// If the reference count reaches 0, the RemoveFunc is called.
|
|
func (rm *Counter[Key, I, O]) Decrement(key Key) (Ref[O], error) {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
return rm.decrement(key)
|
|
}
|
|
|
|
func (rm *Counter[Key, I, O]) decrement(key Key) (Ref[O], error) {
|
|
ref, ok := rm.refCountMap[key]
|
|
if !ok {
|
|
logCallerF("No reference found for key %v", key)
|
|
return ref, nil
|
|
}
|
|
|
|
logCallerF("Decreasing ref count [%d -> %d] for key %v with Out [%v]", ref.Count, ref.Count-1, key, ref.Out)
|
|
if ref.Count == 1 {
|
|
logCallerF("Calling remove for key %v", key)
|
|
if err := rm.remove(key, ref.Out); err != nil {
|
|
return ref, fmt.Errorf("remove for key %v: %w", key, err)
|
|
}
|
|
delete(rm.refCountMap, key)
|
|
} else {
|
|
ref.Count--
|
|
rm.refCountMap[key] = ref
|
|
}
|
|
|
|
return ref, nil
|
|
}
|
|
|
|
// DecrementWithID decrements the reference count for all keys associated with the given ID.
|
|
// If the reference count reaches 0, the RemoveFunc is called.
|
|
func (rm *Counter[Key, I, O]) DecrementWithID(id string) error {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
|
|
var merr *multierror.Error
|
|
for _, key := range rm.idMap[id] {
|
|
if _, err := rm.decrement(key); err != nil {
|
|
merr = multierror.Append(merr, err)
|
|
}
|
|
}
|
|
delete(rm.idMap, id)
|
|
|
|
return nberrors.FormatErrorOrNil(merr)
|
|
}
|
|
|
|
// Flush removes all references and calls RemoveFunc for each key.
|
|
func (rm *Counter[Key, I, O]) Flush() error {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
|
|
var merr *multierror.Error
|
|
for key := range rm.refCountMap {
|
|
logCallerF("Calling remove for key %v", key)
|
|
ref := rm.refCountMap[key]
|
|
if err := rm.remove(key, ref.Out); err != nil {
|
|
merr = multierror.Append(merr, fmt.Errorf("remove for key %v: %w", key, err))
|
|
}
|
|
}
|
|
|
|
clear(rm.refCountMap)
|
|
clear(rm.idMap)
|
|
|
|
return nberrors.FormatErrorOrNil(merr)
|
|
}
|
|
|
|
// Clear removes all references without calling RemoveFunc.
|
|
func (rm *Counter[Key, I, O]) Clear() {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
|
|
clear(rm.refCountMap)
|
|
clear(rm.idMap)
|
|
}
|
|
|
|
// MarshalJSON implements the json.Marshaler interface for Counter.
|
|
func (rm *Counter[Key, I, O]) MarshalJSON() ([]byte, error) {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
|
|
return json.Marshal(struct {
|
|
RefCountMap map[Key]Ref[O] `json:"refCountMap"`
|
|
IDMap map[string][]Key `json:"idMap"`
|
|
}{
|
|
RefCountMap: rm.refCountMap,
|
|
IDMap: rm.idMap,
|
|
})
|
|
}
|
|
|
|
// UnmarshalJSON implements the json.Unmarshaler interface for Counter.
|
|
func (rm *Counter[Key, I, O]) UnmarshalJSON(data []byte) error {
|
|
rm.mu.Lock()
|
|
defer rm.mu.Unlock()
|
|
|
|
var temp struct {
|
|
RefCountMap map[Key]Ref[O] `json:"refCountMap"`
|
|
IDMap map[string][]Key `json:"idMap"`
|
|
}
|
|
if err := json.Unmarshal(data, &temp); err != nil {
|
|
return err
|
|
}
|
|
rm.refCountMap = temp.RefCountMap
|
|
rm.idMap = temp.IDMap
|
|
|
|
if temp.RefCountMap == nil {
|
|
temp.RefCountMap = map[Key]Ref[O]{}
|
|
}
|
|
if temp.IDMap == nil {
|
|
temp.IDMap = map[string][]Key{}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func getCallerInfo(depth int, maxDepth int) (string, bool) {
|
|
if depth >= maxDepth {
|
|
return "", false
|
|
}
|
|
|
|
pc, _, _, ok := runtime.Caller(depth)
|
|
if !ok {
|
|
return "", false
|
|
}
|
|
|
|
if details := runtime.FuncForPC(pc); details != nil {
|
|
name := details.Name()
|
|
|
|
lastDotIndex := strings.LastIndex(name, "/")
|
|
if lastDotIndex != -1 {
|
|
name = name[lastDotIndex+1:]
|
|
}
|
|
|
|
if strings.HasPrefix(name, "refcounter.") {
|
|
// +2 to account for recursion
|
|
return getCallerInfo(depth+2, maxDepth)
|
|
}
|
|
|
|
return name, true
|
|
}
|
|
|
|
return "", false
|
|
}
|
|
|
|
// logCaller logs a message with the package name and method of the function that called the current function.
|
|
func logCallerF(format string, args ...interface{}) {
|
|
if log.GetLevel() < logLevel {
|
|
return
|
|
}
|
|
|
|
if callerName, ok := getCallerInfo(3, 18); ok {
|
|
format = fmt.Sprintf("[%s] %s", callerName, format)
|
|
}
|
|
|
|
log.StandardLogger().Logf(logLevel, format, args...)
|
|
}
|