Roy Crihfield
761d60acdf
The Geth `core/state` and `trie` packages underwent a big refactor between `v1.11.6` and `1.13.14`. This code, which was adapted from those, needed corresponding updates. To do this I applied the diff patches from Geth directly where possible and in some places had to clone new parts of the Geth code and adapt them. In order to make this process as straightforward as possible in the future, I've attempted to minimize the number of changes vs. Geth and added some documentation in the `trie_by_cid` package. Reviewed-on: #5
163 lines
3.9 KiB
Go
163 lines
3.9 KiB
Go
// Copyright 2017 The go-ethereum Authors
|
|
// This file is part of the go-ethereum library.
|
|
//
|
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package trienode
|
|
|
|
import (
|
|
"errors"
|
|
"sync"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/crypto"
|
|
"github.com/ethereum/go-ethereum/ethdb"
|
|
"github.com/ethereum/go-ethereum/rlp"
|
|
)
|
|
|
|
// ProofSet stores a set of trie nodes. It implements trie.Database and can also
|
|
// act as a cache for another trie.Database.
|
|
type ProofSet struct {
|
|
nodes map[string][]byte
|
|
order []string
|
|
|
|
dataSize int
|
|
lock sync.RWMutex
|
|
}
|
|
|
|
// NewProofSet creates an empty node set
|
|
func NewProofSet() *ProofSet {
|
|
return &ProofSet{
|
|
nodes: make(map[string][]byte),
|
|
}
|
|
}
|
|
|
|
// Put stores a new node in the set
|
|
func (db *ProofSet) Put(key []byte, value []byte) error {
|
|
db.lock.Lock()
|
|
defer db.lock.Unlock()
|
|
|
|
if _, ok := db.nodes[string(key)]; ok {
|
|
return nil
|
|
}
|
|
keystr := string(key)
|
|
|
|
db.nodes[keystr] = common.CopyBytes(value)
|
|
db.order = append(db.order, keystr)
|
|
db.dataSize += len(value)
|
|
|
|
return nil
|
|
}
|
|
|
|
// Delete removes a node from the set
|
|
func (db *ProofSet) Delete(key []byte) error {
|
|
db.lock.Lock()
|
|
defer db.lock.Unlock()
|
|
|
|
delete(db.nodes, string(key))
|
|
return nil
|
|
}
|
|
|
|
// Get returns a stored node
|
|
func (db *ProofSet) Get(key []byte) ([]byte, error) {
|
|
db.lock.RLock()
|
|
defer db.lock.RUnlock()
|
|
|
|
if entry, ok := db.nodes[string(key)]; ok {
|
|
return entry, nil
|
|
}
|
|
return nil, errors.New("not found")
|
|
}
|
|
|
|
// Has returns true if the node set contains the given key
|
|
func (db *ProofSet) Has(key []byte) (bool, error) {
|
|
_, err := db.Get(key)
|
|
return err == nil, nil
|
|
}
|
|
|
|
// KeyCount returns the number of nodes in the set
|
|
func (db *ProofSet) KeyCount() int {
|
|
db.lock.RLock()
|
|
defer db.lock.RUnlock()
|
|
|
|
return len(db.nodes)
|
|
}
|
|
|
|
// DataSize returns the aggregated data size of nodes in the set
|
|
func (db *ProofSet) DataSize() int {
|
|
db.lock.RLock()
|
|
defer db.lock.RUnlock()
|
|
|
|
return db.dataSize
|
|
}
|
|
|
|
// List converts the node set to a ProofList
|
|
func (db *ProofSet) List() ProofList {
|
|
db.lock.RLock()
|
|
defer db.lock.RUnlock()
|
|
|
|
var values ProofList
|
|
for _, key := range db.order {
|
|
values = append(values, db.nodes[key])
|
|
}
|
|
return values
|
|
}
|
|
|
|
// Store writes the contents of the set to the given database
|
|
func (db *ProofSet) Store(target ethdb.KeyValueWriter) {
|
|
db.lock.RLock()
|
|
defer db.lock.RUnlock()
|
|
|
|
for key, value := range db.nodes {
|
|
target.Put([]byte(key), value)
|
|
}
|
|
}
|
|
|
|
// ProofList stores an ordered list of trie nodes. It implements ethdb.KeyValueWriter.
|
|
type ProofList []rlp.RawValue
|
|
|
|
// Store writes the contents of the list to the given database
|
|
func (n ProofList) Store(db ethdb.KeyValueWriter) {
|
|
for _, node := range n {
|
|
db.Put(crypto.Keccak256(node), node)
|
|
}
|
|
}
|
|
|
|
// Set converts the node list to a ProofSet
|
|
func (n ProofList) Set() *ProofSet {
|
|
db := NewProofSet()
|
|
n.Store(db)
|
|
return db
|
|
}
|
|
|
|
// Put stores a new node at the end of the list
|
|
func (n *ProofList) Put(key []byte, value []byte) error {
|
|
*n = append(*n, value)
|
|
return nil
|
|
}
|
|
|
|
// Delete panics as there's no reason to remove a node from the list.
|
|
func (n *ProofList) Delete(key []byte) error {
|
|
panic("not supported")
|
|
}
|
|
|
|
// DataSize returns the aggregated data size of nodes in the list
|
|
func (n ProofList) DataSize() int {
|
|
var size int
|
|
for _, node := range n {
|
|
size += len(node)
|
|
}
|
|
return size
|
|
}
|