cosmos-sdk/store/cachekv/mergeiterator.go

252 lines
5.6 KiB
Go
Raw Normal View History

2019-02-01 17:03:09 -08:00
package cachekv
2017-12-11 23:30:44 -08:00
import (
"bytes"
"errors"
2019-02-01 17:03:09 -08:00
"github.com/cosmos/cosmos-sdk/store/types"
)
2017-12-11 23:30:44 -08:00
// cacheMergeIterator merges a parent Iterator and a cache Iterator.
// The cache iterator may return nil keys to signal that an item
// had been deleted (but not deleted in the parent).
// If the cache iterator has the same key as the parent, the
// cache shadows (overrides) the parent.
//
// TODO: Optimize by memoizing.
type cacheMergeIterator struct {
2019-02-01 17:03:09 -08:00
parent types.Iterator
cache types.Iterator
2017-12-11 23:30:44 -08:00
ascending bool
}
2019-02-01 17:03:09 -08:00
var _ types.Iterator = (*cacheMergeIterator)(nil)
2017-12-11 23:30:44 -08:00
2019-02-01 17:03:09 -08:00
func newCacheMergeIterator(parent, cache types.Iterator, ascending bool) *cacheMergeIterator {
2017-12-11 23:30:44 -08:00
iter := &cacheMergeIterator{
parent: parent,
cache: cache,
ascending: ascending,
}
2017-12-11 23:30:44 -08:00
return iter
}
// Domain implements Iterator.
// If the domains are different, returns the union.
func (iter *cacheMergeIterator) Domain() (start, end []byte) {
startP, endP := iter.parent.Domain()
startC, endC := iter.cache.Domain()
2017-12-11 23:30:44 -08:00
if iter.compare(startP, startC) < 0 {
start = startP
} else {
start = startC
}
2017-12-11 23:30:44 -08:00
if iter.compare(endP, endC) < 0 {
end = endC
} else {
end = endP
}
2017-12-11 23:30:44 -08:00
return start, end
}
// Valid implements Iterator.
func (iter *cacheMergeIterator) Valid() bool {
return iter.skipUntilExistsOrInvalid()
2017-12-11 23:30:44 -08:00
}
// Next implements Iterator
func (iter *cacheMergeIterator) Next() {
iter.skipUntilExistsOrInvalid()
iter.assertValid()
// If parent is invalid, get the next cache item.
if !iter.parent.Valid() {
iter.cache.Next()
return
}
// If cache is invalid, get the next parent item.
if !iter.cache.Valid() {
iter.parent.Next()
return
}
// Both are valid. Compare keys.
keyP, keyC := iter.parent.Key(), iter.cache.Key()
switch iter.compare(keyP, keyC) {
2017-12-11 23:30:44 -08:00
case -1: // parent < cache
iter.parent.Next()
case 0: // parent == cache
iter.parent.Next()
iter.cache.Next()
case 1: // parent > cache
iter.cache.Next()
}
}
// Key implements Iterator
func (iter *cacheMergeIterator) Key() []byte {
iter.skipUntilExistsOrInvalid()
iter.assertValid()
// If parent is invalid, get the cache key.
if !iter.parent.Valid() {
return iter.cache.Key()
}
// If cache is invalid, get the parent key.
if !iter.cache.Valid() {
return iter.parent.Key()
}
// Both are valid. Compare keys.
keyP, keyC := iter.parent.Key(), iter.cache.Key()
2017-12-11 23:30:44 -08:00
cmp := iter.compare(keyP, keyC)
switch cmp {
case -1: // parent < cache
return keyP
case 0: // parent == cache
return keyP
case 1: // parent > cache
return keyC
default:
panic("invalid compare result")
}
}
// Value implements Iterator
func (iter *cacheMergeIterator) Value() []byte {
iter.skipUntilExistsOrInvalid()
iter.assertValid()
// If parent is invalid, get the cache value.
if !iter.parent.Valid() {
return iter.cache.Value()
}
// If cache is invalid, get the parent value.
if !iter.cache.Valid() {
return iter.parent.Value()
}
// Both are valid. Compare keys.
keyP, keyC := iter.parent.Key(), iter.cache.Key()
2017-12-11 23:30:44 -08:00
cmp := iter.compare(keyP, keyC)
switch cmp {
case -1: // parent < cache
return iter.parent.Value()
case 0: // parent == cache
return iter.cache.Value()
case 1: // parent > cache
return iter.cache.Value()
default:
panic("invalid comparison result")
}
}
// Close implements Iterator
func (iter *cacheMergeIterator) Close() error {
if err := iter.parent.Close(); err != nil {
return err
}
return iter.cache.Close()
2017-12-11 23:30:44 -08:00
}
// Error returns an error if the cacheMergeIterator is invalid defined by the
// Valid method.
func (iter *cacheMergeIterator) Error() error {
if !iter.Valid() {
return errors.New("invalid cacheMergeIterator")
}
return nil
}
// If not valid, panics.
// NOTE: May have side-effect of iterating over cache.
func (iter *cacheMergeIterator) assertValid() {
if err := iter.Error(); err != nil {
panic(err)
}
}
2017-12-11 23:30:44 -08:00
// Like bytes.Compare but opposite if not ascending.
func (iter *cacheMergeIterator) compare(a, b []byte) int {
if iter.ascending {
return bytes.Compare(a, b)
}
2018-04-18 21:49:24 -07:00
return bytes.Compare(a, b) * -1
2017-12-11 23:30:44 -08:00
}
// Skip all delete-items from the cache w/ `key < until`. After this function,
// current cache item is a non-delete-item, or `until <= key`.
// If the current cache item is not a delete item, does nothing.
// If `until` is nil, there is no limit, and cache may end up invalid.
2017-12-11 23:30:44 -08:00
// CONTRACT: cache is valid.
func (iter *cacheMergeIterator) skipCacheDeletes(until []byte) {
for iter.cache.Valid() &&
iter.cache.Value() == nil &&
(until == nil || iter.compare(iter.cache.Key(), until) < 0) {
2017-12-11 23:30:44 -08:00
iter.cache.Next()
}
}
// Fast forwards cache (or parent+cache in case of deleted items) until current
// item exists, or until iterator becomes invalid.
// Returns whether the iterator is valid.
func (iter *cacheMergeIterator) skipUntilExistsOrInvalid() bool {
2017-12-11 23:30:44 -08:00
for {
// If parent is invalid, fast-forward cache.
if !iter.parent.Valid() {
iter.skipCacheDeletes(nil)
return iter.cache.Valid()
2017-12-11 23:30:44 -08:00
}
// Parent is valid.
2017-12-11 23:30:44 -08:00
if !iter.cache.Valid() {
return true
}
// Parent is valid, cache is valid.
2017-12-11 23:30:44 -08:00
// Compare parent and cache.
keyP := iter.parent.Key()
keyC := iter.cache.Key()
2017-12-11 23:30:44 -08:00
switch iter.compare(keyP, keyC) {
case -1: // parent < cache.
return true
2017-12-11 23:30:44 -08:00
case 0: // parent == cache.
2017-12-11 23:30:44 -08:00
// Skip over if cache item is a delete.
valueC := iter.cache.Value()
if valueC == nil {
iter.parent.Next()
iter.cache.Next()
2017-12-11 23:30:44 -08:00
continue
}
// Cache is not a delete.
2017-12-11 23:30:44 -08:00
return true // cache exists.
case 1: // cache < parent
2017-12-11 23:30:44 -08:00
// Skip over if cache item is a delete.
valueC := iter.cache.Value()
if valueC == nil {
iter.skipCacheDeletes(keyP)
continue
}
// Cache is not a delete.
return true // cache exists.
2017-12-11 23:30:44 -08:00
}
}
}