2022-12-19 08:38:54 +00:00
|
|
|
package changeset
|
|
|
|
|
|
|
|
import (
|
|
|
|
common2 "github.com/ledgerwatch/erigon-lib/common"
|
2023-01-13 18:12:18 +00:00
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/common/hexutility"
|
2022-12-19 08:38:54 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/etl"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv/temporal/historyv2"
|
2023-01-13 18:12:18 +00:00
|
|
|
|
2022-12-19 08:38:54 +00:00
|
|
|
"github.com/ledgerwatch/erigon/ethdb"
|
|
|
|
)
|
|
|
|
|
|
|
|
// GetModifiedAccounts returns a list of addresses that were modified in the block range
|
|
|
|
// [startNum:endNum)
|
2023-01-13 18:12:18 +00:00
|
|
|
func GetModifiedAccounts(db kv.Tx, startNum, endNum uint64) ([]libcommon.Address, error) {
|
|
|
|
changedAddrs := make(map[libcommon.Address]struct{})
|
2022-12-19 08:38:54 +00:00
|
|
|
if err := ForRange(db, kv.AccountChangeSet, startNum, endNum, func(blockN uint64, k, v []byte) error {
|
2023-01-13 18:12:18 +00:00
|
|
|
changedAddrs[libcommon.BytesToAddress(k)] = struct{}{}
|
2022-12-19 08:38:54 +00:00
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(changedAddrs) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
idx := 0
|
2023-01-13 18:12:18 +00:00
|
|
|
result := make([]libcommon.Address, len(changedAddrs))
|
2022-12-19 08:38:54 +00:00
|
|
|
for addr := range changedAddrs {
|
|
|
|
copy(result[idx][:], addr[:])
|
|
|
|
idx++
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// [from:to)
|
|
|
|
func ForRange(db kv.Tx, bucket string, from, to uint64, walker func(blockN uint64, k, v []byte) error) error {
|
|
|
|
var blockN uint64
|
|
|
|
c, err := db.Cursor(bucket)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer c.Close()
|
2023-01-13 18:12:18 +00:00
|
|
|
return ethdb.Walk(c, hexutility.EncodeTs(from), 0, func(k, v []byte) (bool, error) {
|
2022-12-19 08:38:54 +00:00
|
|
|
var err error
|
|
|
|
blockN, k, v, err = historyv2.FromDBFormat(k, v)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if blockN >= to {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
if err = walker(blockN, k, v); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// RewindDataPlain generates rewind data for all plain buckets between the timestamp
|
|
|
|
// timestapSrc is the current timestamp, and timestamp Dst is where we rewind
|
|
|
|
func RewindData(db kv.Tx, timestampSrc, timestampDst uint64, changes *etl.Collector, quit <-chan struct{}) error {
|
|
|
|
if err := walkAndCollect(
|
|
|
|
changes.Collect,
|
|
|
|
db, kv.AccountChangeSet,
|
|
|
|
timestampDst+1, timestampSrc,
|
|
|
|
quit,
|
|
|
|
); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := walkAndCollect(
|
|
|
|
changes.Collect,
|
|
|
|
db, kv.StorageChangeSet,
|
|
|
|
timestampDst+1, timestampSrc,
|
|
|
|
quit,
|
|
|
|
); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func walkAndCollect(collectorFunc func([]byte, []byte) error, db kv.Tx, bucket string, timestampDst, timestampSrc uint64, quit <-chan struct{}) error {
|
|
|
|
return ForRange(db, bucket, timestampDst, timestampSrc+1, func(bl uint64, k, v []byte) error {
|
|
|
|
if err := common2.Stopped(quit); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if innerErr := collectorFunc(common2.Copy(k), common2.Copy(v)); innerErr != nil {
|
|
|
|
return innerErr
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|