erigon-pulse/common/dbutils/history_index.go
ledgerwatch 6372b6ef5a
refuse to append to index is value is not greater than the last element (#807)
* Make stage7 and stage8 idempotent

* Fix formatting

* fix

* Fix formatting
2020-07-29 11:22:48 +01:00

251 lines
8.3 KiB
Go

package dbutils
import (
"bytes"
"encoding/binary"
"errors"
"fmt"
"github.com/ledgerwatch/turbo-geth/common"
"sort"
"strconv"
)
const (
ItemLen = 3
MaxChunkSize = 1000
)
func NewHistoryIndex() HistoryIndexBytes {
return make(HistoryIndexBytes, 8)
}
func WrapHistoryIndex(b []byte) HistoryIndexBytes {
index := HistoryIndexBytes(b)
if len(index) == 0 {
index = make(HistoryIndexBytes, 8)
}
return index
}
type HistoryIndexBytes []byte
func (hi HistoryIndexBytes) String() string {
var buffer bytes.Buffer
numbers, _, _ := hi.Decode()
for _, n := range numbers {
buffer.WriteString(fmt.Sprintf("%d ", n))
}
return buffer.String()
}
// decode is used for debugging and in tests
func (hi HistoryIndexBytes) Decode() ([]uint64, []bool, error) {
if len(hi) < 8 {
return nil, nil, fmt.Errorf("minimal length of index chunk is %d, got %d", 8, len(hi))
}
if (len(hi)-8)%ItemLen != 0 {
return nil, nil, fmt.Errorf("length of index chunk should be 8 (mod %d), got %d", ItemLen, len(hi))
}
numElements := (len(hi) - 8) / 3
minElement := binary.BigEndian.Uint64(hi[:8])
numbers := make([]uint64, 0, numElements)
sets := make([]bool, 0, numElements)
for i := 8; i < len(hi); i += 3 {
numbers = append(numbers, minElement+(uint64(hi[i]&0x7f)<<16)+(uint64(hi[i+1])<<8)+uint64(hi[i+2]))
sets = append(sets, hi[i]&0x80 != 0)
}
return numbers, sets, nil
}
func (hi HistoryIndexBytes) Append(v uint64, emptyValue bool) HistoryIndexBytes {
if len(hi) < 8 {
panic(fmt.Errorf("minimal length of index chunk is %d, got %d", 8, len(hi)))
}
if (len(hi)-8)%ItemLen != 0 {
panic(fmt.Errorf("length of index chunk should be 8 (mod %d), got %d", ItemLen, len(hi)))
}
numElements := (len(hi) - 8) / 3
var minElement uint64
if numElements == 0 {
minElement = v
binary.BigEndian.PutUint64(hi[:], minElement)
} else {
minElement = binary.BigEndian.Uint64(hi[:8])
// last value
lastIdx := len(hi) - 3
lastElement := minElement + (uint64(hi[lastIdx]&0x7f) << 16) + (uint64(hi[lastIdx+1]) << 8) + uint64(hi[lastIdx+2])
// Do not append if the value is less or equal to the last - to make operation idempotent
if v <= lastElement {
return hi
}
}
if v > minElement+0x7fffff { // Maximum number representable in 23 bits
panic(fmt.Errorf("item %d cannot be placed into the chunk with minElement %d", v, minElement))
}
v -= minElement
if emptyValue {
hi = append(hi, 0x80|byte(v>>16))
} else {
hi = append(hi, byte(v>>16))
}
hi = append(hi, byte(v>>8))
hi = append(hi, byte(v))
return hi
}
func (hi HistoryIndexBytes) Len() int {
if len(hi) < 8 {
panic(fmt.Errorf("minimal length of index chunk is %d, got %d", 8, len(hi)))
}
if (len(hi)-8)%ItemLen != 0 {
panic(fmt.Errorf("length of index chunk should be 8 (mod %d), got %d", ItemLen, len(hi)))
}
return (len(hi) - 8) / 3
}
// Truncate all the timestamps that are strictly greater than the given bound
func (hi HistoryIndexBytes) TruncateGreater(lower uint64) HistoryIndexBytes {
if len(hi) < 8 {
panic(fmt.Errorf("minimal length of index chunk is %d, got %d", 8, len(hi)))
}
if (len(hi)-8)%ItemLen != 0 {
panic(fmt.Errorf("length of index chunk should be 8 (mod %d), got %d", ItemLen, len(hi)))
}
numElements := (len(hi) - 8) / 3
minElement := binary.BigEndian.Uint64(hi[:8])
elements := hi[8:]
// We are looking for the truncation point, i.e. the index of the first element which is strictly greater
// than `lower`. Then, we will use that truncation point to shrink the slice
truncationPoint := sort.Search(numElements, func(i int) bool {
return lower < minElement+(uint64(elements[i*ItemLen]&0x7f)<<16)+(uint64(elements[i*ItemLen+1])<<8)+uint64(elements[i*ItemLen+2])
})
return hi[:8+truncationPoint*ItemLen] // We preserve minElement field and all elements prior to the truncation point
}
// Search looks for the element which is equal or greater of given timestamp
func (hi HistoryIndexBytes) Search(v uint64) (uint64, bool, bool) {
if len(hi) < 8 {
panic(fmt.Errorf("minimal length of index chunk is %d, got %d", 8, len(hi)))
}
if (len(hi)-8)%ItemLen != 0 {
panic(fmt.Errorf("length of index chunk should be 8 (mod %d), got %d", ItemLen, len(hi)))
}
numElements := (len(hi) - 8) / 3
minElement := binary.BigEndian.Uint64(hi[:8])
elements := hi[8:]
idx := sort.Search(numElements, func(i int) bool {
return v <= minElement+(uint64(elements[i*ItemLen]&0x7f)<<16)+(uint64(elements[i*ItemLen+1])<<8)+uint64(elements[i*ItemLen+2])
}) * ItemLen
if idx == len(elements) {
return 0, false, false
}
return minElement +
(uint64(elements[idx]&0x7f) << 16) +
(uint64(elements[idx+1]) << 8) +
uint64(elements[idx+2]),
(elements[idx] & 0x80) != 0, true
}
func (hi HistoryIndexBytes) Key(key []byte) ([]byte, error) {
blockNum, ok := hi.LastElement()
if !ok {
return nil, errors.New("empty index")
}
return IndexChunkKey(key, blockNum), nil
}
func (hi HistoryIndexBytes) LastElement() (uint64, bool) {
if len(hi) < 8 {
panic(fmt.Errorf("minimal length of index chunk is %d, got %d", 8, len(hi)))
}
if (len(hi)-8)%ItemLen != 0 {
panic(fmt.Errorf("length of index chunk should be 8 (mod %d), got %d", ItemLen, len(hi)))
}
numElements := (len(hi) - 8) / 3
if numElements == 0 {
return 0, false
}
minElement := binary.BigEndian.Uint64(hi[:8])
idx := 8 + ItemLen*(numElements-1)
return minElement + (uint64(hi[idx]&0x7f) << 16) + (uint64(hi[idx+1]) << 8) + uint64(hi[idx+2]), true
}
func CurrentChunkKey(key []byte) []byte {
return IndexChunkKey(key, ^uint64(0))
}
//
func IndexChunkKey(key []byte, blockNumber uint64) []byte {
var blockNumBytes []byte // make([]byte, len(key)+8)
switch len(key) {
//plain state, accounts
case common.AddressLength:
blockNumBytes = make([]byte, common.AddressLength+8)
copy(blockNumBytes, key)
binary.BigEndian.PutUint64(blockNumBytes[common.AddressLength:], blockNumber)
//hashed state, accounts
case common.HashLength:
blockNumBytes = make([]byte, common.HashLength+8)
copy(blockNumBytes, key)
binary.BigEndian.PutUint64(blockNumBytes[common.HashLength:], blockNumber)
//hashed state storage
case common.HashLength*2 + common.IncarnationLength:
//remove incarnation and add block number
blockNumBytes = make([]byte, common.HashLength*2+8)
copy(blockNumBytes, key[:common.HashLength])
copy(blockNumBytes[common.HashLength:], key[common.HashLength+common.IncarnationLength:])
binary.BigEndian.PutUint64(blockNumBytes[common.HashLength*2:], blockNumber)
//plain state storage
case common.AddressLength + common.HashLength + common.IncarnationLength:
//remove incarnation and add block number
blockNumBytes = make([]byte, common.AddressLength+common.HashLength+8)
copy(blockNumBytes, key[:common.AddressLength])
copy(blockNumBytes[common.AddressLength:], key[common.AddressLength+common.IncarnationLength:])
binary.BigEndian.PutUint64(blockNumBytes[common.AddressLength+common.HashLength:], blockNumber)
default:
panic("unexpected length " + strconv.Itoa(len(key)))
}
return blockNumBytes
}
func CompositeKeyWithoutIncarnation(key []byte) []byte {
if len(key) == common.HashLength*2+common.IncarnationLength {
kk := make([]byte, common.HashLength*2)
copy(kk, key[:common.HashLength])
copy(kk[common.HashLength:], key[common.HashLength+common.IncarnationLength:])
return kk
}
if len(key) == common.AddressLength+common.HashLength+common.IncarnationLength {
kk := make([]byte, common.AddressLength+common.HashLength)
copy(kk, key[:common.AddressLength])
copy(kk[common.AddressLength:], key[common.AddressLength+common.IncarnationLength:])
return kk
}
return key
}
func IsIndexBucket(b []byte) bool {
return bytes.Equal(b, AccountsHistoryBucket) || bytes.Equal(b, StorageHistoryBucket)
}
func CheckNewIndexChunk(b []byte, v uint64) bool {
if len(b) < 8 {
panic(fmt.Errorf("minimal length of index chunk is %d, got %d", 8, len(b)))
}
if (len(b)-8)%ItemLen != 0 {
panic(fmt.Errorf("length of index chunk should be 8 (mod %d), got %d", ItemLen, len(b)))
}
numElements := (len(b) - 8) / 3
if numElements == 0 {
return false
}
minElement := binary.BigEndian.Uint64(b[:8])
// last value
lastIdx := len(b) - 3
lastElement := minElement + (uint64(b[lastIdx]&0x7f) << 16) + (uint64(b[lastIdx+1]) << 8) + uint64(b[lastIdx+2])
// Do not append if the value is less or equal to the last - to make operation idempotent
if v <= lastElement {
return false
}
return (numElements >= MaxChunkSize) || (v > minElement+0x7fffff)
}