Merge pull request #930 from nspcc-dev/fix/keys
Cache storage operations across same block tx executions
This commit is contained in:
commit
5ec70b9fc2
6 changed files with 246 additions and 20 deletions
|
@ -1147,7 +1147,16 @@ func (bc *Blockchain) GetStorageItem(scripthash util.Uint160, key []byte) *state
|
||||||
|
|
||||||
// GetStorageItems returns all storage items for a given scripthash.
|
// GetStorageItems returns all storage items for a given scripthash.
|
||||||
func (bc *Blockchain) GetStorageItems(hash util.Uint160) (map[string]*state.StorageItem, error) {
|
func (bc *Blockchain) GetStorageItems(hash util.Uint160) (map[string]*state.StorageItem, error) {
|
||||||
return bc.dao.GetStorageItems(hash)
|
siMap, err := bc.dao.GetStorageItems(hash)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
m := make(map[string]*state.StorageItem)
|
||||||
|
for i := range siMap {
|
||||||
|
val := siMap[i].StorageItem
|
||||||
|
m[string(siMap[i].Key)] = &val
|
||||||
|
}
|
||||||
|
return m, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetBlock returns a Block by the given hash.
|
// GetBlock returns a Block by the given hash.
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
package dao
|
package dao
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"errors"
|
"errors"
|
||||||
|
"sort"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/io"
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
||||||
|
@ -18,6 +20,7 @@ type Cached struct {
|
||||||
unspents map[util.Uint256]*state.UnspentCoin
|
unspents map[util.Uint256]*state.UnspentCoin
|
||||||
balances map[util.Uint160]*state.NEP5Balances
|
balances map[util.Uint160]*state.NEP5Balances
|
||||||
transfers map[util.Uint160]map[uint32]*state.NEP5TransferLog
|
transfers map[util.Uint160]map[uint32]*state.NEP5TransferLog
|
||||||
|
storage *itemCache
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewCached returns new Cached wrapping around given backing store.
|
// NewCached returns new Cached wrapping around given backing store.
|
||||||
|
@ -27,7 +30,16 @@ func NewCached(d DAO) *Cached {
|
||||||
unspents := make(map[util.Uint256]*state.UnspentCoin)
|
unspents := make(map[util.Uint256]*state.UnspentCoin)
|
||||||
balances := make(map[util.Uint160]*state.NEP5Balances)
|
balances := make(map[util.Uint160]*state.NEP5Balances)
|
||||||
transfers := make(map[util.Uint160]map[uint32]*state.NEP5TransferLog)
|
transfers := make(map[util.Uint160]map[uint32]*state.NEP5TransferLog)
|
||||||
return &Cached{d.GetWrapped(), accs, ctrs, unspents, balances, transfers}
|
st := newItemCache()
|
||||||
|
dao := d.GetWrapped()
|
||||||
|
if cd, ok := dao.(*Cached); ok {
|
||||||
|
for h, m := range cd.storage.st {
|
||||||
|
for _, k := range cd.storage.keys[h] {
|
||||||
|
st.put(h, []byte(k), m[k].State, copyItem(&m[k].StorageItem))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return &Cached{dao, accs, ctrs, unspents, balances, transfers, st}
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetAccountStateOrNew retrieves Account from cache or underlying store
|
// GetAccountStateOrNew retrieves Account from cache or underlying store
|
||||||
|
@ -140,6 +152,10 @@ func (cd *Cached) AppendNEP5Transfer(acc util.Uint160, index uint32, tr *state.N
|
||||||
// Persist flushes all the changes made into the (supposedly) persistent
|
// Persist flushes all the changes made into the (supposedly) persistent
|
||||||
// underlying store.
|
// underlying store.
|
||||||
func (cd *Cached) Persist() (int, error) {
|
func (cd *Cached) Persist() (int, error) {
|
||||||
|
if err := cd.FlushStorage(); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
lowerCache, ok := cd.DAO.(*Cached)
|
lowerCache, ok := cd.DAO.(*Cached)
|
||||||
// If the lower DAO is Cached, we only need to flush the MemCached DB.
|
// If the lower DAO is Cached, we only need to flush the MemCached DB.
|
||||||
// This actually breaks DAO interface incapsulation, but for our current
|
// This actually breaks DAO interface incapsulation, but for our current
|
||||||
|
@ -148,6 +164,9 @@ func (cd *Cached) Persist() (int, error) {
|
||||||
if ok {
|
if ok {
|
||||||
var simpleCache *Simple
|
var simpleCache *Simple
|
||||||
for simpleCache == nil {
|
for simpleCache == nil {
|
||||||
|
if err := lowerCache.FlushStorage(); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
simpleCache, ok = lowerCache.DAO.(*Simple)
|
simpleCache, ok = lowerCache.DAO.(*Simple)
|
||||||
if !ok {
|
if !ok {
|
||||||
lowerCache, ok = cd.DAO.(*Cached)
|
lowerCache, ok = cd.DAO.(*Cached)
|
||||||
|
@ -200,5 +219,137 @@ func (cd *Cached) GetWrapped() DAO {
|
||||||
cd.unspents,
|
cd.unspents,
|
||||||
cd.balances,
|
cd.balances,
|
||||||
cd.transfers,
|
cd.transfers,
|
||||||
|
cd.storage,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FlushStorage flushes storage changes to the underlying DAO.
|
||||||
|
func (cd *Cached) FlushStorage() error {
|
||||||
|
if d, ok := cd.DAO.(*Cached); ok {
|
||||||
|
d.storage.st = cd.storage.st
|
||||||
|
d.storage.keys = cd.storage.keys
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
for h, items := range cd.storage.st {
|
||||||
|
for _, k := range cd.storage.keys[h] {
|
||||||
|
ti := items[k]
|
||||||
|
switch ti.State {
|
||||||
|
case putOp, addOp:
|
||||||
|
err := cd.DAO.PutStorageItem(h, []byte(k), &ti.StorageItem)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
case delOp:
|
||||||
|
err := cd.DAO.DeleteStorageItem(h, []byte(k))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func copyItem(si *state.StorageItem) *state.StorageItem {
|
||||||
|
val := make([]byte, len(si.Value))
|
||||||
|
copy(val, si.Value)
|
||||||
|
return &state.StorageItem{
|
||||||
|
Value: val,
|
||||||
|
IsConst: si.IsConst,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetStorageItem returns StorageItem if it exists in the given store.
|
||||||
|
func (cd *Cached) GetStorageItem(scripthash util.Uint160, key []byte) *state.StorageItem {
|
||||||
|
ti := cd.storage.getItem(scripthash, key)
|
||||||
|
if ti != nil {
|
||||||
|
if ti.State == delOp {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return copyItem(&ti.StorageItem)
|
||||||
|
}
|
||||||
|
|
||||||
|
si := cd.DAO.GetStorageItem(scripthash, key)
|
||||||
|
if si != nil {
|
||||||
|
cd.storage.put(scripthash, key, getOp, si)
|
||||||
|
return copyItem(si)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// PutStorageItem puts given StorageItem for given script with given
|
||||||
|
// key into the given store.
|
||||||
|
func (cd *Cached) PutStorageItem(scripthash util.Uint160, key []byte, si *state.StorageItem) error {
|
||||||
|
item := copyItem(si)
|
||||||
|
ti := cd.storage.getItem(scripthash, key)
|
||||||
|
if ti != nil {
|
||||||
|
if ti.State == delOp || ti.State == getOp {
|
||||||
|
ti.State = putOp
|
||||||
|
}
|
||||||
|
ti.StorageItem = *item
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
op := addOp
|
||||||
|
if it := cd.DAO.GetStorageItem(scripthash, key); it != nil {
|
||||||
|
op = putOp
|
||||||
|
}
|
||||||
|
cd.storage.put(scripthash, key, op, item)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// DeleteStorageItem drops storage item for the given script with the
|
||||||
|
// given key from the store.
|
||||||
|
func (cd *Cached) DeleteStorageItem(scripthash util.Uint160, key []byte) error {
|
||||||
|
ti := cd.storage.getItem(scripthash, key)
|
||||||
|
if ti != nil {
|
||||||
|
ti.State = delOp
|
||||||
|
ti.Value = nil
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
it := cd.DAO.GetStorageItem(scripthash, key)
|
||||||
|
if it != nil {
|
||||||
|
cd.storage.put(scripthash, key, delOp, it)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetStorageItems returns all storage items for a given scripthash.
|
||||||
|
func (cd *Cached) GetStorageItems(hash util.Uint160) ([]StorageItemWithKey, error) {
|
||||||
|
items, err := cd.DAO.GetStorageItems(hash)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
cache := cd.storage.getItems(hash)
|
||||||
|
if len(cache) == 0 {
|
||||||
|
return items, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
result := make([]StorageItemWithKey, 0, len(items))
|
||||||
|
for i := range items {
|
||||||
|
_, ok := cache[string(items[i].Key)]
|
||||||
|
if !ok {
|
||||||
|
result = append(result, items[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sort.Slice(result, func(i, j int) bool { return bytes.Compare(result[i].Key, result[j].Key) == -1 })
|
||||||
|
|
||||||
|
for _, k := range cd.storage.keys[hash] {
|
||||||
|
v := cache[k]
|
||||||
|
if v.State != delOp {
|
||||||
|
val := make([]byte, len(v.StorageItem.Value))
|
||||||
|
copy(val, v.StorageItem.Value)
|
||||||
|
result = append(result, StorageItemWithKey{
|
||||||
|
StorageItem: state.StorageItem{
|
||||||
|
Value: val,
|
||||||
|
IsConst: v.StorageItem.IsConst,
|
||||||
|
},
|
||||||
|
Key: []byte(k),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
|
@ -35,7 +35,7 @@ type DAO interface {
|
||||||
GetNEP5Balances(acc util.Uint160) (*state.NEP5Balances, error)
|
GetNEP5Balances(acc util.Uint160) (*state.NEP5Balances, error)
|
||||||
GetNEP5TransferLog(acc util.Uint160, index uint32) (*state.NEP5TransferLog, error)
|
GetNEP5TransferLog(acc util.Uint160, index uint32) (*state.NEP5TransferLog, error)
|
||||||
GetStorageItem(scripthash util.Uint160, key []byte) *state.StorageItem
|
GetStorageItem(scripthash util.Uint160, key []byte) *state.StorageItem
|
||||||
GetStorageItems(hash util.Uint160) (map[string]*state.StorageItem, error)
|
GetStorageItems(hash util.Uint160) ([]StorageItemWithKey, error)
|
||||||
GetTransaction(hash util.Uint256) (*transaction.Transaction, uint32, error)
|
GetTransaction(hash util.Uint256) (*transaction.Transaction, uint32, error)
|
||||||
GetUnspentCoinState(hash util.Uint256) (*state.UnspentCoin, error)
|
GetUnspentCoinState(hash util.Uint256) (*state.UnspentCoin, error)
|
||||||
GetValidatorState(publicKey *keys.PublicKey) (*state.Validator, error)
|
GetValidatorState(publicKey *keys.PublicKey) (*state.Validator, error)
|
||||||
|
@ -435,9 +435,15 @@ func (dao *Simple) DeleteStorageItem(scripthash util.Uint160, key []byte) error
|
||||||
return dao.Store.Delete(makeStorageItemKey(scripthash, key))
|
return dao.Store.Delete(makeStorageItemKey(scripthash, key))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// StorageItemWithKey is a Key-Value pair together with possible const modifier.
|
||||||
|
type StorageItemWithKey struct {
|
||||||
|
state.StorageItem
|
||||||
|
Key []byte
|
||||||
|
}
|
||||||
|
|
||||||
// GetStorageItems returns all storage items for a given scripthash.
|
// GetStorageItems returns all storage items for a given scripthash.
|
||||||
func (dao *Simple) GetStorageItems(hash util.Uint160) (map[string]*state.StorageItem, error) {
|
func (dao *Simple) GetStorageItems(hash util.Uint160) ([]StorageItemWithKey, error) {
|
||||||
var siMap = make(map[string]*state.StorageItem)
|
var res []StorageItemWithKey
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
saveToMap := func(k, v []byte) {
|
saveToMap := func(k, v []byte) {
|
||||||
|
@ -445,21 +451,22 @@ func (dao *Simple) GetStorageItems(hash util.Uint160) (map[string]*state.Storage
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
r := io.NewBinReaderFromBuf(v)
|
r := io.NewBinReaderFromBuf(v)
|
||||||
si := &state.StorageItem{}
|
var s StorageItemWithKey
|
||||||
si.DecodeBinary(r)
|
s.StorageItem.DecodeBinary(r)
|
||||||
if r.Err != nil {
|
if r.Err != nil {
|
||||||
err = r.Err
|
err = r.Err
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cut prefix and hash.
|
// Cut prefix and hash.
|
||||||
siMap[string(k[21:])] = si
|
s.Key = k[21:]
|
||||||
|
res = append(res, s)
|
||||||
}
|
}
|
||||||
dao.Store.Seek(storage.AppendPrefix(storage.STStorage, hash.BytesLE()), saveToMap)
|
dao.Store.Seek(storage.AppendPrefix(storage.STStorage, hash.BytesLE()), saveToMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return siMap, nil
|
return res, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// makeStorageItemKey returns a key used to store StorageItem in the DB.
|
// makeStorageItemKey returns a key used to store StorageItem in the DB.
|
||||||
|
|
57
pkg/core/dao/storage_item.go
Normal file
57
pkg/core/dao/storage_item.go
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
package dao
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
type (
|
||||||
|
itemState int
|
||||||
|
|
||||||
|
trackedItem struct {
|
||||||
|
state.StorageItem
|
||||||
|
State itemState
|
||||||
|
}
|
||||||
|
|
||||||
|
itemCache struct {
|
||||||
|
st map[util.Uint160]map[string]*trackedItem
|
||||||
|
keys map[util.Uint160][]string
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
getOp itemState = 1 << iota
|
||||||
|
delOp
|
||||||
|
addOp
|
||||||
|
putOp
|
||||||
|
)
|
||||||
|
|
||||||
|
func newItemCache() *itemCache {
|
||||||
|
return &itemCache{
|
||||||
|
make(map[util.Uint160]map[string]*trackedItem),
|
||||||
|
make(map[util.Uint160][]string),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *itemCache) put(h util.Uint160, key []byte, op itemState, item *state.StorageItem) {
|
||||||
|
m := c.getItems(h)
|
||||||
|
m[string(key)] = &trackedItem{
|
||||||
|
StorageItem: *item,
|
||||||
|
State: op,
|
||||||
|
}
|
||||||
|
c.keys[h] = append(c.keys[h], string(key))
|
||||||
|
c.st[h] = m
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *itemCache) getItem(h util.Uint160, key []byte) *trackedItem {
|
||||||
|
m := c.getItems(h)
|
||||||
|
return m[string(key)]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *itemCache) getItems(h util.Uint160) map[string]*trackedItem {
|
||||||
|
m, ok := c.st[h]
|
||||||
|
if !ok {
|
||||||
|
return make(map[string]*trackedItem)
|
||||||
|
}
|
||||||
|
return m
|
||||||
|
}
|
|
@ -1,10 +1,10 @@
|
||||||
package core
|
package core
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math"
|
"math"
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
@ -451,17 +451,18 @@ func (ic *interopContext) storageFind(v *vm.VM) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
prefix := string(v.Estack().Pop().Bytes())
|
pref := v.Estack().Pop().Bytes()
|
||||||
siMap, err := ic.dao.GetStorageItems(stc.ScriptHash)
|
siMap, err := ic.dao.GetStorageItems(stc.ScriptHash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
filteredMap := vm.NewMapItem()
|
filteredMap := vm.NewMapItem()
|
||||||
for k, v := range siMap {
|
for i := range siMap {
|
||||||
if strings.HasPrefix(k, prefix) {
|
k := siMap[i].Key
|
||||||
filteredMap.Add(vm.NewByteArrayItem([]byte(k)),
|
if bytes.HasPrefix(k, pref) {
|
||||||
vm.NewByteArrayItem(v.Value))
|
filteredMap.Add(vm.NewByteArrayItem(siMap[i].Key),
|
||||||
|
vm.NewByteArrayItem(siMap[i].Value))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -585,9 +586,10 @@ func (ic *interopContext) contractMigrate(v *vm.VM) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
for k, v := range siMap {
|
for i := range siMap {
|
||||||
v.IsConst = false
|
v := siMap[i].StorageItem
|
||||||
err = ic.dao.PutStorageItem(contract.ScriptHash(), []byte(k), v)
|
siMap[i].IsConst = false
|
||||||
|
err = ic.dao.PutStorageItem(contract.ScriptHash(), siMap[i].Key, &v)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
|
@ -561,8 +561,8 @@ func (ic *interopContext) contractDestroy(v *vm.VM) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
for k := range siMap {
|
for i := range siMap {
|
||||||
_ = ic.dao.DeleteStorageItem(hash, []byte(k))
|
_ = ic.dao.DeleteStorageItem(hash, siMap[i].Key)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
|
Loading…
Reference in a new issue