initial public commit

This commit is contained in:
Gani Georgiev
2022-07-07 00:19:05 +03:00
commit 3d07f0211d
484 changed files with 92412 additions and 0 deletions
+84
View File
@@ -0,0 +1,84 @@
package store
import "sync"
// Store defines a concurrent safe in memory key-value data store.
type Store[T any] struct {
mux sync.RWMutex
data map[string]T
}
// New creates a new Store[T] instance.
func New[T any](data map[string]T) *Store[T] {
return &Store[T]{data: data}
}
// Remove removes a single entry from the store.
//
// Remove does nothing if key doesn't exist in the store.
func (s *Store[T]) Remove(key string) {
s.mux.Lock()
defer s.mux.Unlock()
delete(s.data, key)
}
// Has checks if element with the specified key exist or not.
func (s *Store[T]) Has(key string) bool {
s.mux.Lock()
defer s.mux.Unlock()
_, ok := s.data[key]
return ok
}
// Get returns a single element value from the store.
//
// If key is not set, the zero T value is returned.
func (s *Store[T]) Get(key string) T {
s.mux.Lock()
defer s.mux.Unlock()
return s.data[key]
}
// Set sets (or overwrite if already exist) a new value for key.
func (s *Store[T]) Set(key string, value T) {
s.mux.Lock()
defer s.mux.Unlock()
if s.data == nil {
s.data = make(map[string]T)
}
s.data[key] = value
}
// SetIfLessThanLimit sets (or overwrite if already exist) a new value for key.
//
// This is method is similar to Set() but **it will skip adding new elements**
// to the store if the store length has reached the specified limit.
// `false` is returned if maxAllowedElements limit is reached.
func (s *Store[T]) SetIfLessThanLimit(key string, value T, maxAllowedElements int) bool {
s.mux.Lock()
defer s.mux.Unlock()
// init map if not already
if s.data == nil {
s.data = make(map[string]T)
}
// check for existing item
_, ok := s.data[key]
if !ok && len(s.data) >= maxAllowedElements {
// cannot add more items
return false
}
// add/overwrite item
s.data[key] = value
return true
}
+126
View File
@@ -0,0 +1,126 @@
package store_test
import (
"testing"
"github.com/pocketbase/pocketbase/tools/store"
)
func TestNew(t *testing.T) {
s := store.New(map[string]int{"test": 1})
if s.Get("test") != 1 {
t.Error("Expected the initizialized store map to be loaded")
}
}
func TestRemove(t *testing.T) {
s := store.New(map[string]bool{"test": true})
keys := []string{"test", "missing"}
for i, key := range keys {
s.Remove(key)
if s.Has(key) {
t.Errorf("(%d) Expected %q to be removed", i, key)
}
}
}
func TestHas(t *testing.T) {
s := store.New(map[string]int{"test1": 0, "test2": 1})
scenarios := []struct {
key string
exist bool
}{
{"test1", true},
{"test2", true},
{"missing", false},
}
for i, scenario := range scenarios {
exist := s.Has(scenario.key)
if exist != scenario.exist {
t.Errorf("(%d) Expected %v, got %v", i, scenario.exist, exist)
}
}
}
func TestGet(t *testing.T) {
s := store.New(map[string]int{"test1": 0, "test2": 1})
scenarios := []struct {
key string
expect int
}{
{"test1", 0},
{"test2", 1},
{"missing", 0}, // should auto fallback to the zero value
}
for i, scenario := range scenarios {
val := s.Get(scenario.key)
if val != scenario.expect {
t.Errorf("(%d) Expected %v, got %v", i, scenario.expect, val)
}
}
}
func TestSet(t *testing.T) {
s := store.New[int](nil)
data := map[string]int{"test1": 0, "test2": 1, "test3": 3}
// set values
for k, v := range data {
s.Set(k, v)
}
// verify that the values are set
for k, v := range data {
if !s.Has(k) {
t.Errorf("Expected key %q", k)
}
val := s.Get(k)
if val != v {
t.Errorf("Expected %v, got %v for key %q", v, val, k)
}
}
}
func TestSetIfLessThanLimit(t *testing.T) {
s := store.New[int](nil)
limit := 2
// set values
scenarios := []struct {
key string
value int
expected bool
}{
{"test1", 1, true},
{"test2", 2, true},
{"test3", 3, false},
{"test2", 4, true}, // overwrite
}
for i, scenario := range scenarios {
result := s.SetIfLessThanLimit(scenario.key, scenario.value, limit)
if result != scenario.expected {
t.Errorf("(%d) Expected result %v, got %v", i, scenario.expected, result)
}
if !scenario.expected && s.Has(scenario.key) {
t.Errorf("(%d) Expected key %q to not be set", i, scenario.key)
}
val := s.Get(scenario.key)
if scenario.expected && val != scenario.value {
t.Errorf("(%d) Expected value %v, got %v", i, scenario.value, val)
}
}
}