mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-09 20:41:20 +00:00
225 lines
4.9 KiB
Go
225 lines
4.9 KiB
Go
package trie
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ugorji/go/codec"
|
|
)
|
|
|
|
var cbor codec.CborHandle
|
|
|
|
// OperatorMarshaller provides all needed primitives to read witness operators from a serialized form.
|
|
type OperatorUnmarshaller struct {
|
|
reader io.Reader
|
|
decoder *codec.Decoder
|
|
}
|
|
|
|
func NewOperatorUnmarshaller(r io.Reader) *OperatorUnmarshaller {
|
|
return &OperatorUnmarshaller{r, codec.NewDecoder(r, &cbor)}
|
|
}
|
|
|
|
func (l *OperatorUnmarshaller) ReadByteArray() ([]byte, error) {
|
|
var buffer []byte
|
|
err := l.decoder.Decode(&buffer)
|
|
if err != nil {
|
|
return []byte{}, err
|
|
}
|
|
|
|
return buffer, nil
|
|
}
|
|
|
|
func (l *OperatorUnmarshaller) ReadHash() (libcommon.Hash, error) {
|
|
var hash libcommon.Hash
|
|
bytesRead, err := l.reader.Read(hash[:])
|
|
if err != nil {
|
|
return hash, err
|
|
}
|
|
if bytesRead != len(hash) {
|
|
return hash, fmt.Errorf("error while reading hash from input. expected to read %d bytes, read only %d", len(hash), bytesRead)
|
|
}
|
|
return hash, nil
|
|
}
|
|
|
|
func (l *OperatorUnmarshaller) ReadUint32() (uint32, error) {
|
|
var value uint32
|
|
if err := l.decoder.Decode(&value); err != nil {
|
|
return 0, err
|
|
}
|
|
return value, nil
|
|
}
|
|
|
|
func (l *OperatorUnmarshaller) ReadByte() (byte, error) {
|
|
values := make([]byte, 1)
|
|
bytesRead, err := l.reader.Read(values)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
if bytesRead < 1 {
|
|
return 0, fmt.Errorf("could not read a byte from a reader")
|
|
}
|
|
return values[0], nil
|
|
}
|
|
|
|
func (l *OperatorUnmarshaller) ReadKey() ([]byte, error) {
|
|
b, err := l.ReadByteArray()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return keyBytesToNibbles(b), nil
|
|
}
|
|
|
|
func (l *OperatorUnmarshaller) ReadUInt64() (uint64, error) {
|
|
var value uint64
|
|
err := l.decoder.Decode(&value)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return value, nil
|
|
|
|
}
|
|
|
|
// OperatorMarshaller is responsible for encoding operators to a stream
|
|
// and collecting stats
|
|
// IMPORTANT: not thread-safe! use from a single thread only
|
|
type OperatorMarshaller struct {
|
|
currentColumn StatsColumn
|
|
encoder *codec.Encoder
|
|
w io.Writer
|
|
stats map[StatsColumn]uint64
|
|
total uint64
|
|
}
|
|
|
|
func NewOperatorMarshaller(w io.Writer) *OperatorMarshaller {
|
|
marshaller := &OperatorMarshaller{w: w, stats: make(map[StatsColumn]uint64)}
|
|
// so we can collect stats
|
|
marshaller.encoder = codec.NewEncoder(marshaller, &cbor)
|
|
return marshaller
|
|
}
|
|
|
|
func (w *OperatorMarshaller) WriteOpCode(opcode OperatorKindCode) error {
|
|
w.WithColumn(ColumnStructure)
|
|
_, err := w.Write([]byte{byte(opcode)})
|
|
return err
|
|
}
|
|
|
|
func (w *OperatorMarshaller) WriteKey(keyNibbles []byte) error {
|
|
w.WithColumn(ColumnLeafKeys)
|
|
return w.encoder.Encode(keyNibblesToBytes(keyNibbles))
|
|
}
|
|
|
|
func (w *OperatorMarshaller) WriteByteValue(value byte) error {
|
|
w.WithColumn(ColumnLeafValues)
|
|
_, err := w.Write([]byte{value})
|
|
return err
|
|
}
|
|
|
|
func (w *OperatorMarshaller) WriteUint64Value(value uint64) error {
|
|
w.WithColumn(ColumnLeafValues)
|
|
return w.encoder.Encode(value)
|
|
}
|
|
|
|
func (w *OperatorMarshaller) WriteByteArrayValue(value []byte) error {
|
|
w.WithColumn(ColumnLeafValues)
|
|
return w.encoder.Encode(value)
|
|
}
|
|
|
|
func (w *OperatorMarshaller) WriteCode(value []byte) error {
|
|
w.WithColumn(ColumnCodes)
|
|
return w.encoder.Encode(value)
|
|
}
|
|
|
|
func (w *OperatorMarshaller) WriteHash(hash libcommon.Hash) error {
|
|
w.WithColumn(ColumnHashes)
|
|
_, err := w.Write(hash[:])
|
|
return err
|
|
}
|
|
|
|
func (w *OperatorMarshaller) Write(p []byte) (int, error) {
|
|
val := w.stats[w.currentColumn]
|
|
|
|
written, err := w.w.Write(p)
|
|
|
|
val += uint64(written)
|
|
w.total += uint64(written)
|
|
|
|
w.stats[w.currentColumn] = val
|
|
|
|
return written, err
|
|
}
|
|
|
|
func (w *OperatorMarshaller) WithColumn(column StatsColumn) *OperatorMarshaller {
|
|
w.currentColumn = column
|
|
return w
|
|
}
|
|
|
|
func (w *OperatorMarshaller) GetStats() *BlockWitnessStats {
|
|
return &BlockWitnessStats{
|
|
witnessSize: w.total,
|
|
stats: w.stats,
|
|
}
|
|
}
|
|
|
|
func keyNibblesToBytes(nibbles []byte) []byte {
|
|
if len(nibbles) < 1 {
|
|
return []byte{}
|
|
}
|
|
if len(nibbles) < 2 {
|
|
return nibbles
|
|
}
|
|
hasTerminator := false
|
|
if nibbles[len(nibbles)-1] == 0x10 {
|
|
nibbles = nibbles[:len(nibbles)-1]
|
|
hasTerminator = true
|
|
}
|
|
|
|
targetLen := len(nibbles)/2 + len(nibbles)%2 + 1
|
|
|
|
result := make([]byte, targetLen)
|
|
nibbleIndex := 0
|
|
result[0] = byte(len(nibbles) % 2) // parity bit
|
|
for i := 1; i < len(result); i++ {
|
|
result[i] = nibbles[nibbleIndex] * 16
|
|
nibbleIndex++
|
|
if nibbleIndex < len(nibbles) {
|
|
result[i] += nibbles[nibbleIndex]
|
|
nibbleIndex++
|
|
}
|
|
}
|
|
if hasTerminator {
|
|
result[0] |= 1 << 1
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func keyBytesToNibbles(b []byte) []byte {
|
|
if len(b) < 1 {
|
|
return []byte{}
|
|
}
|
|
if len(b) < 2 {
|
|
return b
|
|
}
|
|
|
|
hasTerminator := b[0]&(1<<1) != 0
|
|
|
|
targetLen := (len(b)-1)*2 - int(b[0]&1)
|
|
|
|
nibbles := make([]byte, targetLen)
|
|
|
|
nibbleIndex := 0
|
|
for i := 1; i < len(b); i++ {
|
|
nibbles[nibbleIndex] = b[i] / 16
|
|
nibbleIndex++
|
|
if nibbleIndex < len(nibbles) {
|
|
nibbles[nibbleIndex] = b[i] % 16
|
|
nibbleIndex++
|
|
}
|
|
}
|
|
if hasTerminator {
|
|
return append(nibbles, 0x10)
|
|
}
|
|
return nibbles
|
|
}
|