2020-06-10 20:07:14 +00:00
|
|
|
package etl
|
|
|
|
|
|
|
|
import (
|
2020-08-17 06:45:52 +00:00
|
|
|
"bytes"
|
2020-06-10 20:07:14 +00:00
|
|
|
"container/heap"
|
|
|
|
"fmt"
|
2020-07-07 04:00:25 +00:00
|
|
|
"io"
|
|
|
|
"runtime"
|
2020-08-17 06:45:52 +00:00
|
|
|
"time"
|
2020-07-07 04:00:25 +00:00
|
|
|
|
2020-06-10 20:07:14 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/ethdb"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/log"
|
|
|
|
"github.com/ugorji/go/codec"
|
|
|
|
)
|
|
|
|
|
2020-07-01 14:56:56 +00:00
|
|
|
type LoadNextFunc func(originalK, k, v []byte) error
|
2020-06-10 20:07:14 +00:00
|
|
|
type LoadFunc func(k []byte, value []byte, state State, next LoadNextFunc) error
|
|
|
|
|
|
|
|
// Collector performs the job of ETL Transform, but can also be used without "E" (Extract) part
|
|
|
|
// as a Collect Transform Load
|
|
|
|
type Collector struct {
|
|
|
|
extractNextFunc ExtractNextFunc
|
|
|
|
flushBuffer func([]byte, bool) error
|
|
|
|
dataProviders []dataProvider
|
|
|
|
allFlushed bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewCollector(datadir string, sortableBuffer Buffer) *Collector {
|
|
|
|
c := &Collector{}
|
|
|
|
encoder := codec.NewEncoder(nil, &cbor)
|
|
|
|
|
|
|
|
c.flushBuffer = func(currentKey []byte, canStoreInRam bool) error {
|
|
|
|
if sortableBuffer.Len() == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
var provider dataProvider
|
|
|
|
var err error
|
|
|
|
sortableBuffer.Sort()
|
|
|
|
if canStoreInRam && len(c.dataProviders) == 0 {
|
|
|
|
provider = KeepInRAM(sortableBuffer)
|
|
|
|
c.allFlushed = true
|
|
|
|
} else {
|
|
|
|
provider, err = FlushToDisk(encoder, currentKey, sortableBuffer, datadir)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if provider != nil {
|
|
|
|
c.dataProviders = append(c.dataProviders, provider)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
c.extractNextFunc = func(originalK, k []byte, v []byte) error {
|
|
|
|
sortableBuffer.Put(common.CopyBytes(k), common.CopyBytes(v))
|
|
|
|
if sortableBuffer.CheckFlushSize() {
|
|
|
|
if err := c.flushBuffer(originalK, false); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Collector) Collect(k, v []byte) error {
|
|
|
|
return c.extractNextFunc(k, k, v)
|
|
|
|
}
|
|
|
|
|
2020-08-10 23:55:32 +00:00
|
|
|
func (c *Collector) Load(db ethdb.Database, toBucket string, loadFunc LoadFunc, args TransformArgs) error {
|
2020-06-10 20:07:14 +00:00
|
|
|
defer func() {
|
|
|
|
disposeProviders(c.dataProviders)
|
|
|
|
}()
|
|
|
|
if !c.allFlushed {
|
|
|
|
if err := c.flushBuffer(nil, true); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return loadFilesIntoBucket(db, toBucket, c.dataProviders, loadFunc, args)
|
|
|
|
}
|
|
|
|
|
2020-08-10 23:55:32 +00:00
|
|
|
func loadFilesIntoBucket(db ethdb.Database, bucket string, providers []dataProvider, loadFunc LoadFunc, args TransformArgs) error {
|
2020-06-10 20:07:14 +00:00
|
|
|
decoder := codec.NewDecoder(nil, &cbor)
|
|
|
|
var m runtime.MemStats
|
|
|
|
h := &Heap{}
|
|
|
|
heap.Init(h)
|
|
|
|
for i, provider := range providers {
|
|
|
|
if key, value, err := provider.Next(decoder); err == nil {
|
|
|
|
he := HeapElem{key, i, value}
|
|
|
|
heap.Push(h, he)
|
|
|
|
} else /* we must have at least one entry per file */ {
|
|
|
|
eee := fmt.Errorf("error reading first readers: n=%d current=%d provider=%s err=%v",
|
|
|
|
len(providers), i, provider, err)
|
|
|
|
panic(eee)
|
|
|
|
}
|
|
|
|
}
|
2020-08-17 06:45:52 +00:00
|
|
|
|
|
|
|
batch, err := db.Begin()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer batch.Rollback()
|
|
|
|
|
2020-06-10 20:07:14 +00:00
|
|
|
state := &bucketState{batch, bucket, args.Quit}
|
2020-08-17 06:45:52 +00:00
|
|
|
haveSortingGuaranties := isIdentityLoadFunc(loadFunc) // user-defined loadFunc may change ordering
|
|
|
|
var lastKey []byte
|
|
|
|
if bucket != "" { // passing empty bucket name is valid case for etl when DB modification is not expected
|
|
|
|
var errLast error
|
|
|
|
lastKey, _, errLast = batch.Last(bucket)
|
|
|
|
if errLast != nil {
|
|
|
|
return errLast
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var canUseAppend bool
|
2020-06-10 20:07:14 +00:00
|
|
|
|
2020-08-22 10:12:33 +00:00
|
|
|
logEvery := time.NewTicker(30 * time.Second)
|
|
|
|
defer logEvery.Stop()
|
|
|
|
|
2020-08-17 06:45:52 +00:00
|
|
|
i := 0
|
2020-07-01 14:56:56 +00:00
|
|
|
loadNextFunc := func(originalK, k, v []byte) error {
|
2020-08-17 06:45:52 +00:00
|
|
|
if i == 0 {
|
|
|
|
isEndOfBucket := lastKey == nil || bytes.Compare(lastKey, k) == -1
|
|
|
|
canUseAppend = haveSortingGuaranties && isEndOfBucket
|
|
|
|
}
|
|
|
|
i++
|
2020-08-22 10:12:33 +00:00
|
|
|
|
|
|
|
select {
|
|
|
|
default:
|
|
|
|
case <-logEvery.C:
|
|
|
|
logArs := []interface{}{"into", bucket}
|
|
|
|
if args.LogDetailsLoad != nil {
|
|
|
|
logArs = append(logArs, args.LogDetailsLoad(k, v)...)
|
|
|
|
} else {
|
|
|
|
logArs = append(logArs, "current key", makeCurrentKeyStr(k))
|
|
|
|
}
|
|
|
|
|
2020-08-17 06:45:52 +00:00
|
|
|
runtime.ReadMemStats(&m)
|
2020-08-22 10:12:33 +00:00
|
|
|
logArs = append(logArs, "alloc", common.StorageSize(m.Alloc), "sys", common.StorageSize(m.Sys), "numGC", int(m.NumGC))
|
|
|
|
log.Info("ETL [2/2] Loading", logArs...)
|
|
|
|
}
|
2020-08-17 06:45:52 +00:00
|
|
|
|
|
|
|
if canUseAppend && len(v) == 0 {
|
|
|
|
return nil // nothing to delete after end of bucket
|
|
|
|
}
|
2020-06-10 20:07:14 +00:00
|
|
|
if len(v) == 0 {
|
|
|
|
if err := batch.Delete(bucket, k); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-08-17 06:45:52 +00:00
|
|
|
return nil
|
2020-06-10 20:07:14 +00:00
|
|
|
}
|
2020-08-17 06:45:52 +00:00
|
|
|
if canUseAppend {
|
|
|
|
if err := batch.(*ethdb.TxDb).Append(bucket, k, v); err != nil {
|
2020-06-18 18:14:10 +00:00
|
|
|
return err
|
|
|
|
}
|
2020-08-17 06:45:52 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err := batch.Put(bucket, k, v); err != nil {
|
|
|
|
return err
|
2020-06-10 20:07:14 +00:00
|
|
|
}
|
2020-06-28 06:10:27 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// Main loading loop
|
|
|
|
for h.Len() > 0 {
|
|
|
|
if err := common.Stopped(args.Quit); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
element := (heap.Pop(h)).(HeapElem)
|
|
|
|
provider := providers[element.TimeIdx]
|
|
|
|
err := loadFunc(element.Key, element.Value, state, loadNextFunc)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2020-06-10 20:07:14 +00:00
|
|
|
}
|
2020-06-28 06:10:27 +00:00
|
|
|
if element.Key, element.Value, err = provider.Next(decoder); err == nil {
|
|
|
|
heap.Push(h, element)
|
|
|
|
} else if err != io.EOF {
|
|
|
|
return fmt.Errorf("error while reading next element from disk: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Final commit
|
|
|
|
if args.OnLoadCommit != nil {
|
|
|
|
if err := args.OnLoadCommit(batch, []byte{}, true); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2020-08-17 06:45:52 +00:00
|
|
|
commitTimer := time.Now()
|
2020-06-28 06:10:27 +00:00
|
|
|
if _, err := batch.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-08-17 06:45:52 +00:00
|
|
|
commitTook := time.Since(commitTimer)
|
|
|
|
|
2020-06-28 06:10:27 +00:00
|
|
|
runtime.ReadMemStats(&m)
|
2020-07-10 21:37:34 +00:00
|
|
|
log.Debug(
|
2020-06-28 06:10:27 +00:00
|
|
|
"Committed batch",
|
2020-08-10 23:55:32 +00:00
|
|
|
"bucket", bucket,
|
2020-08-17 06:45:52 +00:00
|
|
|
"commit", commitTook,
|
2020-08-22 10:12:33 +00:00
|
|
|
"records", i,
|
2020-06-28 06:10:27 +00:00
|
|
|
"current key", makeCurrentKeyStr(nil),
|
|
|
|
"alloc", common.StorageSize(m.Alloc), "sys", common.StorageSize(m.Sys), "numGC", int(m.NumGC))
|
2020-06-10 20:07:14 +00:00
|
|
|
|
2020-06-28 06:10:27 +00:00
|
|
|
return nil
|
2020-06-10 20:07:14 +00:00
|
|
|
}
|
2020-06-21 14:13:17 +00:00
|
|
|
|
|
|
|
func makeCurrentKeyStr(k []byte) string {
|
|
|
|
var currentKeyStr string
|
|
|
|
if k == nil {
|
|
|
|
currentKeyStr = "final"
|
|
|
|
} else if len(k) < 4 {
|
|
|
|
currentKeyStr = fmt.Sprintf("%x", k)
|
2020-08-17 06:45:52 +00:00
|
|
|
} else if k[0] == 0 && k[1] == 0 && k[2] == 0 && k[3] == 0 && len(k) >= 8 { // if key has leading zeroes, show a bit more info
|
|
|
|
currentKeyStr = fmt.Sprintf("%x...", k[:8])
|
2020-06-21 14:13:17 +00:00
|
|
|
} else {
|
|
|
|
currentKeyStr = fmt.Sprintf("%x...", k[:4])
|
|
|
|
}
|
|
|
|
return currentKeyStr
|
|
|
|
}
|