cosmos-sdk/store/gaskvstore.go

150 lines
3.4 KiB
Go
Raw Normal View History

2018-05-08 13:03:42 -07:00
package store
import (
sdk "github.com/cosmos/cosmos-sdk/types"
)
2018-05-08 15:27:41 -07:00
const (
HasCost = 10
ReadCostFlat = 10
ReadCostPerByte = 1
WriteCostFlat = 10
WriteCostPerByte = 10
)
2018-05-08 13:03:42 -07:00
// gasKVStore applies gas tracking to an underlying kvstore
type gasKVStore struct {
gasMeter sdk.GasMeter
parent KVStore
}
// nolint
func NewGasKVStore(gasMeter sdk.GasMeter, parent KVStore) *gasKVStore {
kvs := &gasKVStore{
gasMeter: gasMeter,
parent: parent,
}
return kvs
}
// Implements Store.
func (gi *gasKVStore) GetStoreType() StoreType {
return gi.parent.GetStoreType()
}
// Implements KVStore.
func (gi *gasKVStore) Get(key []byte) (value []byte) {
2018-05-08 15:27:41 -07:00
gi.gasMeter.ConsumeGas(ReadCostFlat, "GetFlat")
value = gi.parent.Get(key)
// TODO overflow-safe math?
gi.gasMeter.ConsumeGas(ReadCostPerByte*sdk.Gas(len(value)), "ReadPerByte")
return value
2018-05-08 13:03:42 -07:00
}
// Implements KVStore.
func (gi *gasKVStore) Set(key []byte, value []byte) {
2018-05-08 15:27:41 -07:00
gi.gasMeter.ConsumeGas(WriteCostFlat, "SetFlat")
// TODO overflow-safe math?
gi.gasMeter.ConsumeGas(WriteCostPerByte*sdk.Gas(len(value)), "SetPerByte")
2018-05-08 13:03:42 -07:00
gi.parent.Set(key, value)
}
// Implements KVStore.
func (gi *gasKVStore) Has(key []byte) bool {
2018-05-08 15:27:41 -07:00
gi.gasMeter.ConsumeGas(HasCost, "Has")
2018-05-08 13:03:42 -07:00
return gi.parent.Has(key)
}
// Implements KVStore.
func (gi *gasKVStore) Delete(key []byte) {
2018-05-08 15:27:41 -07:00
// No gas costs for deletion
2018-05-08 13:03:42 -07:00
gi.parent.Delete(key)
}
// Implements KVStore.
func (gi *gasKVStore) Iterator(start, end []byte) Iterator {
return gi.iterator(start, end, true)
}
// Implements KVStore.
func (gi *gasKVStore) ReverseIterator(start, end []byte) Iterator {
return gi.iterator(start, end, false)
}
// Implements KVStore.
func (gi *gasKVStore) SubspaceIterator(prefix []byte) Iterator {
return gi.iterator(prefix, sdk.PrefixEndBytes(prefix), true)
}
// Implements KVStore.
func (gi *gasKVStore) ReverseSubspaceIterator(prefix []byte) Iterator {
return gi.iterator(prefix, sdk.PrefixEndBytes(prefix), false)
}
// Implements KVStore.
func (gi *gasKVStore) CacheWrap() CacheWrap {
2018-05-08 15:27:41 -07:00
panic("you cannot CacheWrap a GasKVStore")
2018-05-08 13:03:42 -07:00
}
func (gi *gasKVStore) iterator(start, end []byte, ascending bool) Iterator {
var parent Iterator
if ascending {
parent = gi.parent.Iterator(start, end)
} else {
parent = gi.parent.ReverseIterator(start, end)
}
2018-05-08 15:27:41 -07:00
return newGasIterator(gi.gasMeter, parent)
}
type gasIterator struct {
gasMeter sdk.GasMeter
parent Iterator
}
func newGasIterator(gasMeter sdk.GasMeter, parent Iterator) Iterator {
return &gasIterator{
gasMeter: gasMeter,
parent: parent,
}
}
// Implements Iterator.
func (g *gasIterator) Domain() (start []byte, end []byte) {
return g.parent.Domain()
}
// Implements Iterator.
func (g *gasIterator) Valid() bool {
return g.parent.Valid()
}
/*
TODO
Not quite sure what to charge for here. Depends on underlying retrieval model.
Could charge for Next(), and Key()/Value() are free, but want to have value-size-proportional gas.
*/
// Implements Iterator.
func (g *gasIterator) Next() {
g.parent.Next()
}
// Implements Iterator.
func (g *gasIterator) Key() (key []byte) {
return g.parent.Key()
}
// Implements Iterator.
func (g *gasIterator) Value() (value []byte) {
value = g.parent.Value()
g.gasMeter.ConsumeGas(ReadCostFlat, "ValueFlat")
g.gasMeter.ConsumeGas(ReadCostPerByte*sdk.Gas(len(value)), "ValuePerByte")
return value
}
// Implements Iterator.
func (g *gasIterator) Close() {
g.parent.Close()
2018-05-08 13:03:42 -07:00
}