2019-02-01 17:03:09 -08:00
|
|
|
package prefix
|
2018-06-21 14:33:36 -07:00
|
|
|
|
|
|
|
import (
|
2018-09-05 21:48:32 -07:00
|
|
|
"bytes"
|
2020-01-16 13:46:51 -08:00
|
|
|
"errors"
|
2018-07-12 16:58:51 -07:00
|
|
|
"io"
|
|
|
|
|
2019-02-01 17:03:09 -08:00
|
|
|
"github.com/cosmos/cosmos-sdk/store/cachekv"
|
|
|
|
"github.com/cosmos/cosmos-sdk/store/tracekv"
|
|
|
|
"github.com/cosmos/cosmos-sdk/store/types"
|
2018-06-21 14:33:36 -07:00
|
|
|
)
|
|
|
|
|
2019-02-01 17:03:09 -08:00
|
|
|
var _ types.KVStore = Store{}
|
2018-07-26 18:24:18 -07:00
|
|
|
|
2019-02-01 17:03:09 -08:00
|
|
|
// Store is similar with tendermint/tendermint/libs/db/prefix_db
|
2018-10-08 12:43:55 -07:00
|
|
|
// both gives access only to the limited subset of the store
|
|
|
|
// for convinience or safety
|
2019-02-01 17:03:09 -08:00
|
|
|
type Store struct {
|
|
|
|
parent types.KVStore
|
2018-06-21 14:33:36 -07:00
|
|
|
prefix []byte
|
|
|
|
}
|
|
|
|
|
2019-02-01 17:03:09 -08:00
|
|
|
func NewStore(parent types.KVStore, prefix []byte) Store {
|
|
|
|
return Store{
|
|
|
|
parent: parent,
|
|
|
|
prefix: prefix,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-10 23:59:58 -07:00
|
|
|
func cloneAppend(bz []byte, tail []byte) (res []byte) {
|
|
|
|
res = make([]byte, len(bz)+len(tail))
|
2018-08-31 02:03:43 -07:00
|
|
|
copy(res, bz)
|
2018-10-08 11:06:16 -07:00
|
|
|
copy(res[len(bz):], tail)
|
2018-08-31 02:03:43 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) key(key []byte) (res []byte) {
|
2018-09-28 12:53:12 -07:00
|
|
|
if key == nil {
|
2019-02-01 17:03:09 -08:00
|
|
|
panic("nil key on Store")
|
2018-09-28 12:53:12 -07:00
|
|
|
}
|
2018-09-27 11:12:52 -07:00
|
|
|
res = cloneAppend(s.prefix, key)
|
2018-08-31 02:03:43 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-06-21 14:33:36 -07:00
|
|
|
// Implements Store
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) GetStoreType() types.StoreType {
|
2018-07-26 18:24:18 -07:00
|
|
|
return s.parent.GetStoreType()
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements CacheWrap
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) CacheWrap() types.CacheWrap {
|
|
|
|
return cachekv.NewStore(s)
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
2018-07-12 16:58:51 -07:00
|
|
|
// CacheWrapWithTrace implements the KVStore interface.
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) CacheWrapWithTrace(w io.Writer, tc types.TraceContext) types.CacheWrap {
|
|
|
|
return cachekv.NewStore(tracekv.NewStore(s, w, tc))
|
2018-07-12 16:58:51 -07:00
|
|
|
}
|
|
|
|
|
2018-06-21 14:33:36 -07:00
|
|
|
// Implements KVStore
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) Get(key []byte) []byte {
|
2018-08-31 02:03:43 -07:00
|
|
|
res := s.parent.Get(s.key(key))
|
|
|
|
return res
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements KVStore
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) Has(key []byte) bool {
|
2018-08-31 02:03:43 -07:00
|
|
|
return s.parent.Has(s.key(key))
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements KVStore
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) Set(key, value []byte) {
|
|
|
|
types.AssertValidKey(key)
|
|
|
|
types.AssertValidValue(value)
|
2018-08-31 02:03:43 -07:00
|
|
|
s.parent.Set(s.key(key), value)
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements KVStore
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) Delete(key []byte) {
|
2018-08-31 02:03:43 -07:00
|
|
|
s.parent.Delete(s.key(key))
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements KVStore
|
2018-09-05 21:48:32 -07:00
|
|
|
// Check https://github.com/tendermint/tendermint/blob/master/libs/db/prefix_db.go#L106
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) Iterator(start, end []byte) types.Iterator {
|
2018-09-27 11:12:52 -07:00
|
|
|
newstart := cloneAppend(s.prefix, start)
|
2018-08-31 02:03:43 -07:00
|
|
|
|
|
|
|
var newend []byte
|
2018-06-27 10:21:12 -07:00
|
|
|
if end == nil {
|
2018-09-05 21:48:32 -07:00
|
|
|
newend = cpIncr(s.prefix)
|
2018-06-27 10:21:12 -07:00
|
|
|
} else {
|
2018-09-27 11:12:52 -07:00
|
|
|
newend = cloneAppend(s.prefix, end)
|
2018-06-27 10:21:12 -07:00
|
|
|
}
|
2018-08-31 02:03:43 -07:00
|
|
|
|
2018-10-08 12:43:55 -07:00
|
|
|
iter := s.parent.Iterator(newstart, newend)
|
2018-09-05 21:48:32 -07:00
|
|
|
|
2018-10-08 12:43:55 -07:00
|
|
|
return newPrefixIterator(s.prefix, start, end, iter)
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
2020-03-06 04:28:19 -08:00
|
|
|
// ReverseIterator implements KVStore
|
2018-09-05 21:48:32 -07:00
|
|
|
// Check https://github.com/tendermint/tendermint/blob/master/libs/db/prefix_db.go#L129
|
2019-02-01 17:03:09 -08:00
|
|
|
func (s Store) ReverseIterator(start, end []byte) types.Iterator {
|
2018-11-27 00:14:22 -08:00
|
|
|
newstart := cloneAppend(s.prefix, start)
|
2018-08-31 02:03:43 -07:00
|
|
|
|
2018-09-05 21:48:32 -07:00
|
|
|
var newend []byte
|
2018-06-27 10:21:12 -07:00
|
|
|
if end == nil {
|
2018-11-27 00:14:22 -08:00
|
|
|
newend = cpIncr(s.prefix)
|
2018-06-27 10:21:12 -07:00
|
|
|
} else {
|
2018-09-10 23:59:58 -07:00
|
|
|
newend = cloneAppend(s.prefix, end)
|
2018-06-27 10:21:12 -07:00
|
|
|
}
|
2018-08-31 02:03:43 -07:00
|
|
|
|
2018-09-05 21:48:32 -07:00
|
|
|
iter := s.parent.ReverseIterator(newstart, newend)
|
|
|
|
|
2018-10-08 12:43:55 -07:00
|
|
|
return newPrefixIterator(s.prefix, start, end, iter)
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
2019-02-01 17:03:09 -08:00
|
|
|
var _ types.Iterator = (*prefixIterator)(nil)
|
2018-06-21 14:33:36 -07:00
|
|
|
|
|
|
|
type prefixIterator struct {
|
2020-03-06 04:28:19 -08:00
|
|
|
prefix []byte
|
|
|
|
start []byte
|
|
|
|
end []byte
|
|
|
|
iter types.Iterator
|
|
|
|
valid bool
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
2019-02-01 17:03:09 -08:00
|
|
|
func newPrefixIterator(prefix, start, end []byte, parent types.Iterator) *prefixIterator {
|
2018-10-08 13:37:18 -07:00
|
|
|
return &prefixIterator{
|
2018-10-08 12:43:55 -07:00
|
|
|
prefix: prefix,
|
|
|
|
start: start,
|
|
|
|
end: end,
|
|
|
|
iter: parent,
|
|
|
|
valid: parent.Valid() && bytes.HasPrefix(parent.Key(), prefix),
|
|
|
|
}
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements Iterator
|
2020-03-06 04:28:19 -08:00
|
|
|
func (pi *prefixIterator) Domain() ([]byte, []byte) {
|
|
|
|
return pi.start, pi.end
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements Iterator
|
2020-03-06 04:28:19 -08:00
|
|
|
func (pi *prefixIterator) Valid() bool {
|
|
|
|
return pi.valid && pi.iter.Valid()
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements Iterator
|
2020-03-06 04:28:19 -08:00
|
|
|
func (pi *prefixIterator) Next() {
|
|
|
|
if !pi.valid {
|
2018-09-05 21:48:32 -07:00
|
|
|
panic("prefixIterator invalid, cannot call Next()")
|
|
|
|
}
|
2020-04-29 19:36:34 -07:00
|
|
|
|
|
|
|
if pi.iter.Next(); !pi.iter.Valid() || !bytes.HasPrefix(pi.iter.Key(), pi.prefix) {
|
2020-03-06 04:28:19 -08:00
|
|
|
// TODO: shouldn't pi be set to nil instead?
|
|
|
|
pi.valid = false
|
2018-09-05 21:48:32 -07:00
|
|
|
}
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements Iterator
|
2020-03-06 04:28:19 -08:00
|
|
|
func (pi *prefixIterator) Key() (key []byte) {
|
|
|
|
if !pi.valid {
|
2018-10-08 13:06:40 -07:00
|
|
|
panic("prefixIterator invalid, cannot call Key()")
|
|
|
|
}
|
2020-04-29 19:36:34 -07:00
|
|
|
|
2020-03-06 04:28:19 -08:00
|
|
|
key = pi.iter.Key()
|
|
|
|
key = stripPrefix(key, pi.prefix)
|
2020-04-29 19:36:34 -07:00
|
|
|
|
2018-06-21 14:33:36 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Implements Iterator
|
2020-03-06 04:28:19 -08:00
|
|
|
func (pi *prefixIterator) Value() []byte {
|
|
|
|
if !pi.valid {
|
2018-10-08 13:06:40 -07:00
|
|
|
panic("prefixIterator invalid, cannot call Value()")
|
|
|
|
}
|
2020-04-29 19:36:34 -07:00
|
|
|
|
2020-03-06 04:28:19 -08:00
|
|
|
return pi.iter.Value()
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implements Iterator
|
2020-08-14 10:58:53 -07:00
|
|
|
func (pi *prefixIterator) Close() error {
|
|
|
|
return pi.iter.Close()
|
2018-06-21 14:33:36 -07:00
|
|
|
}
|
2018-09-05 21:48:32 -07:00
|
|
|
|
2020-01-16 13:46:51 -08:00
|
|
|
// Error returns an error if the prefixIterator is invalid defined by the Valid
|
|
|
|
// method.
|
2020-03-06 04:28:19 -08:00
|
|
|
func (pi *prefixIterator) Error() error {
|
|
|
|
if !pi.Valid() {
|
2020-01-16 13:46:51 -08:00
|
|
|
return errors.New("invalid prefixIterator")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-09-05 21:48:32 -07:00
|
|
|
// copied from github.com/tendermint/tendermint/libs/db/prefix_db.go
|
|
|
|
func stripPrefix(key []byte, prefix []byte) []byte {
|
|
|
|
if len(key) < len(prefix) || !bytes.Equal(key[:len(prefix)], prefix) {
|
|
|
|
panic("should not happen")
|
|
|
|
}
|
2020-04-29 19:36:34 -07:00
|
|
|
|
2018-09-05 21:48:32 -07:00
|
|
|
return key[len(prefix):]
|
|
|
|
}
|
|
|
|
|
2019-02-01 17:03:09 -08:00
|
|
|
// wrapping types.PrefixEndBytes
|
2018-09-05 21:48:32 -07:00
|
|
|
func cpIncr(bz []byte) []byte {
|
2019-02-01 17:03:09 -08:00
|
|
|
return types.PrefixEndBytes(bz)
|
2018-09-05 21:48:32 -07:00
|
|
|
}
|