cosmos-sdk/store/iavl/store.go

418 lines
9.9 KiB
Go
Raw Normal View History

2019-02-01 17:03:09 -08:00
package iavl
2017-11-29 03:08:32 -08:00
import (
"fmt"
"io"
2017-12-10 00:24:55 -08:00
"sync"
2017-11-29 04:19:41 -08:00
2017-11-29 03:08:32 -08:00
"github.com/tendermint/iavl"
abci "github.com/tendermint/tendermint/abci/types"
"github.com/tendermint/tendermint/crypto/merkle"
cmn "github.com/tendermint/tendermint/libs/common"
dbm "github.com/tendermint/tendermint/libs/db"
2017-11-29 03:08:32 -08:00
2019-02-01 17:03:09 -08:00
"github.com/cosmos/cosmos-sdk/store/cachekv"
2019-02-05 10:39:22 -08:00
"github.com/cosmos/cosmos-sdk/store/errors"
2019-02-01 17:03:09 -08:00
"github.com/cosmos/cosmos-sdk/store/tracekv"
2019-02-05 10:39:22 -08:00
"github.com/cosmos/cosmos-sdk/store/types"
)
const (
2018-07-12 18:20:26 -07:00
defaultIAVLCacheSize = 10000
)
2017-11-29 04:19:41 -08:00
2018-04-18 21:49:24 -07:00
// load the iavl store
2019-02-01 17:03:09 -08:00
func LoadStore(db dbm.DB, id types.CommitID, pruning types.PruningOptions) (types.CommitStore, error) {
2018-08-03 16:24:03 -07:00
tree := iavl.NewMutableTree(db, defaultIAVLCacheSize)
2018-03-22 21:32:57 -07:00
_, err := tree.LoadVersion(id.Version)
if err != nil {
return nil, err
}
2019-02-01 17:03:09 -08:00
iavl := UnsafeNewStore(tree, int64(0), int64(0))
2018-07-12 18:20:26 -07:00
iavl.SetPruning(pruning)
return iavl, nil
}
//----------------------------------------
2019-02-01 17:03:09 -08:00
var _ types.KVStore = (*Store)(nil)
var _ types.CommitStore = (*Store)(nil)
var _ types.Queryable = (*Store)(nil)
2017-11-29 04:19:41 -08:00
2019-02-01 17:03:09 -08:00
// Store Implements types.KVStore and CommitStore.
type Store struct {
2017-12-09 12:35:51 -08:00
// The underlying tree.
2018-08-03 16:24:03 -07:00
tree *iavl.MutableTree
2017-11-29 05:07:39 -08:00
2017-12-09 12:35:51 -08:00
// How many old versions we hold onto.
2018-07-12 18:20:26 -07:00
// A value of 0 means keep no recent states.
2018-06-26 13:01:45 -07:00
numRecent int64
2018-07-12 18:20:26 -07:00
// This is the distance between state-sync waypoint states to be stored.
2018-07-02 15:17:45 -07:00
// See https://github.com/tendermint/tendermint/issues/828
2018-07-12 18:20:26 -07:00
// A value of 1 means store every state.
// A value of 0 means store no waypoints. (node cannot assist in state-sync)
2018-07-02 15:17:45 -07:00
// By default this value should be set the same across all nodes,
2018-07-12 18:20:26 -07:00
// so that nodes can know the waypoints their peers store.
2018-06-26 13:01:45 -07:00
storeEvery int64
2017-11-29 04:19:41 -08:00
}
2017-12-09 12:35:51 -08:00
// CONTRACT: tree should be fully loaded.
2018-08-31 15:22:37 -07:00
// nolint: unparam
2019-02-01 17:03:09 -08:00
func UnsafeNewStore(tree *iavl.MutableTree, numRecent int64, storeEvery int64) *Store {
st := &Store{
2017-12-09 12:35:51 -08:00
tree: tree,
2018-06-26 13:01:45 -07:00
numRecent: numRecent,
storeEvery: storeEvery,
2017-11-29 05:07:39 -08:00
}
2017-12-09 12:35:51 -08:00
return st
2017-11-29 05:07:39 -08:00
}
// Implements Committer.
2019-02-01 17:03:09 -08:00
func (st *Store) Commit() types.CommitID {
2017-12-09 12:35:51 -08:00
// Save a new version.
hash, version, err := st.tree.SaveVersion()
2017-11-29 04:19:41 -08:00
if err != nil {
2017-12-09 12:35:51 -08:00
// TODO: Do we want to extend Commit to allow returning errors?
2017-11-29 04:19:41 -08:00
panic(err)
}
2018-07-12 18:20:26 -07:00
// Release an old version of history, if not a sync waypoint.
2018-07-02 15:17:45 -07:00
previous := version - 1
if st.numRecent < previous {
toRelease := previous - st.numRecent
if st.storeEvery == 0 || toRelease%st.storeEvery != 0 {
2018-06-26 13:01:45 -07:00
err := st.tree.DeleteVersion(toRelease)
2018-07-12 18:20:26 -07:00
if err != nil && err.(cmn.Error).Data() != iavl.ErrVersionDoesNotExist {
2018-06-26 13:01:45 -07:00
panic(err)
}
}
2017-11-29 04:19:41 -08:00
}
2019-02-01 17:03:09 -08:00
return types.CommitID{
2017-12-09 12:35:51 -08:00
Version: version,
2017-11-29 04:19:41 -08:00
Hash: hash,
}
2017-11-29 03:08:32 -08:00
}
// Implements Committer.
2019-02-01 17:03:09 -08:00
func (st *Store) LastCommitID() types.CommitID {
return types.CommitID{
2018-09-09 08:13:17 -07:00
Version: st.tree.Version(),
Hash: st.tree.Hash(),
}
}
2018-07-12 18:20:26 -07:00
// Implements Committer.
2019-02-01 17:03:09 -08:00
func (st *Store) SetPruning(opt types.PruningOptions) {
st.numRecent = opt.KeepRecent()
st.storeEvery = opt.KeepEvery()
2018-07-12 18:20:26 -07:00
}
// VersionExists returns whether or not a given version is stored.
2019-02-01 17:03:09 -08:00
func (st *Store) VersionExists(version int64) bool {
2018-06-26 16:26:24 -07:00
return st.tree.VersionExists(version)
}
// Implements Store.
2019-02-01 17:03:09 -08:00
func (st *Store) GetStoreType() types.StoreType {
return types.StoreTypeIAVL
}
// Implements Store.
2019-02-01 17:03:09 -08:00
func (st *Store) CacheWrap() types.CacheWrap {
return cachekv.NewStore(st)
2017-12-10 00:24:55 -08:00
}
// CacheWrapWithTrace implements the Store interface.
2019-02-01 17:03:09 -08:00
func (st *Store) CacheWrapWithTrace(w io.Writer, tc types.TraceContext) types.CacheWrap {
return cachekv.NewStore(tracekv.NewStore(st, w, tc))
}
2019-02-01 17:03:09 -08:00
// Implements types.KVStore.
func (st *Store) Set(key, value []byte) {
2019-02-05 10:39:22 -08:00
types.AssertValidValue(value)
2017-12-09 12:35:51 -08:00
st.tree.Set(key, value)
2017-11-29 05:07:39 -08:00
}
2019-02-01 17:03:09 -08:00
// Implements types.KVStore.
func (st *Store) Get(key []byte) (value []byte) {
2017-12-09 12:35:51 -08:00
_, v := st.tree.Get(key)
2017-12-11 23:30:44 -08:00
return v
2017-11-29 05:07:39 -08:00
}
2019-02-01 17:03:09 -08:00
// Implements types.KVStore.
func (st *Store) Has(key []byte) (exists bool) {
2017-12-09 12:35:51 -08:00
return st.tree.Has(key)
2017-11-29 05:07:39 -08:00
}
2019-02-01 17:03:09 -08:00
// Implements types.KVStore.
func (st *Store) Delete(key []byte) {
2017-12-11 23:30:44 -08:00
st.tree.Remove(key)
2017-11-29 05:07:39 -08:00
}
2019-02-01 17:03:09 -08:00
// Implements types.KVStore.
func (st *Store) Iterator(start, end []byte) types.Iterator {
2018-08-03 16:24:03 -07:00
return newIAVLIterator(st.tree.ImmutableTree, start, end, true)
}
2019-02-01 17:03:09 -08:00
// Implements types.KVStore.
func (st *Store) ReverseIterator(start, end []byte) types.Iterator {
2018-08-03 16:24:03 -07:00
return newIAVLIterator(st.tree.ImmutableTree, start, end, false)
2018-04-01 09:00:28 -07:00
}
// Handle gatest the latest height, if height is 0
2018-08-03 16:24:03 -07:00
func getHeight(tree *iavl.MutableTree, req abci.RequestQuery) int64 {
height := req.Height
if height == 0 {
2018-09-09 08:13:17 -07:00
latest := tree.Version()
if tree.VersionExists(latest - 1) {
height = latest - 1
} else {
height = latest
}
}
return height
}
2018-01-30 10:37:03 -08:00
// Query implements ABCI interface, allows queries
//
// by default we will return from (latest height -1),
// as we will have merkle proofs immediately (header height = data height + 1)
// If latest-1 is not present, use latest (which must be present)
// if you care to have the latest data to see a tx results, you must
// explicitly set the height you want to see
2019-02-01 17:03:09 -08:00
func (st *Store) Query(req abci.RequestQuery) (res abci.ResponseQuery) {
2018-01-30 10:37:03 -08:00
if len(req.Data) == 0 {
msg := "Query cannot be zero length"
2019-02-05 10:39:22 -08:00
return errors.ErrTxDecode(msg).QueryResult()
2018-01-30 10:37:03 -08:00
}
tree := st.tree
// store the height we chose in the response, with 0 being changed to the
// latest height
res.Height = getHeight(tree, req)
2018-01-30 10:37:03 -08:00
switch req.Path {
case "/key": // get by key
key := req.Data // data holds the key bytes
2018-01-30 10:37:03 -08:00
res.Key = key
2018-07-12 18:20:26 -07:00
if !st.VersionExists(res.Height) {
res.Log = cmn.ErrorWrap(iavl.ErrVersionDoesNotExist, "").Error()
break
}
2018-01-30 10:37:03 -08:00
if req.Prove {
2018-11-04 19:42:11 -08:00
value, proof, err := tree.GetVersionedWithProof(key, res.Height)
2018-01-30 10:37:03 -08:00
if err != nil {
res.Log = err.Error()
break
}
if proof == nil {
// Proof == nil implies that the store is empty.
if value != nil {
panic("unexpected value for an empty proof")
}
}
if value != nil {
// value was found
res.Value = value
res.Proof = &merkle.Proof{Ops: []merkle.ProofOp{iavl.NewIAVLValueOp(key, proof).ProofOp()}}
} else {
// value wasn't found
res.Value = nil
res.Proof = &merkle.Proof{Ops: []merkle.ProofOp{iavl.NewIAVLAbsenceOp(key, proof).ProofOp()}}
}
2018-01-30 10:37:03 -08:00
} else {
_, res.Value = tree.GetVersioned(key, res.Height)
2018-01-30 10:37:03 -08:00
}
case "/subspace":
2019-02-01 17:03:09 -08:00
var KVs []types.KVPair
subspace := req.Data
res.Key = subspace
2019-02-05 10:39:22 -08:00
iterator := types.KVStorePrefixIterator(st, subspace)
for ; iterator.Valid(); iterator.Next() {
2019-02-01 17:03:09 -08:00
KVs = append(KVs, types.KVPair{Key: iterator.Key(), Value: iterator.Value()})
}
iterator.Close()
res.Value = cdc.MustMarshalBinaryLengthPrefixed(KVs)
2018-01-30 10:37:03 -08:00
default:
msg := fmt.Sprintf("Unexpected Query path: %v", req.Path)
2019-02-05 10:39:22 -08:00
return errors.ErrUnknownRequest(msg).QueryResult()
2018-01-30 10:37:03 -08:00
}
2018-01-30 10:37:03 -08:00
return
}
2017-12-09 12:35:51 -08:00
//----------------------------------------
2019-02-01 17:03:09 -08:00
// Implements types.Iterator.
type iavlIterator struct {
2017-12-10 00:24:55 -08:00
// Underlying store
2018-08-03 16:24:03 -07:00
tree *iavl.ImmutableTree
2017-12-10 00:24:55 -08:00
// Domain
start, end []byte
// Iteration order
ascending bool
// Channel to push iteration values.
2017-12-17 18:09:39 -08:00
iterCh chan cmn.KVPair
2017-12-10 00:24:55 -08:00
// Close this to release goroutine.
quitCh chan struct{}
// Close this to signal that state is initialized.
initCh chan struct{}
//----------------------------------------
// What follows are mutable state.
mtx sync.Mutex
invalid bool // True once, true forever
key []byte // The current key
value []byte // The current value
}
2019-02-01 17:03:09 -08:00
var _ types.Iterator = (*iavlIterator)(nil)
2017-12-10 00:24:55 -08:00
// newIAVLIterator will create a new iavlIterator.
// CONTRACT: Caller must release the iavlIterator, as each one creates a new
// goroutine.
2018-08-03 16:24:03 -07:00
func newIAVLIterator(tree *iavl.ImmutableTree, start, end []byte, ascending bool) *iavlIterator {
2017-12-11 23:30:44 -08:00
iter := &iavlIterator{
tree: tree,
2019-02-05 10:39:22 -08:00
start: types.Cp(start),
end: types.Cp(end),
2017-12-10 00:24:55 -08:00
ascending: ascending,
iterCh: make(chan cmn.KVPair), // Set capacity > 0?
2017-12-10 00:24:55 -08:00
quitCh: make(chan struct{}),
initCh: make(chan struct{}),
}
2017-12-11 23:30:44 -08:00
go iter.iterateRoutine()
go iter.initRoutine()
return iter
2017-12-10 00:24:55 -08:00
}
// Run this to funnel items from the tree to iterCh.
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) iterateRoutine() {
iter.tree.IterateRange(
iter.start, iter.end, iter.ascending,
2017-12-10 00:24:55 -08:00
func(key, value []byte) bool {
select {
2017-12-11 23:30:44 -08:00
case <-iter.quitCh:
2017-12-10 00:24:55 -08:00
return true // done with iteration.
case iter.iterCh <- cmn.KVPair{Key: key, Value: value}:
2017-12-10 00:24:55 -08:00
return false // yay.
}
},
)
2017-12-11 23:30:44 -08:00
close(iter.iterCh) // done.
2017-12-10 00:24:55 -08:00
}
// Run this to fetch the first item.
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) initRoutine() {
iter.receiveNext()
close(iter.initCh)
2017-12-10 00:24:55 -08:00
}
2019-02-01 17:03:09 -08:00
// Implements types.Iterator.
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) Domain() (start, end []byte) {
return iter.start, iter.end
}
2019-02-01 17:03:09 -08:00
// Implements types.Iterator.
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) Valid() bool {
iter.waitInit()
iter.mtx.Lock()
2017-12-10 00:24:55 -08:00
validity := !iter.invalid
iter.mtx.Unlock()
return validity
}
2019-02-01 17:03:09 -08:00
// Implements types.Iterator.
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) Next() {
iter.waitInit()
iter.mtx.Lock()
iter.assertIsValid(true)
2017-12-10 00:24:55 -08:00
2017-12-11 23:30:44 -08:00
iter.receiveNext()
iter.mtx.Unlock()
}
2019-02-01 17:03:09 -08:00
// Implements types.Iterator.
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) Key() []byte {
iter.waitInit()
iter.mtx.Lock()
iter.assertIsValid(true)
2017-12-10 00:24:55 -08:00
key := iter.key
iter.mtx.Unlock()
return key
}
2019-02-01 17:03:09 -08:00
// Implements types.Iterator.
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) Value() []byte {
iter.waitInit()
iter.mtx.Lock()
iter.assertIsValid(true)
2017-12-10 00:24:55 -08:00
val := iter.value
iter.mtx.Unlock()
return val
}
2019-02-01 17:03:09 -08:00
// Implements types.Iterator.
func (iter *iavlIterator) Close() {
2017-12-11 23:30:44 -08:00
close(iter.quitCh)
2017-12-10 00:24:55 -08:00
}
//----------------------------------------
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) setNext(key, value []byte) {
iter.assertIsValid(false)
2017-12-10 00:24:55 -08:00
2017-12-11 23:30:44 -08:00
iter.key = key
iter.value = value
2017-12-10 00:24:55 -08:00
}
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) setInvalid() {
iter.assertIsValid(false)
2017-12-10 00:24:55 -08:00
2017-12-11 23:30:44 -08:00
iter.invalid = true
2017-12-10 00:24:55 -08:00
}
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) waitInit() {
<-iter.initCh
2017-12-10 00:24:55 -08:00
}
2017-12-11 23:30:44 -08:00
func (iter *iavlIterator) receiveNext() {
kvPair, ok := <-iter.iterCh
2017-12-10 00:24:55 -08:00
if ok {
2017-12-11 23:30:44 -08:00
iter.setNext(kvPair.Key, kvPair.Value)
2017-12-10 00:24:55 -08:00
} else {
2017-12-11 23:30:44 -08:00
iter.setInvalid()
2017-12-10 00:24:55 -08:00
}
}
// assertIsValid panics if the iterator is invalid. If unlockMutex is true,
// it also unlocks the mutex before panicing, to prevent deadlocks in code that
// recovers from panics
func (iter *iavlIterator) assertIsValid(unlockMutex bool) {
2017-12-11 23:30:44 -08:00
if iter.invalid {
if unlockMutex {
iter.mtx.Unlock()
}
2017-12-10 00:24:55 -08:00
panic("invalid iterator")
}
}