2022-07-23 08:06:52 +00:00
|
|
|
/*
|
|
|
|
Copyright 2022 Erigon contributors
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package state
|
|
|
|
|
|
|
|
import (
|
2022-08-14 10:53:53 +00:00
|
|
|
"context"
|
2022-10-05 10:54:48 +00:00
|
|
|
"encoding/binary"
|
2022-07-23 08:06:52 +00:00
|
|
|
"fmt"
|
2022-08-13 11:51:23 +00:00
|
|
|
math2 "math"
|
2022-10-01 02:25:59 +00:00
|
|
|
"runtime"
|
|
|
|
"strings"
|
2022-10-13 02:46:24 +00:00
|
|
|
"sync"
|
2022-10-15 01:21:15 +00:00
|
|
|
"time"
|
2022-07-23 08:06:52 +00:00
|
|
|
|
|
|
|
"github.com/RoaringBitmap/roaring/roaring64"
|
2022-10-01 02:25:59 +00:00
|
|
|
common2 "github.com/ledgerwatch/erigon-lib/common"
|
2022-08-14 03:21:38 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/etl"
|
2022-07-23 08:06:52 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
2022-10-01 02:25:59 +00:00
|
|
|
"github.com/ledgerwatch/log/v3"
|
2022-10-02 03:03:49 +00:00
|
|
|
"go.uber.org/atomic"
|
2022-10-21 06:31:03 +00:00
|
|
|
"golang.org/x/sync/semaphore"
|
2022-07-23 08:06:52 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type Aggregator22 struct {
|
2022-10-21 08:31:23 +00:00
|
|
|
rwTx kv.RwTx
|
|
|
|
db kv.RoDB
|
|
|
|
storage *History
|
|
|
|
tracesTo *InvertedIndex
|
2022-10-02 03:03:49 +00:00
|
|
|
backgroundResult *BackgroundResult
|
2022-10-21 08:31:23 +00:00
|
|
|
code *History
|
|
|
|
logAddrs *InvertedIndex
|
|
|
|
logTopics *InvertedIndex
|
|
|
|
tracesFrom *InvertedIndex
|
|
|
|
accounts *History
|
|
|
|
logPrefix string
|
|
|
|
dir string
|
|
|
|
tmpdir string
|
|
|
|
txNum uint64
|
|
|
|
aggregationStep uint64
|
|
|
|
maxTxNum atomic.Uint64
|
2022-10-02 03:03:49 +00:00
|
|
|
working atomic.Bool
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
|
2022-10-15 01:21:15 +00:00
|
|
|
func NewAggregator22(dir, tmpdir string, aggregationStep uint64, db kv.RoDB) (*Aggregator22, error) {
|
|
|
|
a := &Aggregator22{dir: dir, tmpdir: tmpdir, aggregationStep: aggregationStep, backgroundResult: &BackgroundResult{}, db: db}
|
2022-10-11 04:25:08 +00:00
|
|
|
return a, nil
|
2022-09-18 10:38:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) ReopenFiles() error {
|
|
|
|
dir := a.dir
|
|
|
|
aggregationStep := a.aggregationStep
|
2022-07-23 08:06:52 +00:00
|
|
|
var err error
|
2022-10-21 06:31:03 +00:00
|
|
|
if a.accounts, err = NewHistory(dir, a.tmpdir, aggregationStep, "accounts", kv.AccountHistoryKeys, kv.AccountIdx, kv.AccountHistoryVals, kv.AccountSettings, false /* compressVals */); err != nil {
|
2022-09-18 10:38:43 +00:00
|
|
|
return fmt.Errorf("ReopenFiles: %w", err)
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-10-21 06:31:03 +00:00
|
|
|
if a.storage, err = NewHistory(dir, a.tmpdir, aggregationStep, "storage", kv.StorageHistoryKeys, kv.StorageIdx, kv.StorageHistoryVals, kv.StorageSettings, false /* compressVals */); err != nil {
|
2022-09-18 10:38:43 +00:00
|
|
|
return fmt.Errorf("ReopenFiles: %w", err)
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-10-21 06:31:03 +00:00
|
|
|
if a.code, err = NewHistory(dir, a.tmpdir, aggregationStep, "code", kv.CodeHistoryKeys, kv.CodeIdx, kv.CodeHistoryVals, kv.CodeSettings, true /* compressVals */); err != nil {
|
2022-09-18 10:38:43 +00:00
|
|
|
return fmt.Errorf("ReopenFiles: %w", err)
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-10-21 06:31:03 +00:00
|
|
|
if a.logAddrs, err = NewInvertedIndex(dir, a.tmpdir, aggregationStep, "logaddrs", kv.LogAddressKeys, kv.LogAddressIdx); err != nil {
|
2022-09-18 10:38:43 +00:00
|
|
|
return fmt.Errorf("ReopenFiles: %w", err)
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-10-21 06:31:03 +00:00
|
|
|
if a.logTopics, err = NewInvertedIndex(dir, a.tmpdir, aggregationStep, "logtopics", kv.LogTopicsKeys, kv.LogTopicsIdx); err != nil {
|
2022-09-18 10:38:43 +00:00
|
|
|
return fmt.Errorf("ReopenFiles: %w", err)
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-10-21 06:31:03 +00:00
|
|
|
if a.tracesFrom, err = NewInvertedIndex(dir, a.tmpdir, aggregationStep, "tracesfrom", kv.TracesFromKeys, kv.TracesFromIdx); err != nil {
|
2022-09-18 10:38:43 +00:00
|
|
|
return fmt.Errorf("ReopenFiles: %w", err)
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-10-21 06:31:03 +00:00
|
|
|
if a.tracesTo, err = NewInvertedIndex(dir, a.tmpdir, aggregationStep, "tracesto", kv.TracesToKeys, kv.TracesToIdx); err != nil {
|
2022-09-18 10:38:43 +00:00
|
|
|
return fmt.Errorf("ReopenFiles: %w", err)
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-10-02 03:03:49 +00:00
|
|
|
a.recalcMaxTxNum()
|
2022-09-18 10:38:43 +00:00
|
|
|
return nil
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) Close() {
|
2022-09-18 10:38:43 +00:00
|
|
|
a.closeFiles()
|
|
|
|
}
|
|
|
|
|
2022-09-26 08:26:58 +00:00
|
|
|
func (a *Aggregator22) SetWorkers(i int) {
|
|
|
|
a.accounts.workers = i
|
|
|
|
a.storage.workers = i
|
|
|
|
a.code.workers = i
|
|
|
|
a.logAddrs.workers = i
|
|
|
|
a.logTopics.workers = i
|
|
|
|
a.tracesFrom.workers = i
|
|
|
|
a.tracesTo.workers = i
|
|
|
|
}
|
|
|
|
|
2022-09-18 10:38:43 +00:00
|
|
|
func (a *Aggregator22) Files() (res []string) {
|
|
|
|
res = append(res, a.accounts.Files()...)
|
|
|
|
res = append(res, a.storage.Files()...)
|
|
|
|
res = append(res, a.code.Files()...)
|
|
|
|
res = append(res, a.logAddrs.Files()...)
|
|
|
|
res = append(res, a.logTopics.Files()...)
|
|
|
|
res = append(res, a.tracesFrom.Files()...)
|
|
|
|
res = append(res, a.tracesTo.Files()...)
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) closeFiles() {
|
2022-07-23 08:06:52 +00:00
|
|
|
if a.accounts != nil {
|
|
|
|
a.accounts.Close()
|
|
|
|
}
|
|
|
|
if a.storage != nil {
|
|
|
|
a.storage.Close()
|
|
|
|
}
|
|
|
|
if a.code != nil {
|
|
|
|
a.code.Close()
|
|
|
|
}
|
|
|
|
if a.logAddrs != nil {
|
|
|
|
a.logAddrs.Close()
|
|
|
|
}
|
|
|
|
if a.logTopics != nil {
|
|
|
|
a.logTopics.Close()
|
|
|
|
}
|
|
|
|
if a.tracesFrom != nil {
|
|
|
|
a.tracesFrom.Close()
|
|
|
|
}
|
|
|
|
if a.tracesTo != nil {
|
|
|
|
a.tracesTo.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-21 06:31:03 +00:00
|
|
|
func (a *Aggregator22) BuildMissedIndices(ctx context.Context, sem *semaphore.Weighted) error {
|
2022-10-13 02:46:24 +00:00
|
|
|
wg := sync.WaitGroup{}
|
|
|
|
errs := make(chan error, 7)
|
2022-09-26 08:26:58 +00:00
|
|
|
if a.accounts != nil {
|
2022-10-13 02:46:24 +00:00
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2022-10-21 06:31:03 +00:00
|
|
|
errs <- a.accounts.BuildMissedIndices(ctx, sem)
|
2022-10-13 02:46:24 +00:00
|
|
|
}()
|
2022-09-26 08:26:58 +00:00
|
|
|
}
|
|
|
|
if a.storage != nil {
|
2022-10-13 02:46:24 +00:00
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2022-10-21 06:31:03 +00:00
|
|
|
errs <- a.storage.BuildMissedIndices(ctx, sem)
|
2022-10-13 02:46:24 +00:00
|
|
|
}()
|
2022-09-26 08:26:58 +00:00
|
|
|
}
|
|
|
|
if a.code != nil {
|
2022-10-13 02:46:24 +00:00
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2022-10-21 06:31:03 +00:00
|
|
|
errs <- a.code.BuildMissedIndices(ctx, sem)
|
2022-10-13 02:46:24 +00:00
|
|
|
}()
|
2022-09-26 08:26:58 +00:00
|
|
|
}
|
|
|
|
if a.logAddrs != nil {
|
2022-10-13 02:46:24 +00:00
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2022-10-21 06:31:03 +00:00
|
|
|
errs <- a.logAddrs.BuildMissedIndices(ctx, sem)
|
2022-10-13 02:46:24 +00:00
|
|
|
}()
|
2022-09-26 08:26:58 +00:00
|
|
|
}
|
|
|
|
if a.logTopics != nil {
|
2022-10-13 02:46:24 +00:00
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2022-10-21 06:31:03 +00:00
|
|
|
errs <- a.logTopics.BuildMissedIndices(ctx, sem)
|
2022-10-13 02:46:24 +00:00
|
|
|
}()
|
2022-09-26 08:26:58 +00:00
|
|
|
}
|
|
|
|
if a.tracesFrom != nil {
|
2022-10-13 02:46:24 +00:00
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2022-10-21 06:31:03 +00:00
|
|
|
errs <- a.tracesFrom.BuildMissedIndices(ctx, sem)
|
2022-10-13 02:46:24 +00:00
|
|
|
}()
|
2022-09-26 08:26:58 +00:00
|
|
|
}
|
|
|
|
if a.tracesTo != nil {
|
2022-10-13 02:46:24 +00:00
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2022-10-21 06:31:03 +00:00
|
|
|
errs <- a.tracesTo.BuildMissedIndices(ctx, sem)
|
2022-10-13 02:46:24 +00:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
wg.Wait()
|
|
|
|
close(errs)
|
|
|
|
}()
|
|
|
|
var lastError error
|
|
|
|
for err := range errs {
|
|
|
|
if err != nil {
|
|
|
|
lastError = err
|
2022-09-26 08:26:58 +00:00
|
|
|
}
|
|
|
|
}
|
2022-10-13 02:46:24 +00:00
|
|
|
return lastError
|
2022-09-26 08:26:58 +00:00
|
|
|
}
|
|
|
|
|
2022-08-15 03:27:08 +00:00
|
|
|
func (a *Aggregator22) SetLogPrefix(v string) { a.logPrefix = v }
|
|
|
|
|
2022-07-23 08:06:52 +00:00
|
|
|
func (a *Aggregator22) SetTx(tx kv.RwTx) {
|
|
|
|
a.rwTx = tx
|
|
|
|
a.accounts.SetTx(tx)
|
|
|
|
a.storage.SetTx(tx)
|
|
|
|
a.code.SetTx(tx)
|
|
|
|
a.logAddrs.SetTx(tx)
|
|
|
|
a.logTopics.SetTx(tx)
|
|
|
|
a.tracesFrom.SetTx(tx)
|
|
|
|
a.tracesTo.SetTx(tx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) SetTxNum(txNum uint64) {
|
|
|
|
a.txNum = txNum
|
|
|
|
a.accounts.SetTxNum(txNum)
|
|
|
|
a.storage.SetTxNum(txNum)
|
|
|
|
a.code.SetTxNum(txNum)
|
|
|
|
a.logAddrs.SetTxNum(txNum)
|
|
|
|
a.logTopics.SetTxNum(txNum)
|
|
|
|
a.tracesFrom.SetTxNum(txNum)
|
|
|
|
a.tracesTo.SetTxNum(txNum)
|
|
|
|
}
|
|
|
|
|
|
|
|
type Agg22Collation struct {
|
|
|
|
logAddrs map[string]*roaring64.Bitmap
|
|
|
|
logTopics map[string]*roaring64.Bitmap
|
|
|
|
tracesFrom map[string]*roaring64.Bitmap
|
|
|
|
tracesTo map[string]*roaring64.Bitmap
|
2022-10-21 08:31:23 +00:00
|
|
|
accounts HistoryCollation
|
|
|
|
storage HistoryCollation
|
|
|
|
code HistoryCollation
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c Agg22Collation) Close() {
|
|
|
|
c.accounts.Close()
|
|
|
|
c.storage.Close()
|
|
|
|
c.code.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) collate(step uint64, txFrom, txTo uint64, roTx kv.Tx) (Agg22Collation, error) {
|
|
|
|
var ac Agg22Collation
|
|
|
|
var err error
|
|
|
|
closeColl := true
|
|
|
|
defer func() {
|
|
|
|
if closeColl {
|
|
|
|
ac.Close()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
if ac.accounts, err = a.accounts.collate(step, txFrom, txTo, roTx); err != nil {
|
|
|
|
return Agg22Collation{}, err
|
|
|
|
}
|
|
|
|
if ac.storage, err = a.storage.collate(step, txFrom, txTo, roTx); err != nil {
|
|
|
|
return Agg22Collation{}, err
|
|
|
|
}
|
|
|
|
if ac.code, err = a.code.collate(step, txFrom, txTo, roTx); err != nil {
|
|
|
|
return Agg22Collation{}, err
|
|
|
|
}
|
|
|
|
if ac.logAddrs, err = a.logAddrs.collate(txFrom, txTo, roTx); err != nil {
|
|
|
|
return Agg22Collation{}, err
|
|
|
|
}
|
|
|
|
if ac.logTopics, err = a.logTopics.collate(txFrom, txTo, roTx); err != nil {
|
|
|
|
return Agg22Collation{}, err
|
|
|
|
}
|
|
|
|
if ac.tracesFrom, err = a.tracesFrom.collate(txFrom, txTo, roTx); err != nil {
|
|
|
|
return Agg22Collation{}, err
|
|
|
|
}
|
|
|
|
if ac.tracesTo, err = a.tracesTo.collate(txFrom, txTo, roTx); err != nil {
|
|
|
|
return Agg22Collation{}, err
|
|
|
|
}
|
|
|
|
closeColl = false
|
|
|
|
return ac, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type Agg22StaticFiles struct {
|
|
|
|
accounts HistoryFiles
|
|
|
|
storage HistoryFiles
|
|
|
|
code HistoryFiles
|
|
|
|
logAddrs InvertedFiles
|
|
|
|
logTopics InvertedFiles
|
|
|
|
tracesFrom InvertedFiles
|
|
|
|
tracesTo InvertedFiles
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sf Agg22StaticFiles) Close() {
|
|
|
|
sf.accounts.Close()
|
|
|
|
sf.storage.Close()
|
|
|
|
sf.code.Close()
|
|
|
|
sf.logAddrs.Close()
|
|
|
|
sf.logTopics.Close()
|
|
|
|
sf.tracesFrom.Close()
|
|
|
|
sf.tracesTo.Close()
|
|
|
|
}
|
|
|
|
|
2022-10-21 06:31:03 +00:00
|
|
|
func (a *Aggregator22) buildFilesInBackground(ctx context.Context, step uint64, collation Agg22Collation) error {
|
2022-10-05 10:54:48 +00:00
|
|
|
log.Info("[snapshots] history build", "step", fmt.Sprintf("%d-%d", step, step+1))
|
|
|
|
closeAll := true
|
2022-10-21 06:31:03 +00:00
|
|
|
sf, err := a.buildFiles(ctx, step, collation)
|
2022-10-05 10:54:48 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
if closeAll {
|
|
|
|
sf.Close()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
a.integrateFiles(sf, step*a.aggregationStep, (step+1)*a.aggregationStep)
|
|
|
|
log.Info("[snapshots] history build done", "step", fmt.Sprintf("%d-%d", step, step+1))
|
2022-10-21 06:31:03 +00:00
|
|
|
if err := a.Merge(ctx); err != nil {
|
2022-10-10 02:47:05 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
closeAll = false
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-10-21 06:31:03 +00:00
|
|
|
func (a *Aggregator22) Merge(ctx context.Context) error {
|
2022-10-10 02:47:05 +00:00
|
|
|
closeAll := true
|
2022-10-05 10:54:48 +00:00
|
|
|
maxSpan := uint64(32) * a.aggregationStep
|
2022-10-09 13:16:26 +00:00
|
|
|
for r := a.findMergeRange(a.maxTxNum.Load(), maxSpan); r.any(); r = a.findMergeRange(a.maxTxNum.Load(), maxSpan) {
|
2022-10-05 10:54:48 +00:00
|
|
|
outs := a.staticFilesInRange(r)
|
|
|
|
defer func() {
|
|
|
|
if closeAll {
|
|
|
|
outs.Close()
|
|
|
|
}
|
|
|
|
}()
|
2022-10-21 06:31:03 +00:00
|
|
|
in, err := a.mergeFiles(ctx, outs, r, maxSpan)
|
2022-10-05 10:54:48 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
if closeAll {
|
|
|
|
in.Close()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
a.integrateMergedFiles(outs, in)
|
|
|
|
if err = a.deleteFiles(outs); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
closeAll = false
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-10-21 06:31:03 +00:00
|
|
|
func (a *Aggregator22) buildFiles(ctx context.Context, step uint64, collation Agg22Collation) (Agg22StaticFiles, error) {
|
2022-07-23 08:06:52 +00:00
|
|
|
var sf Agg22StaticFiles
|
|
|
|
closeFiles := true
|
|
|
|
defer func() {
|
|
|
|
if closeFiles {
|
|
|
|
sf.Close()
|
|
|
|
}
|
|
|
|
}()
|
2022-10-15 01:21:15 +00:00
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(7)
|
2022-07-23 08:06:52 +00:00
|
|
|
errCh := make(chan error, 7)
|
2022-10-15 01:21:15 +00:00
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
var err error
|
2022-10-21 06:31:03 +00:00
|
|
|
if sf.accounts, err = a.accounts.buildFiles(ctx, step, collation.accounts); err != nil {
|
2022-10-15 01:21:15 +00:00
|
|
|
errCh <- err
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
var err error
|
2022-10-21 06:31:03 +00:00
|
|
|
if sf.storage, err = a.storage.buildFiles(ctx, step, collation.storage); err != nil {
|
2022-10-15 01:21:15 +00:00
|
|
|
errCh <- err
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
var err error
|
2022-10-21 06:31:03 +00:00
|
|
|
if sf.code, err = a.code.buildFiles(ctx, step, collation.code); err != nil {
|
2022-10-15 01:21:15 +00:00
|
|
|
errCh <- err
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
var err error
|
2022-10-21 06:31:03 +00:00
|
|
|
if sf.logAddrs, err = a.logAddrs.buildFiles(ctx, step, collation.logAddrs); err != nil {
|
2022-10-15 01:21:15 +00:00
|
|
|
errCh <- err
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
var err error
|
2022-10-21 06:31:03 +00:00
|
|
|
if sf.logTopics, err = a.logTopics.buildFiles(ctx, step, collation.logTopics); err != nil {
|
2022-10-15 01:21:15 +00:00
|
|
|
errCh <- err
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
var err error
|
2022-10-21 06:31:03 +00:00
|
|
|
if sf.tracesFrom, err = a.tracesFrom.buildFiles(ctx, step, collation.tracesFrom); err != nil {
|
2022-10-15 01:21:15 +00:00
|
|
|
errCh <- err
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
var err error
|
2022-10-21 06:31:03 +00:00
|
|
|
if sf.tracesTo, err = a.tracesTo.buildFiles(ctx, step, collation.tracesTo); err != nil {
|
2022-10-15 01:21:15 +00:00
|
|
|
errCh <- err
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
wg.Wait()
|
|
|
|
close(errCh)
|
|
|
|
}()
|
2022-07-23 08:06:52 +00:00
|
|
|
var lastError error
|
|
|
|
for err := range errCh {
|
2022-10-13 02:46:24 +00:00
|
|
|
if err != nil {
|
|
|
|
lastError = err
|
|
|
|
}
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
if lastError == nil {
|
|
|
|
closeFiles = false
|
|
|
|
}
|
|
|
|
return sf, lastError
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) integrateFiles(sf Agg22StaticFiles, txNumFrom, txNumTo uint64) {
|
|
|
|
a.accounts.integrateFiles(sf.accounts, txNumFrom, txNumTo)
|
|
|
|
a.storage.integrateFiles(sf.storage, txNumFrom, txNumTo)
|
|
|
|
a.code.integrateFiles(sf.code, txNumFrom, txNumTo)
|
|
|
|
a.logAddrs.integrateFiles(sf.logAddrs, txNumFrom, txNumTo)
|
|
|
|
a.logTopics.integrateFiles(sf.logTopics, txNumFrom, txNumTo)
|
|
|
|
a.tracesFrom.integrateFiles(sf.tracesFrom, txNumFrom, txNumTo)
|
|
|
|
a.tracesTo.integrateFiles(sf.tracesTo, txNumFrom, txNumTo)
|
2022-10-02 03:03:49 +00:00
|
|
|
a.recalcMaxTxNum()
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
|
2022-08-14 10:53:53 +00:00
|
|
|
func (a *Aggregator22) Unwind(ctx context.Context, txUnwindTo uint64, stateLoad etl.LoadFunc) error {
|
2022-08-15 03:27:08 +00:00
|
|
|
stateChanges := etl.NewCollector(a.logPrefix, "", etl.NewOldestEntryBuffer(etl.BufferOptimalSize))
|
|
|
|
defer stateChanges.Close()
|
2022-08-14 03:21:38 +00:00
|
|
|
|
2022-10-05 06:17:23 +00:00
|
|
|
if err := a.accounts.pruneF(txUnwindTo, math2.MaxUint64, func(_ uint64, k, v []byte) error {
|
|
|
|
return stateChanges.Collect(k, v)
|
2022-08-14 03:21:38 +00:00
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-10-05 06:17:23 +00:00
|
|
|
if err := a.storage.pruneF(txUnwindTo, math2.MaxUint64, func(_ uint64, k, v []byte) error {
|
|
|
|
return stateChanges.Collect(k, v)
|
2022-08-14 03:21:38 +00:00
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-08-14 10:53:53 +00:00
|
|
|
|
2022-08-15 03:27:08 +00:00
|
|
|
if err := stateChanges.Load(a.rwTx, kv.PlainState, stateLoad, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
2022-08-14 10:53:53 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.logAddrs.prune(txUnwindTo, math2.MaxUint64, math2.MaxUint64); err != nil {
|
2022-08-14 03:21:38 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.logTopics.prune(txUnwindTo, math2.MaxUint64, math2.MaxUint64); err != nil {
|
2022-08-14 03:21:38 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.tracesFrom.prune(txUnwindTo, math2.MaxUint64, math2.MaxUint64); err != nil {
|
2022-08-14 03:21:38 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.tracesTo.prune(txUnwindTo, math2.MaxUint64, math2.MaxUint64); err != nil {
|
2022-08-14 03:21:38 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
2022-08-13 11:51:23 +00:00
|
|
|
}
|
|
|
|
|
2022-10-11 04:25:08 +00:00
|
|
|
func (a *Aggregator22) warmup(txFrom, limit uint64) {
|
|
|
|
if a.db == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
if err := a.db.View(context.Background(), func(tx kv.Tx) error {
|
|
|
|
if err := a.accounts.warmup(txFrom, limit, tx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.storage.warmup(txFrom, limit, tx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.code.warmup(txFrom, limit, tx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.logAddrs.warmup(txFrom, limit, tx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.logTopics.warmup(txFrom, limit, tx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.tracesFrom.warmup(txFrom, limit, tx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.tracesTo.warmup(txFrom, limit, tx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
log.Warn("[snapshots] prune warmup", "err", err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2022-10-15 01:21:15 +00:00
|
|
|
// StartWrites - pattern: `defer agg.StartWrites().FinishWrites()`
|
|
|
|
func (a *Aggregator22) StartWrites() *Aggregator22 {
|
|
|
|
a.accounts.StartWrites(a.tmpdir)
|
|
|
|
a.storage.StartWrites(a.tmpdir)
|
|
|
|
a.code.StartWrites(a.tmpdir)
|
|
|
|
a.logAddrs.StartWrites(a.tmpdir)
|
|
|
|
a.logTopics.StartWrites(a.tmpdir)
|
|
|
|
a.tracesFrom.StartWrites(a.tmpdir)
|
|
|
|
a.tracesTo.StartWrites(a.tmpdir)
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
func (a *Aggregator22) FinishWrites() {
|
|
|
|
a.accounts.FinishWrites()
|
|
|
|
a.storage.FinishWrites()
|
|
|
|
a.code.FinishWrites()
|
|
|
|
a.logAddrs.FinishWrites()
|
|
|
|
a.logTopics.FinishWrites()
|
|
|
|
a.tracesFrom.FinishWrites()
|
|
|
|
a.tracesTo.FinishWrites()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) Flush() error {
|
|
|
|
defer func(t time.Time) { log.Info("[snapshots] hitory flush", "took", time.Since(t)) }(time.Now())
|
|
|
|
if err := a.accounts.Flush(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.storage.Flush(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.code.Flush(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.logAddrs.Flush(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.logTopics.Flush(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.tracesFrom.Flush(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.tracesTo.Flush(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-10-11 04:25:08 +00:00
|
|
|
func (a *Aggregator22) Prune(limit uint64) error {
|
|
|
|
return a.prune(0, a.maxTxNum.Load(), limit)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) prune(txFrom, txTo, limit uint64) error {
|
|
|
|
a.warmup(txFrom, limit) // warmup is asyn and moving faster than data deletion
|
|
|
|
if err := a.accounts.prune(txFrom, txTo, limit); err != nil {
|
2022-07-23 08:06:52 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.storage.prune(txFrom, txTo, limit); err != nil {
|
2022-07-23 08:06:52 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.code.prune(txFrom, txTo, limit); err != nil {
|
2022-07-23 08:06:52 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.logAddrs.prune(txFrom, txTo, limit); err != nil {
|
2022-07-23 08:06:52 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.logTopics.prune(txFrom, txTo, limit); err != nil {
|
2022-07-23 08:06:52 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.tracesFrom.prune(txFrom, txTo, limit); err != nil {
|
2022-07-23 08:06:52 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-10-11 04:25:08 +00:00
|
|
|
if err := a.tracesTo.prune(txFrom, txTo, limit); err != nil {
|
2022-07-23 08:06:52 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-10-05 10:54:48 +00:00
|
|
|
func (a *Aggregator22) LogStats(tx kv.Tx, tx2block func(endTxNumMinimax uint64) uint64) {
|
2022-10-09 13:16:26 +00:00
|
|
|
if a.maxTxNum.Load() == 0 {
|
2022-10-01 02:25:59 +00:00
|
|
|
return
|
|
|
|
}
|
2022-10-09 13:16:26 +00:00
|
|
|
histBlockNumProgress := tx2block(a.maxTxNum.Load())
|
2022-10-01 02:25:59 +00:00
|
|
|
str := make([]string, 0, a.accounts.InvertedIndex.files.Len())
|
|
|
|
a.accounts.InvertedIndex.files.Ascend(func(it *filesItem) bool {
|
|
|
|
bn := tx2block(it.endTxNum)
|
2022-10-12 03:18:51 +00:00
|
|
|
str = append(str, fmt.Sprintf("%d=%dK", it.endTxNum/a.aggregationStep, bn/1_000))
|
2022-10-01 02:25:59 +00:00
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
2022-10-05 10:54:48 +00:00
|
|
|
c, err := tx.CursorDupSort(a.accounts.InvertedIndex.indexTable)
|
|
|
|
if err != nil {
|
|
|
|
// TODO pass error properly around
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
_, v, err := c.First()
|
|
|
|
if err != nil {
|
|
|
|
// TODO pass error properly around
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
var firstHistoryIndexBlockInDB uint64
|
|
|
|
if len(v) != 0 {
|
|
|
|
firstHistoryIndexBlockInDB = tx2block(binary.BigEndian.Uint64(v))
|
|
|
|
}
|
|
|
|
|
2022-10-01 02:25:59 +00:00
|
|
|
var m runtime.MemStats
|
|
|
|
common2.ReadMemStats(&m)
|
|
|
|
log.Info("[Snapshots] History Stat",
|
|
|
|
"blocks", fmt.Sprintf("%dk", (histBlockNumProgress+1)/1000),
|
2022-10-13 02:46:24 +00:00
|
|
|
"txs", fmt.Sprintf("%dm", a.maxTxNum.Load()/1_000_000),
|
2022-10-01 02:25:59 +00:00
|
|
|
"txNum2blockNum", strings.Join(str, ","),
|
2022-10-05 10:54:48 +00:00
|
|
|
"first_history_idx_in_db", firstHistoryIndexBlockInDB,
|
2022-10-01 02:25:59 +00:00
|
|
|
"alloc", common2.ByteCount(m.Alloc), "sys", common2.ByteCount(m.Sys))
|
|
|
|
}
|
|
|
|
|
2022-10-09 13:16:26 +00:00
|
|
|
func (a *Aggregator22) EndTxNumMinimax() uint64 { return a.maxTxNum.Load() }
|
2022-10-02 03:03:49 +00:00
|
|
|
func (a *Aggregator22) recalcMaxTxNum() {
|
2022-07-23 08:06:52 +00:00
|
|
|
min := a.accounts.endTxNumMinimax()
|
|
|
|
if txNum := a.storage.endTxNumMinimax(); txNum < min {
|
|
|
|
min = txNum
|
|
|
|
}
|
|
|
|
if txNum := a.code.endTxNumMinimax(); txNum < min {
|
|
|
|
min = txNum
|
|
|
|
}
|
|
|
|
if txNum := a.logAddrs.endTxNumMinimax(); txNum < min {
|
|
|
|
min = txNum
|
|
|
|
}
|
|
|
|
if txNum := a.logTopics.endTxNumMinimax(); txNum < min {
|
|
|
|
min = txNum
|
|
|
|
}
|
|
|
|
if txNum := a.tracesFrom.endTxNumMinimax(); txNum < min {
|
|
|
|
min = txNum
|
|
|
|
}
|
|
|
|
if txNum := a.tracesTo.endTxNumMinimax(); txNum < min {
|
|
|
|
min = txNum
|
|
|
|
}
|
2022-10-09 13:16:26 +00:00
|
|
|
a.maxTxNum.Store(min)
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type Ranges22 struct {
|
2022-10-21 08:31:23 +00:00
|
|
|
accounts HistoryRanges
|
|
|
|
storage HistoryRanges
|
|
|
|
code HistoryRanges
|
|
|
|
logTopicsStartTxNum uint64
|
|
|
|
logAddrsEndTxNum uint64
|
|
|
|
logAddrsStartTxNum uint64
|
|
|
|
logTopicsEndTxNum uint64
|
|
|
|
tracesFromStartTxNum uint64
|
|
|
|
tracesFromEndTxNum uint64
|
|
|
|
tracesToStartTxNum uint64
|
|
|
|
tracesToEndTxNum uint64
|
|
|
|
logAddrs bool
|
|
|
|
logTopics bool
|
|
|
|
tracesFrom bool
|
|
|
|
tracesTo bool
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r Ranges22) any() bool {
|
|
|
|
return r.accounts.any() || r.storage.any() || r.code.any() || r.logAddrs || r.logTopics || r.tracesFrom || r.tracesTo
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) findMergeRange(maxEndTxNum, maxSpan uint64) Ranges22 {
|
|
|
|
var r Ranges22
|
|
|
|
r.accounts = a.accounts.findMergeRange(maxEndTxNum, maxSpan)
|
|
|
|
r.storage = a.storage.findMergeRange(maxEndTxNum, maxSpan)
|
|
|
|
r.code = a.code.findMergeRange(maxEndTxNum, maxSpan)
|
|
|
|
r.logAddrs, r.logAddrsStartTxNum, r.logAddrsEndTxNum = a.logAddrs.findMergeRange(maxEndTxNum, maxSpan)
|
|
|
|
r.logTopics, r.logTopicsStartTxNum, r.logTopicsEndTxNum = a.logTopics.findMergeRange(maxEndTxNum, maxSpan)
|
|
|
|
r.tracesFrom, r.tracesFromStartTxNum, r.tracesFromEndTxNum = a.tracesFrom.findMergeRange(maxEndTxNum, maxSpan)
|
|
|
|
r.tracesTo, r.tracesToStartTxNum, r.tracesToEndTxNum = a.tracesTo.findMergeRange(maxEndTxNum, maxSpan)
|
2022-10-11 04:25:08 +00:00
|
|
|
log.Info(fmt.Sprintf("findMergeRange(%d, %d)=%+v\n", maxEndTxNum, maxSpan, r))
|
2022-07-23 08:06:52 +00:00
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
|
|
|
type SelectedStaticFiles22 struct {
|
2022-10-21 08:31:23 +00:00
|
|
|
logTopics []*filesItem
|
|
|
|
accountsHist []*filesItem
|
|
|
|
tracesTo []*filesItem
|
|
|
|
storageIdx []*filesItem
|
|
|
|
storageHist []*filesItem
|
|
|
|
tracesFrom []*filesItem
|
|
|
|
codeIdx []*filesItem
|
|
|
|
codeHist []*filesItem
|
|
|
|
accountsIdx []*filesItem
|
|
|
|
logAddrs []*filesItem
|
|
|
|
codeI int
|
|
|
|
logAddrsI int
|
|
|
|
logTopicsI int
|
|
|
|
storageI int
|
|
|
|
tracesFromI int
|
|
|
|
accountsI int
|
|
|
|
tracesToI int
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (sf SelectedStaticFiles22) Close() {
|
|
|
|
for _, group := range [][]*filesItem{sf.accountsIdx, sf.accountsHist, sf.storageIdx, sf.accountsHist, sf.codeIdx, sf.codeHist,
|
|
|
|
sf.logAddrs, sf.logTopics, sf.tracesFrom, sf.tracesTo} {
|
|
|
|
for _, item := range group {
|
|
|
|
if item != nil {
|
|
|
|
if item.decompressor != nil {
|
|
|
|
item.decompressor.Close()
|
|
|
|
}
|
2022-10-06 05:20:28 +00:00
|
|
|
if item.index != nil {
|
2022-07-23 08:06:52 +00:00
|
|
|
item.index.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) staticFilesInRange(r Ranges22) SelectedStaticFiles22 {
|
|
|
|
var sf SelectedStaticFiles22
|
|
|
|
if r.accounts.any() {
|
|
|
|
sf.accountsIdx, sf.accountsHist, sf.accountsI = a.accounts.staticFilesInRange(r.accounts)
|
|
|
|
}
|
|
|
|
if r.storage.any() {
|
|
|
|
sf.storageIdx, sf.storageHist, sf.storageI = a.storage.staticFilesInRange(r.storage)
|
|
|
|
}
|
|
|
|
if r.code.any() {
|
|
|
|
sf.codeIdx, sf.codeHist, sf.codeI = a.code.staticFilesInRange(r.code)
|
|
|
|
}
|
|
|
|
if r.logAddrs {
|
|
|
|
sf.logAddrs, sf.logAddrsI = a.logAddrs.staticFilesInRange(r.logAddrsStartTxNum, r.logAddrsEndTxNum)
|
|
|
|
}
|
|
|
|
if r.logTopics {
|
|
|
|
sf.logTopics, sf.logTopicsI = a.logTopics.staticFilesInRange(r.logTopicsStartTxNum, r.logTopicsEndTxNum)
|
|
|
|
}
|
|
|
|
if r.tracesFrom {
|
|
|
|
sf.tracesFrom, sf.tracesFromI = a.tracesFrom.staticFilesInRange(r.tracesFromStartTxNum, r.tracesFromEndTxNum)
|
|
|
|
}
|
|
|
|
if r.tracesTo {
|
|
|
|
sf.tracesTo, sf.tracesToI = a.tracesTo.staticFilesInRange(r.tracesToStartTxNum, r.tracesToEndTxNum)
|
|
|
|
}
|
|
|
|
return sf
|
|
|
|
}
|
|
|
|
|
|
|
|
type MergedFiles22 struct {
|
|
|
|
accountsIdx, accountsHist *filesItem
|
|
|
|
storageIdx, storageHist *filesItem
|
|
|
|
codeIdx, codeHist *filesItem
|
|
|
|
logAddrs *filesItem
|
|
|
|
logTopics *filesItem
|
|
|
|
tracesFrom *filesItem
|
|
|
|
tracesTo *filesItem
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mf MergedFiles22) Close() {
|
|
|
|
for _, item := range []*filesItem{mf.accountsIdx, mf.accountsHist, mf.storageIdx, mf.storageHist, mf.codeIdx, mf.codeHist,
|
|
|
|
mf.logAddrs, mf.logTopics, mf.tracesFrom, mf.tracesTo} {
|
|
|
|
if item != nil {
|
|
|
|
if item.decompressor != nil {
|
|
|
|
item.decompressor.Close()
|
|
|
|
}
|
2022-10-06 05:20:28 +00:00
|
|
|
if item.index != nil {
|
2022-07-23 08:06:52 +00:00
|
|
|
item.index.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-21 06:31:03 +00:00
|
|
|
func (a *Aggregator22) mergeFiles(ctx context.Context, files SelectedStaticFiles22, r Ranges22, maxSpan uint64) (MergedFiles22, error) {
|
2022-07-23 08:06:52 +00:00
|
|
|
var mf MergedFiles22
|
|
|
|
closeFiles := true
|
|
|
|
defer func() {
|
|
|
|
if closeFiles {
|
|
|
|
mf.Close()
|
|
|
|
}
|
|
|
|
}()
|
2022-08-14 13:56:47 +00:00
|
|
|
//var wg sync.WaitGroup
|
|
|
|
//wg.Add(7)
|
2022-07-23 08:06:52 +00:00
|
|
|
errCh := make(chan error, 7)
|
2022-08-14 13:56:47 +00:00
|
|
|
//go func() {
|
|
|
|
// defer wg.Done()
|
|
|
|
var err error
|
|
|
|
if r.accounts.any() {
|
2022-10-21 06:31:03 +00:00
|
|
|
if mf.accountsIdx, mf.accountsHist, err = a.accounts.mergeFiles(ctx, files.accountsIdx, files.accountsHist, r.accounts, maxSpan); err != nil {
|
2022-08-14 13:56:47 +00:00
|
|
|
errCh <- err
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
}
|
|
|
|
//}()
|
|
|
|
//go func() {
|
|
|
|
// defer wg.Done()
|
|
|
|
// var err error
|
|
|
|
if r.storage.any() {
|
2022-10-21 06:31:03 +00:00
|
|
|
if mf.storageIdx, mf.storageHist, err = a.storage.mergeFiles(ctx, files.storageIdx, files.storageHist, r.storage, maxSpan); err != nil {
|
2022-08-14 13:56:47 +00:00
|
|
|
errCh <- err
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
}
|
|
|
|
//}()
|
|
|
|
//go func() {
|
|
|
|
// defer wg.Done()
|
|
|
|
// var err error
|
|
|
|
if r.code.any() {
|
2022-10-21 06:31:03 +00:00
|
|
|
if mf.codeIdx, mf.codeHist, err = a.code.mergeFiles(ctx, files.codeIdx, files.codeHist, r.code, maxSpan); err != nil {
|
2022-08-14 13:56:47 +00:00
|
|
|
errCh <- err
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
}
|
|
|
|
//}()
|
|
|
|
//go func() {
|
|
|
|
// defer wg.Done()
|
|
|
|
// var err error
|
|
|
|
if r.logAddrs {
|
2022-10-21 06:31:03 +00:00
|
|
|
if mf.logAddrs, err = a.logAddrs.mergeFiles(ctx, files.logAddrs, r.logAddrsStartTxNum, r.logAddrsEndTxNum, maxSpan); err != nil {
|
2022-08-14 13:56:47 +00:00
|
|
|
errCh <- err
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
}
|
|
|
|
//}()
|
|
|
|
//go func() {
|
|
|
|
// defer wg.Done()
|
|
|
|
// var err error
|
|
|
|
if r.logTopics {
|
2022-10-21 06:31:03 +00:00
|
|
|
if mf.logTopics, err = a.logTopics.mergeFiles(ctx, files.logTopics, r.logTopicsStartTxNum, r.logTopicsEndTxNum, maxSpan); err != nil {
|
2022-08-14 13:56:47 +00:00
|
|
|
errCh <- err
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
}
|
|
|
|
//}()
|
|
|
|
//go func() {
|
|
|
|
// defer wg.Done()
|
|
|
|
// var err error
|
|
|
|
if r.tracesFrom {
|
2022-10-21 06:31:03 +00:00
|
|
|
if mf.tracesFrom, err = a.tracesFrom.mergeFiles(ctx, files.tracesFrom, r.tracesFromStartTxNum, r.tracesFromEndTxNum, maxSpan); err != nil {
|
2022-08-14 13:56:47 +00:00
|
|
|
errCh <- err
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
}
|
|
|
|
//}()
|
|
|
|
//go func() {
|
|
|
|
// defer wg.Done()
|
|
|
|
// var err error
|
|
|
|
if r.tracesTo {
|
2022-10-21 06:31:03 +00:00
|
|
|
if mf.tracesTo, err = a.tracesTo.mergeFiles(ctx, files.tracesTo, r.tracesToStartTxNum, r.tracesToEndTxNum, maxSpan); err != nil {
|
2022-08-14 13:56:47 +00:00
|
|
|
errCh <- err
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
}
|
|
|
|
//}()
|
|
|
|
//go func() {
|
|
|
|
// wg.Wait()
|
|
|
|
close(errCh)
|
|
|
|
//}()
|
2022-07-23 08:06:52 +00:00
|
|
|
var lastError error
|
|
|
|
for err := range errCh {
|
|
|
|
lastError = err
|
|
|
|
}
|
|
|
|
if lastError == nil {
|
|
|
|
closeFiles = false
|
|
|
|
}
|
|
|
|
return mf, lastError
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) integrateMergedFiles(outs SelectedStaticFiles22, in MergedFiles22) {
|
|
|
|
a.accounts.integrateMergedFiles(outs.accountsIdx, outs.accountsHist, in.accountsIdx, in.accountsHist)
|
|
|
|
a.storage.integrateMergedFiles(outs.storageIdx, outs.storageHist, in.storageIdx, in.storageHist)
|
|
|
|
a.code.integrateMergedFiles(outs.codeIdx, outs.codeHist, in.codeIdx, in.codeHist)
|
|
|
|
a.logAddrs.integrateMergedFiles(outs.logAddrs, in.logAddrs)
|
|
|
|
a.logTopics.integrateMergedFiles(outs.logTopics, in.logTopics)
|
|
|
|
a.tracesFrom.integrateMergedFiles(outs.tracesFrom, in.tracesFrom)
|
|
|
|
a.tracesTo.integrateMergedFiles(outs.tracesTo, in.tracesTo)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) deleteFiles(outs SelectedStaticFiles22) error {
|
|
|
|
if err := a.accounts.deleteFiles(outs.accountsIdx, outs.accountsHist); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.storage.deleteFiles(outs.storageIdx, outs.storageHist); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.code.deleteFiles(outs.codeIdx, outs.codeHist); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.logAddrs.deleteFiles(outs.logAddrs); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.logTopics.deleteFiles(outs.logTopics); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.tracesFrom.deleteFiles(outs.tracesFrom); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := a.tracesTo.deleteFiles(outs.tracesTo); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) ReadyToFinishTx() bool {
|
|
|
|
return (a.txNum+1)%a.aggregationStep == 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) FinishTx() error {
|
2022-10-09 13:16:26 +00:00
|
|
|
if (a.txNum + 1) <= a.maxTxNum.Load()+2*a.aggregationStep { // Leave one step worth in the DB
|
2022-09-16 07:34:11 +00:00
|
|
|
return nil
|
|
|
|
}
|
2022-10-09 13:16:26 +00:00
|
|
|
step := a.maxTxNum.Load() / a.aggregationStep
|
2022-10-02 03:03:49 +00:00
|
|
|
if a.working.Load() {
|
2022-07-23 08:06:52 +00:00
|
|
|
return nil
|
|
|
|
}
|
2022-10-05 10:54:48 +00:00
|
|
|
|
2022-10-15 01:21:15 +00:00
|
|
|
closeAll := true
|
|
|
|
|
|
|
|
if err := a.Flush(); err != nil {
|
2022-10-05 10:54:48 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-07-23 08:06:52 +00:00
|
|
|
collation, err := a.collate(step, step*a.aggregationStep, (step+1)*a.aggregationStep, a.rwTx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
if closeAll {
|
|
|
|
collation.Close()
|
|
|
|
}
|
|
|
|
}()
|
2022-10-02 03:03:49 +00:00
|
|
|
|
|
|
|
a.working.Store(true)
|
2022-10-05 10:54:48 +00:00
|
|
|
go func() {
|
|
|
|
defer a.working.Store(false)
|
2022-10-21 06:31:03 +00:00
|
|
|
if err := a.buildFilesInBackground(context.Background(), step, collation); err != nil {
|
2022-10-05 10:54:48 +00:00
|
|
|
log.Warn("buildFilesInBackground", "err", err)
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
}()
|
2022-10-15 01:21:15 +00:00
|
|
|
|
|
|
|
if err := a.prune(0, a.maxTxNum.Load(), a.aggregationStep); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-07-23 08:06:52 +00:00
|
|
|
closeAll = false
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-07-28 07:47:13 +00:00
|
|
|
func (a *Aggregator22) AddAccountPrev(addr []byte, prev []byte) error {
|
|
|
|
if err := a.accounts.AddPrevValue(addr, nil, prev); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) AddStoragePrev(addr []byte, loc []byte, prev []byte) error {
|
|
|
|
if err := a.storage.AddPrevValue(addr, loc, prev); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-08-15 03:27:08 +00:00
|
|
|
// AddCodePrev - addr+inc => code
|
2022-07-28 07:47:13 +00:00
|
|
|
func (a *Aggregator22) AddCodePrev(addr []byte, prev []byte) error {
|
|
|
|
if err := a.code.AddPrevValue(addr, nil, prev); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-07-23 08:06:52 +00:00
|
|
|
func (a *Aggregator22) AddTraceFrom(addr []byte) error {
|
|
|
|
return a.tracesFrom.Add(addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) AddTraceTo(addr []byte) error {
|
|
|
|
return a.tracesTo.Add(addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) AddLogAddr(addr []byte) error {
|
|
|
|
return a.logAddrs.Add(addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) AddLogTopic(topic []byte) error {
|
|
|
|
return a.logTopics.Add(topic)
|
|
|
|
}
|
|
|
|
|
2022-10-05 06:17:23 +00:00
|
|
|
// DisableReadAhead - usage: `defer d.EnableReadAhead().DisableReadAhead()`. Please don't use this funcs without `defer` to avoid leak.
|
|
|
|
func (a *Aggregator22) DisableReadAhead() {
|
|
|
|
a.accounts.DisableReadAhead()
|
|
|
|
a.storage.DisableReadAhead()
|
|
|
|
a.code.DisableReadAhead()
|
|
|
|
a.logAddrs.DisableReadAhead()
|
|
|
|
a.logTopics.DisableReadAhead()
|
|
|
|
a.tracesFrom.DisableReadAhead()
|
|
|
|
a.tracesTo.DisableReadAhead()
|
|
|
|
}
|
|
|
|
func (a *Aggregator22) EnableReadAhead() *Aggregator22 {
|
|
|
|
a.accounts.EnableReadAhead()
|
|
|
|
a.storage.EnableReadAhead()
|
|
|
|
a.code.EnableReadAhead()
|
|
|
|
a.logAddrs.EnableReadAhead()
|
|
|
|
a.logTopics.EnableReadAhead()
|
|
|
|
a.tracesFrom.EnableReadAhead()
|
|
|
|
a.tracesTo.EnableReadAhead()
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
func (a *Aggregator22) EnableMadvWillNeed() *Aggregator22 {
|
|
|
|
a.accounts.EnableMadvWillNeed()
|
|
|
|
a.storage.EnableMadvWillNeed()
|
|
|
|
a.code.EnableMadvWillNeed()
|
|
|
|
a.logAddrs.EnableMadvWillNeed()
|
|
|
|
a.logTopics.EnableMadvWillNeed()
|
|
|
|
a.tracesFrom.EnableMadvWillNeed()
|
|
|
|
a.tracesTo.EnableMadvWillNeed()
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
func (a *Aggregator22) EnableMadvNormal() *Aggregator22 {
|
|
|
|
a.accounts.EnableMadvNormalReadAhead()
|
|
|
|
a.storage.EnableMadvNormalReadAhead()
|
|
|
|
a.code.EnableMadvNormalReadAhead()
|
|
|
|
a.logAddrs.EnableMadvNormalReadAhead()
|
|
|
|
a.logTopics.EnableMadvNormalReadAhead()
|
|
|
|
a.tracesFrom.EnableMadvNormalReadAhead()
|
|
|
|
a.tracesTo.EnableMadvNormalReadAhead()
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
|
2022-07-23 08:06:52 +00:00
|
|
|
func (ac *Aggregator22Context) LogAddrIterator(addr []byte, startTxNum, endTxNum uint64, roTx kv.Tx) InvertedIterator {
|
|
|
|
return ac.logAddrs.IterateRange(addr, startTxNum, endTxNum, roTx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ac *Aggregator22Context) LogTopicIterator(topic []byte, startTxNum, endTxNum uint64, roTx kv.Tx) InvertedIterator {
|
|
|
|
return ac.logTopics.IterateRange(topic, startTxNum, endTxNum, roTx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ac *Aggregator22Context) TraceFromIterator(addr []byte, startTxNum, endTxNum uint64, roTx kv.Tx) InvertedIterator {
|
|
|
|
return ac.tracesFrom.IterateRange(addr, startTxNum, endTxNum, roTx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ac *Aggregator22Context) TraceToIterator(addr []byte, startTxNum, endTxNum uint64, roTx kv.Tx) InvertedIterator {
|
|
|
|
return ac.tracesTo.IterateRange(addr, startTxNum, endTxNum, roTx)
|
|
|
|
}
|
|
|
|
|
2022-08-14 13:56:47 +00:00
|
|
|
func (ac *Aggregator22Context) IsMaxAccountsTxNum(addr []byte, txNum uint64) bool {
|
|
|
|
return ac.accounts.IsMaxTxNum(addr, txNum)
|
2022-07-28 07:47:13 +00:00
|
|
|
}
|
|
|
|
|
2022-08-14 13:56:47 +00:00
|
|
|
func (ac *Aggregator22Context) IsMaxStorageTxNum(addr []byte, loc []byte, txNum uint64) bool {
|
2022-07-28 07:47:13 +00:00
|
|
|
if cap(ac.keyBuf) < len(addr)+len(loc) {
|
|
|
|
ac.keyBuf = make([]byte, len(addr)+len(loc))
|
|
|
|
} else if len(ac.keyBuf) != len(addr)+len(loc) {
|
|
|
|
ac.keyBuf = ac.keyBuf[:len(addr)+len(loc)]
|
|
|
|
}
|
|
|
|
copy(ac.keyBuf, addr)
|
|
|
|
copy(ac.keyBuf[len(addr):], loc)
|
2022-08-14 13:56:47 +00:00
|
|
|
return ac.storage.IsMaxTxNum(ac.keyBuf, txNum)
|
2022-07-28 07:47:13 +00:00
|
|
|
}
|
|
|
|
|
2022-08-14 13:56:47 +00:00
|
|
|
func (ac *Aggregator22Context) IsMaxCodeTxNum(addr []byte, txNum uint64) bool {
|
|
|
|
return ac.code.IsMaxTxNum(addr, txNum)
|
2022-07-28 07:47:13 +00:00
|
|
|
}
|
|
|
|
|
2022-08-28 04:25:53 +00:00
|
|
|
func (ac *Aggregator22Context) ReadAccountDataNoStateWithRecent(addr []byte, txNum uint64) ([]byte, bool, error) {
|
|
|
|
return ac.accounts.GetNoStateWithRecent(addr, txNum, ac.tx)
|
|
|
|
}
|
|
|
|
|
2022-08-14 13:56:47 +00:00
|
|
|
func (ac *Aggregator22Context) ReadAccountDataNoState(addr []byte, txNum uint64) ([]byte, bool, error) {
|
2022-07-28 07:47:13 +00:00
|
|
|
return ac.accounts.GetNoState(addr, txNum)
|
|
|
|
}
|
|
|
|
|
2022-08-28 04:25:53 +00:00
|
|
|
func (ac *Aggregator22Context) ReadAccountStorageNoStateWithRecent(addr []byte, loc []byte, txNum uint64) ([]byte, bool, error) {
|
|
|
|
if cap(ac.keyBuf) < len(addr)+len(loc) {
|
|
|
|
ac.keyBuf = make([]byte, len(addr)+len(loc))
|
|
|
|
} else if len(ac.keyBuf) != len(addr)+len(loc) {
|
|
|
|
ac.keyBuf = ac.keyBuf[:len(addr)+len(loc)]
|
|
|
|
}
|
|
|
|
copy(ac.keyBuf, addr)
|
|
|
|
copy(ac.keyBuf[len(addr):], loc)
|
|
|
|
return ac.storage.GetNoStateWithRecent(ac.keyBuf, txNum, ac.tx)
|
|
|
|
}
|
|
|
|
|
2022-08-14 13:56:47 +00:00
|
|
|
func (ac *Aggregator22Context) ReadAccountStorageNoState(addr []byte, loc []byte, txNum uint64) ([]byte, bool, error) {
|
2022-07-28 07:47:13 +00:00
|
|
|
if cap(ac.keyBuf) < len(addr)+len(loc) {
|
|
|
|
ac.keyBuf = make([]byte, len(addr)+len(loc))
|
|
|
|
} else if len(ac.keyBuf) != len(addr)+len(loc) {
|
|
|
|
ac.keyBuf = ac.keyBuf[:len(addr)+len(loc)]
|
|
|
|
}
|
|
|
|
copy(ac.keyBuf, addr)
|
|
|
|
copy(ac.keyBuf[len(addr):], loc)
|
|
|
|
return ac.storage.GetNoState(ac.keyBuf, txNum)
|
|
|
|
}
|
|
|
|
|
2022-08-28 04:25:53 +00:00
|
|
|
func (ac *Aggregator22Context) ReadAccountCodeNoStateWithRecent(addr []byte, txNum uint64) ([]byte, bool, error) {
|
|
|
|
return ac.code.GetNoStateWithRecent(addr, txNum, ac.tx)
|
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
func (ac *Aggregator22Context) ReadAccountCodeNoState(addr []byte, txNum uint64) ([]byte, bool, error) {
|
2022-07-28 07:47:13 +00:00
|
|
|
return ac.code.GetNoState(addr, txNum)
|
|
|
|
}
|
|
|
|
|
2022-08-28 04:25:53 +00:00
|
|
|
func (ac *Aggregator22Context) ReadAccountCodeSizeNoStateWithRecent(addr []byte, txNum uint64) (int, bool, error) {
|
|
|
|
code, noState, err := ac.code.GetNoStateWithRecent(addr, txNum, ac.tx)
|
|
|
|
if err != nil {
|
|
|
|
return 0, false, err
|
|
|
|
}
|
|
|
|
return len(code), noState, nil
|
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
func (ac *Aggregator22Context) ReadAccountCodeSizeNoState(addr []byte, txNum uint64) (int, bool, error) {
|
|
|
|
code, noState, err := ac.code.GetNoState(addr, txNum)
|
2022-07-28 07:47:13 +00:00
|
|
|
if err != nil {
|
2022-08-14 13:56:47 +00:00
|
|
|
return 0, false, err
|
2022-07-28 07:47:13 +00:00
|
|
|
}
|
2022-08-14 13:56:47 +00:00
|
|
|
return len(code), noState, nil
|
2022-07-28 07:47:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ac *Aggregator22Context) IterateAccountsHistory(fromKey, toKey []byte, txNum uint64) *HistoryIterator {
|
|
|
|
return ac.accounts.iterateHistoryBeforeTxNum(fromKey, toKey, txNum)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ac *Aggregator22Context) IterateStorageHistory(fromKey, toKey []byte, txNum uint64) *HistoryIterator {
|
|
|
|
return ac.storage.iterateHistoryBeforeTxNum(fromKey, toKey, txNum)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ac *Aggregator22Context) IterateCodeHistory(fromKey, toKey []byte, txNum uint64) *HistoryIterator {
|
|
|
|
return ac.code.iterateHistoryBeforeTxNum(fromKey, toKey, txNum)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ac *Aggregator22Context) IterateAccountsReconTxs(fromKey, toKey []byte, txNum uint64) *ScanIterator {
|
|
|
|
return ac.accounts.iterateReconTxs(fromKey, toKey, txNum)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ac *Aggregator22Context) IterateStorageReconTxs(fromKey, toKey []byte, txNum uint64) *ScanIterator {
|
|
|
|
return ac.storage.iterateReconTxs(fromKey, toKey, txNum)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ac *Aggregator22Context) IterateCodeReconTxs(fromKey, toKey []byte, txNum uint64) *ScanIterator {
|
|
|
|
return ac.code.iterateReconTxs(fromKey, toKey, txNum)
|
|
|
|
}
|
|
|
|
|
2022-07-23 08:06:52 +00:00
|
|
|
type FilesStats22 struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) Stats() FilesStats22 {
|
|
|
|
var fs FilesStats22
|
|
|
|
return fs
|
|
|
|
}
|
|
|
|
|
2022-08-19 04:23:56 +00:00
|
|
|
func (a *Aggregator22) Code() *History { return a.code }
|
|
|
|
func (a *Aggregator22) Accounts() *History { return a.accounts }
|
|
|
|
func (a *Aggregator22) Storage() *History { return a.storage }
|
|
|
|
|
2022-07-23 08:06:52 +00:00
|
|
|
type Aggregator22Context struct {
|
2022-10-21 08:31:23 +00:00
|
|
|
tx kv.Tx
|
2022-07-23 08:06:52 +00:00
|
|
|
a *Aggregator22
|
|
|
|
accounts *HistoryContext
|
|
|
|
storage *HistoryContext
|
|
|
|
code *HistoryContext
|
|
|
|
logAddrs *InvertedIndexContext
|
|
|
|
logTopics *InvertedIndexContext
|
|
|
|
tracesFrom *InvertedIndexContext
|
|
|
|
tracesTo *InvertedIndexContext
|
2022-10-21 08:31:23 +00:00
|
|
|
keyBuf []byte
|
2022-07-23 08:06:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Aggregator22) MakeContext() *Aggregator22Context {
|
|
|
|
return &Aggregator22Context{
|
|
|
|
a: a,
|
|
|
|
accounts: a.accounts.MakeContext(),
|
|
|
|
storage: a.storage.MakeContext(),
|
|
|
|
code: a.code.MakeContext(),
|
|
|
|
logAddrs: a.logAddrs.MakeContext(),
|
|
|
|
logTopics: a.logTopics.MakeContext(),
|
|
|
|
tracesFrom: a.tracesFrom.MakeContext(),
|
|
|
|
tracesTo: a.tracesTo.MakeContext(),
|
|
|
|
}
|
|
|
|
}
|
2022-08-28 04:25:53 +00:00
|
|
|
func (ac *Aggregator22Context) SetTx(tx kv.Tx) { ac.tx = tx }
|
2022-10-02 03:03:49 +00:00
|
|
|
|
|
|
|
// BackgroundResult - used only indicate that some work is done
|
|
|
|
// no much reason to pass exact results by this object, just get latest state when need
|
|
|
|
type BackgroundResult struct {
|
|
|
|
err error
|
2022-10-21 08:31:23 +00:00
|
|
|
has bool
|
2022-10-02 03:03:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (br *BackgroundResult) Has() bool { return br.has }
|
|
|
|
func (br *BackgroundResult) Set(err error) { br.has, br.err = true, err }
|
|
|
|
func (br *BackgroundResult) GetAndReset() (bool, error) {
|
|
|
|
has, err := br.has, br.err
|
|
|
|
br.has, br.err = false, nil
|
|
|
|
return has, err
|
|
|
|
}
|