144 lines
2.9 KiB
Go
144 lines
2.9 KiB
Go
package state
|
|
|
|
import (
|
|
"sort"
|
|
|
|
"github.com/tendermint/go-wire/data"
|
|
)
|
|
|
|
// KVStore is a simple interface to get/set data
|
|
type KVStore interface {
|
|
Set(key, value []byte)
|
|
Get(key []byte) (value []byte)
|
|
}
|
|
|
|
//----------------------------------------
|
|
|
|
// Model grabs together key and value to allow easier return values
|
|
type Model struct {
|
|
Key data.Bytes
|
|
Value data.Bytes
|
|
}
|
|
|
|
// SimpleDB allows us to do some basic range queries on a db
|
|
type SimpleDB interface {
|
|
KVStore
|
|
|
|
Has(key []byte) (has bool)
|
|
Remove(key []byte) (value []byte) // returns old value if there was one
|
|
|
|
// Start is inclusive, End is exclusive...
|
|
// Thus List ([]byte{12, 13}, []byte{12, 14}) will return anything with
|
|
// the prefix []byte{12, 13}
|
|
List(start, end []byte, limit int) []Model
|
|
First(start, end []byte) Model
|
|
Last(start, end []byte) Model
|
|
|
|
// // Checkpoint returns the same state, but where writes
|
|
// // are buffered and don't affect the parent
|
|
// Checkpoint() SimpleDB
|
|
|
|
// // Commit will take all changes from the checkpoint and write
|
|
// // them to the parent.
|
|
// // Returns an error if this is not a child of this one
|
|
// Commit(SimpleDB) error
|
|
|
|
// Discard will remove reference to this
|
|
Discard()
|
|
}
|
|
|
|
//----------------------------------------
|
|
|
|
type MemKVStore struct {
|
|
m map[string][]byte
|
|
}
|
|
|
|
var _ SimpleDB = NewMemKVStore()
|
|
|
|
func NewMemKVStore() *MemKVStore {
|
|
return &MemKVStore{
|
|
m: make(map[string][]byte, 0),
|
|
}
|
|
}
|
|
|
|
func (mkv *MemKVStore) Set(key []byte, value []byte) {
|
|
mkv.m[string(key)] = value
|
|
}
|
|
|
|
func (mkv *MemKVStore) Get(key []byte) (value []byte) {
|
|
return mkv.m[string(key)]
|
|
}
|
|
|
|
func (mkv *MemKVStore) Has(key []byte) (has bool) {
|
|
_, ok := mkv.m[string(key)]
|
|
return ok
|
|
}
|
|
|
|
func (mkv *MemKVStore) Remove(key []byte) (value []byte) {
|
|
val := mkv.m[string(key)]
|
|
delete(mkv.m, string(key))
|
|
return val
|
|
}
|
|
|
|
func (mkv *MemKVStore) List(start, end []byte, limit int) []Model {
|
|
keys := mkv.keysInRange(start, end)
|
|
sort.Strings(keys)
|
|
keys = keys[:limit]
|
|
|
|
res := make([]Model, len(keys))
|
|
for i, k := range keys {
|
|
res[i] = Model{
|
|
Key: []byte(k),
|
|
Value: mkv.m[k],
|
|
}
|
|
}
|
|
return res
|
|
}
|
|
|
|
// First iterates through all keys to find the one that matches
|
|
func (mkv *MemKVStore) First(start, end []byte) Model {
|
|
key := ""
|
|
for _, k := range mkv.keysInRange(start, end) {
|
|
if key == "" || k < key {
|
|
key = k
|
|
}
|
|
}
|
|
if key == "" {
|
|
return Model{}
|
|
}
|
|
return Model{
|
|
Key: []byte(key),
|
|
Value: mkv.m[key],
|
|
}
|
|
}
|
|
|
|
func (mkv *MemKVStore) Last(start, end []byte) Model {
|
|
key := ""
|
|
for _, k := range mkv.keysInRange(start, end) {
|
|
if key == "" || k > key {
|
|
key = k
|
|
}
|
|
}
|
|
if key == "" {
|
|
return Model{}
|
|
}
|
|
return Model{
|
|
Key: []byte(key),
|
|
Value: mkv.m[key],
|
|
}
|
|
}
|
|
|
|
func (mkv *MemKVStore) Discard() {
|
|
mkv.m = make(map[string][]byte, 0)
|
|
}
|
|
|
|
func (mkv *MemKVStore) keysInRange(start, end []byte) (res []string) {
|
|
s, e := string(start), string(end)
|
|
for k := range mkv.m {
|
|
if k >= s && k < e {
|
|
res = append(res, k)
|
|
}
|
|
}
|
|
return
|
|
}
|