2020-10-10 18:53:59 +02:00
|
|
|
// Copyright 2020 Vikunja and contriubtors. All rights reserved.
|
|
|
|
//
|
|
|
|
// This file is part of Vikunja.
|
|
|
|
//
|
|
|
|
// Vikunja is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// Vikunja is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Vikunja. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
package memory
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
2020-10-11 22:10:03 +02:00
|
|
|
|
|
|
|
e "code.vikunja.io/api/pkg/modules/keyvalue/error"
|
2020-10-10 18:53:59 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// Storage is the memory implementation of a storage backend
|
|
|
|
type Storage struct {
|
|
|
|
store map[string]interface{}
|
|
|
|
mutex sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewStorage creates a new memory storage
|
|
|
|
func NewStorage() *Storage {
|
|
|
|
s := &Storage{}
|
|
|
|
s.store = make(map[string]interface{})
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put puts a value into the memory storage
|
|
|
|
func (s *Storage) Put(key string, value interface{}) (err error) {
|
|
|
|
s.mutex.Lock()
|
|
|
|
defer s.mutex.Unlock()
|
|
|
|
s.store[key] = value
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get retrieves a saved value from memory storage
|
|
|
|
func (s *Storage) Get(key string) (value interface{}, err error) {
|
|
|
|
s.mutex.Lock()
|
|
|
|
defer s.mutex.Unlock()
|
|
|
|
|
|
|
|
var exists bool
|
|
|
|
value, exists = s.store[key]
|
|
|
|
if !exists {
|
|
|
|
return nil, &e.ErrValueNotFoundForKey{Key: key}
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Del removes a saved value from a memory storage
|
|
|
|
func (s *Storage) Del(key string) (err error) {
|
|
|
|
s.mutex.Lock()
|
|
|
|
defer s.mutex.Unlock()
|
|
|
|
delete(s.store, key)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// IncrBy increases the value saved at key by the amount provided through update
|
|
|
|
// It assumes the value saved for the key either does not exist or has a type of int64
|
|
|
|
func (s *Storage) IncrBy(key string, update int64) (err error) {
|
|
|
|
s.mutex.Lock()
|
|
|
|
defer s.mutex.Unlock()
|
|
|
|
|
|
|
|
v, err := s.Get(key)
|
|
|
|
if err != nil && !e.IsErrValueNotFoundForKey(err) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
val, is := v.(int64)
|
|
|
|
if !is {
|
|
|
|
return &e.ErrValueHasWrongType{Key: key, ExpectedValue: "int64"}
|
|
|
|
}
|
|
|
|
s.store[key] = val + update
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DecrBy decreases the value saved at key by the amount provided through update
|
|
|
|
// It assumes the value saved for the key either does not exist or has a type of int64
|
|
|
|
func (s *Storage) DecrBy(key string, update int64) (err error) {
|
|
|
|
s.mutex.Lock()
|
|
|
|
defer s.mutex.Unlock()
|
|
|
|
|
|
|
|
v, err := s.Get(key)
|
|
|
|
if err != nil && !e.IsErrValueNotFoundForKey(err) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
val, is := v.(int64)
|
|
|
|
if !is {
|
|
|
|
return &e.ErrValueHasWrongType{Key: key, ExpectedValue: "int64"}
|
|
|
|
}
|
|
|
|
s.store[key] = val - update
|
|
|
|
return nil
|
|
|
|
}
|