2023-06-14 12:43:34 +00:00
|
|
|
// VulcanizeDB
|
|
|
|
// Copyright © 2021 Vulcanize
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
package dump
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
2023-08-06 05:29:19 +00:00
|
|
|
"math/big"
|
2023-06-14 12:43:34 +00:00
|
|
|
|
2023-06-23 12:42:55 +00:00
|
|
|
"github.com/cerc-io/plugeth-statediff/indexer/ipld"
|
|
|
|
"github.com/cerc-io/plugeth-statediff/indexer/models"
|
2023-08-06 05:29:19 +00:00
|
|
|
"github.com/cerc-io/plugeth-statediff/utils/log"
|
2023-06-14 12:43:34 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// BatchTx wraps a void with the state necessary for building the tx concurrently during trie difference iteration
|
|
|
|
type BatchTx struct {
|
2023-08-06 05:29:19 +00:00
|
|
|
blockNum string
|
|
|
|
dump io.Writer
|
|
|
|
quit chan struct{}
|
|
|
|
iplds chan models.IPLDModel
|
|
|
|
ipldCache models.IPLDBatch
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBatch(number *big.Int, dest io.Writer) *BatchTx {
|
|
|
|
batch := &BatchTx{
|
|
|
|
blockNum: number.String(),
|
|
|
|
dump: dest,
|
|
|
|
iplds: make(chan models.IPLDModel),
|
|
|
|
quit: make(chan struct{}),
|
|
|
|
ipldCache: models.IPLDBatch{},
|
|
|
|
}
|
|
|
|
go batch.cache()
|
|
|
|
return batch
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *BatchTx) Submit() error {
|
|
|
|
close(self.quit)
|
|
|
|
close(self.iplds)
|
2023-06-14 12:43:34 +00:00
|
|
|
|
2023-08-06 05:29:19 +00:00
|
|
|
if err := self.flush(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
2023-06-14 12:43:34 +00:00
|
|
|
}
|
|
|
|
|
2023-08-06 05:29:19 +00:00
|
|
|
func (tx *BatchTx) BlockNumber() string {
|
|
|
|
return tx.blockNum
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tx *BatchTx) RollbackOnFailure(err error) {
|
|
|
|
if p := recover(); p != nil {
|
|
|
|
log.Info("panic detected before tx submission, but rollback not supported", "panic", p)
|
|
|
|
panic(p)
|
|
|
|
} else if err != nil {
|
|
|
|
log.Info("error detected before tx submission, but rollback not supported", "error", err)
|
|
|
|
}
|
2023-06-14 12:43:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (tx *BatchTx) flush() error {
|
|
|
|
if _, err := fmt.Fprintf(tx.dump, "%+v\r\n", tx.ipldCache); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
tx.ipldCache = models.IPLDBatch{}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// run in background goroutine to synchronize concurrent appends to the ipldCache
|
|
|
|
func (tx *BatchTx) cache() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case i := <-tx.iplds:
|
|
|
|
tx.ipldCache.Keys = append(tx.ipldCache.Keys, i.Key)
|
|
|
|
tx.ipldCache.Values = append(tx.ipldCache.Values, i.Data)
|
|
|
|
case <-tx.quit:
|
|
|
|
tx.ipldCache = models.IPLDBatch{}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tx *BatchTx) cacheDirect(key string, value []byte) {
|
|
|
|
tx.iplds <- models.IPLDModel{
|
2023-08-06 05:29:19 +00:00
|
|
|
BlockNumber: tx.BlockNumber(),
|
2023-06-14 12:43:34 +00:00
|
|
|
Key: key,
|
|
|
|
Data: value,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tx *BatchTx) cacheIPLD(i ipld.IPLD) {
|
|
|
|
tx.iplds <- models.IPLDModel{
|
2023-08-06 05:29:19 +00:00
|
|
|
BlockNumber: tx.BlockNumber(),
|
2023-06-14 12:43:34 +00:00
|
|
|
Key: i.Cid().String(),
|
|
|
|
Data: i.RawData(),
|
|
|
|
}
|
|
|
|
}
|