forked from cerc-io/ipld-eth-server
(VDB-371) Recheck queued storage
- Iterate through queued storage at defined interval, popping rows from the queue if successfully persisted
This commit is contained in:
parent
bf4b1687a0
commit
6a86de87b4
@ -27,6 +27,7 @@ import (
|
|||||||
|
|
||||||
"github.com/vulcanize/vulcanizedb/libraries/shared/constants"
|
"github.com/vulcanize/vulcanizedb/libraries/shared/constants"
|
||||||
"github.com/vulcanize/vulcanizedb/libraries/shared/fetcher"
|
"github.com/vulcanize/vulcanizedb/libraries/shared/fetcher"
|
||||||
|
storageUtils "github.com/vulcanize/vulcanizedb/libraries/shared/storage/utils"
|
||||||
"github.com/vulcanize/vulcanizedb/libraries/shared/transformer"
|
"github.com/vulcanize/vulcanizedb/libraries/shared/transformer"
|
||||||
"github.com/vulcanize/vulcanizedb/libraries/shared/watcher"
|
"github.com/vulcanize/vulcanizedb/libraries/shared/watcher"
|
||||||
"github.com/vulcanize/vulcanizedb/pkg/fs"
|
"github.com/vulcanize/vulcanizedb/pkg/fs"
|
||||||
@ -168,7 +169,9 @@ func watchEthStorage(w *watcher.StorageWatcher, wg *syn.WaitGroup) {
|
|||||||
ticker := time.NewTicker(pollingInterval)
|
ticker := time.NewTicker(pollingInterval)
|
||||||
defer ticker.Stop()
|
defer ticker.Stop()
|
||||||
for range ticker.C {
|
for range ticker.C {
|
||||||
w.Execute()
|
errs := make(chan error)
|
||||||
|
rows := make(chan storageUtils.StorageDiffRow)
|
||||||
|
w.Execute(rows, errs, queueRecheckInterval)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -49,8 +49,9 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
pollingInterval = 7 * time.Second
|
pollingInterval = 7 * time.Second
|
||||||
validationWindow = 15
|
queueRecheckInterval = 5 * time.Minute
|
||||||
|
validationWindow = 15
|
||||||
)
|
)
|
||||||
|
|
||||||
var rootCmd = &cobra.Command{
|
var rootCmd = &cobra.Command{
|
||||||
|
@ -30,39 +30,35 @@ var _ = Describe("Csv Tail Storage Fetcher", func() {
|
|||||||
storageFetcher = fetcher.NewCsvTailStorageFetcher(mockTailer)
|
storageFetcher = fetcher.NewCsvTailStorageFetcher(mockTailer)
|
||||||
})
|
})
|
||||||
|
|
||||||
It("adds error to errors channel if tailing file fails", func() {
|
It("adds error to errors channel if tailing file fails", func(done Done) {
|
||||||
mockTailer.TailErr = fakes.FakeError
|
mockTailer.TailErr = fakes.FakeError
|
||||||
|
|
||||||
go storageFetcher.FetchStorageDiffs(rowsChannel, errorsChannel)
|
go storageFetcher.FetchStorageDiffs(rowsChannel, errorsChannel)
|
||||||
|
|
||||||
close(mockTailer.Lines)
|
Expect(<-errorsChannel).To(MatchError(fakes.FakeError))
|
||||||
returnedErr := <-errorsChannel
|
close(done)
|
||||||
Expect(returnedErr).To(HaveOccurred())
|
|
||||||
Expect(returnedErr).To(MatchError(fakes.FakeError))
|
|
||||||
})
|
})
|
||||||
|
|
||||||
It("adds parsed csv row to rows channel for storage diff", func() {
|
It("adds parsed csv row to rows channel for storage diff", func(done Done) {
|
||||||
line := getFakeLine()
|
line := getFakeLine()
|
||||||
|
|
||||||
go storageFetcher.FetchStorageDiffs(rowsChannel, errorsChannel)
|
go storageFetcher.FetchStorageDiffs(rowsChannel, errorsChannel)
|
||||||
mockTailer.Lines <- line
|
mockTailer.Lines <- line
|
||||||
|
|
||||||
close(mockTailer.Lines)
|
|
||||||
returnedRow := <-rowsChannel
|
|
||||||
expectedRow, err := utils.FromStrings(strings.Split(line.Text, ","))
|
expectedRow, err := utils.FromStrings(strings.Split(line.Text, ","))
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
Expect(expectedRow).To(Equal(returnedRow))
|
Expect(<-rowsChannel).To(Equal(expectedRow))
|
||||||
|
close(done)
|
||||||
})
|
})
|
||||||
|
|
||||||
It("adds error to errors channel if parsing csv fails", func() {
|
It("adds error to errors channel if parsing csv fails", func(done Done) {
|
||||||
line := &tail.Line{Text: "invalid"}
|
line := &tail.Line{Text: "invalid"}
|
||||||
|
|
||||||
go storageFetcher.FetchStorageDiffs(rowsChannel, errorsChannel)
|
go storageFetcher.FetchStorageDiffs(rowsChannel, errorsChannel)
|
||||||
mockTailer.Lines <- line
|
mockTailer.Lines <- line
|
||||||
|
|
||||||
close(mockTailer.Lines)
|
Expect(<-errorsChannel).To(HaveOccurred())
|
||||||
returnedErr := <-errorsChannel
|
close(done)
|
||||||
Expect(returnedErr).To(HaveOccurred())
|
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -12,12 +12,12 @@ func NewMockStorageFetcher() *MockStorageFetcher {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (fetcher *MockStorageFetcher) FetchStorageDiffs(out chan<- utils.StorageDiffRow, errs chan<- error) {
|
func (fetcher *MockStorageFetcher) FetchStorageDiffs(out chan<- utils.StorageDiffRow, errs chan<- error) {
|
||||||
|
defer close(out)
|
||||||
|
defer close(errs)
|
||||||
for _, err := range fetcher.ErrsToReturn {
|
for _, err := range fetcher.ErrsToReturn {
|
||||||
errs <- err
|
errs <- err
|
||||||
}
|
}
|
||||||
for _, row := range fetcher.RowsToReturn {
|
for _, row := range fetcher.RowsToReturn {
|
||||||
out <- row
|
out <- row
|
||||||
}
|
}
|
||||||
close(out)
|
|
||||||
close(errs)
|
|
||||||
}
|
}
|
||||||
|
@ -21,21 +21,26 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type MockStorageQueue struct {
|
type MockStorageQueue struct {
|
||||||
AddCalled bool
|
AddCalled bool
|
||||||
AddError error
|
AddError error
|
||||||
PassedRow utils.StorageDiffRow
|
AddPassedRow utils.StorageDiffRow
|
||||||
|
DeleteErr error
|
||||||
|
DeletePassedId int
|
||||||
|
GetAllErr error
|
||||||
|
RowsToReturn []utils.StorageDiffRow
|
||||||
}
|
}
|
||||||
|
|
||||||
func (queue *MockStorageQueue) Add(row utils.StorageDiffRow) error {
|
func (queue *MockStorageQueue) Add(row utils.StorageDiffRow) error {
|
||||||
queue.AddCalled = true
|
queue.AddCalled = true
|
||||||
queue.PassedRow = row
|
queue.AddPassedRow = row
|
||||||
return queue.AddError
|
return queue.AddError
|
||||||
}
|
}
|
||||||
|
|
||||||
func (queue *MockStorageQueue) Delete(id int) error {
|
func (queue *MockStorageQueue) Delete(id int) error {
|
||||||
panic("implement me")
|
queue.DeletePassedId = id
|
||||||
|
return queue.DeleteErr
|
||||||
}
|
}
|
||||||
|
|
||||||
func (queue *MockStorageQueue) GetAll() ([]utils.StorageDiffRow, error) {
|
func (queue *MockStorageQueue) GetAll() ([]utils.StorageDiffRow, error) {
|
||||||
panic("implement me")
|
return queue.RowsToReturn, queue.GetAllErr
|
||||||
}
|
}
|
||||||
|
@ -4,10 +4,10 @@ import (
|
|||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
. "github.com/onsi/ginkgo"
|
. "github.com/onsi/ginkgo"
|
||||||
. "github.com/onsi/gomega"
|
. "github.com/onsi/gomega"
|
||||||
"github.com/vulcanize/vulcanizedb/pkg/datastore/postgres"
|
|
||||||
|
|
||||||
"github.com/vulcanize/vulcanizedb/libraries/shared/storage"
|
"github.com/vulcanize/vulcanizedb/libraries/shared/storage"
|
||||||
"github.com/vulcanize/vulcanizedb/libraries/shared/storage/utils"
|
"github.com/vulcanize/vulcanizedb/libraries/shared/storage/utils"
|
||||||
|
"github.com/vulcanize/vulcanizedb/pkg/datastore/postgres"
|
||||||
"github.com/vulcanize/vulcanizedb/test_config"
|
"github.com/vulcanize/vulcanizedb/test_config"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -17,7 +17,9 @@
|
|||||||
package watcher
|
package watcher
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/sirupsen/logrus"
|
"github.com/sirupsen/logrus"
|
||||||
@ -47,29 +49,30 @@ func NewStorageWatcher(fetcher fetcher.IStorageFetcher, db *postgres.DB) Storage
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (watcher StorageWatcher) AddTransformers(initializers []transformer.StorageTransformerInitializer) {
|
func (storageWatcher StorageWatcher) AddTransformers(initializers []transformer.StorageTransformerInitializer) {
|
||||||
for _, initializer := range initializers {
|
for _, initializer := range initializers {
|
||||||
storageTransformer := initializer(watcher.db)
|
storageTransformer := initializer(storageWatcher.db)
|
||||||
watcher.Transformers[storageTransformer.ContractAddress()] = storageTransformer
|
storageWatcher.Transformers[storageTransformer.ContractAddress()] = storageTransformer
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (watcher StorageWatcher) Execute() error {
|
func (storageWatcher StorageWatcher) Execute(rows chan utils.StorageDiffRow, errs chan error, queueRecheckInterval time.Duration) {
|
||||||
rows := make(chan utils.StorageDiffRow)
|
ticker := time.NewTicker(queueRecheckInterval)
|
||||||
errs := make(chan error)
|
go storageWatcher.StorageFetcher.FetchStorageDiffs(rows, errs)
|
||||||
go watcher.StorageFetcher.FetchStorageDiffs(rows, errs)
|
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
|
case fetchErr := <-errs:
|
||||||
|
logrus.Warn(fmt.Sprintf("error fetching storage diffs: %s", fetchErr))
|
||||||
case row := <-rows:
|
case row := <-rows:
|
||||||
watcher.processRow(row)
|
storageWatcher.processRow(row)
|
||||||
case err := <-errs:
|
case <-ticker.C:
|
||||||
return err
|
storageWatcher.processQueue()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (watcher StorageWatcher) processRow(row utils.StorageDiffRow) {
|
func (storageWatcher StorageWatcher) processRow(row utils.StorageDiffRow) {
|
||||||
storageTransformer, ok := watcher.Transformers[row.Contract]
|
storageTransformer, ok := storageWatcher.Transformers[row.Contract]
|
||||||
if !ok {
|
if !ok {
|
||||||
// ignore rows from unwatched contracts
|
// ignore rows from unwatched contracts
|
||||||
return
|
return
|
||||||
@ -77,16 +80,42 @@ func (watcher StorageWatcher) processRow(row utils.StorageDiffRow) {
|
|||||||
executeErr := storageTransformer.Execute(row)
|
executeErr := storageTransformer.Execute(row)
|
||||||
if executeErr != nil {
|
if executeErr != nil {
|
||||||
if isKeyNotFound(executeErr) {
|
if isKeyNotFound(executeErr) {
|
||||||
queueErr := watcher.Queue.Add(row)
|
queueErr := storageWatcher.Queue.Add(row)
|
||||||
if queueErr != nil {
|
if queueErr != nil {
|
||||||
logrus.Warn(queueErr.Error())
|
logrus.Warn(fmt.Sprintf("error queueing storage diff with unrecognized key: %s", queueErr))
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
logrus.Warn(executeErr.Error())
|
logrus.Warn(fmt.Sprintf("error executing storage transformer: %s", executeErr))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (storageWatcher StorageWatcher) processQueue() {
|
||||||
|
rows, fetchErr := storageWatcher.Queue.GetAll()
|
||||||
|
if fetchErr != nil {
|
||||||
|
logrus.Warn(fmt.Sprintf("error getting queued storage: %s", fetchErr))
|
||||||
|
}
|
||||||
|
for _, row := range rows {
|
||||||
|
storageTransformer, ok := storageWatcher.Transformers[row.Contract]
|
||||||
|
if !ok {
|
||||||
|
// delete row from queue if address no longer watched
|
||||||
|
storageWatcher.deleteRow(row.Id)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
executeErr := storageTransformer.Execute(row)
|
||||||
|
if executeErr == nil {
|
||||||
|
storageWatcher.deleteRow(row.Id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (storageWatcher StorageWatcher) deleteRow(id int) {
|
||||||
|
deleteErr := storageWatcher.Queue.Delete(id)
|
||||||
|
if deleteErr != nil {
|
||||||
|
logrus.Warn(fmt.Sprintf("error deleting persisted row from queue: %s", deleteErr))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func isKeyNotFound(executeErr error) bool {
|
func isKeyNotFound(executeErr error) bool {
|
||||||
return reflect.TypeOf(executeErr) == reflect.TypeOf(utils.ErrStorageKeyNotFound{})
|
return reflect.TypeOf(executeErr) == reflect.TypeOf(utils.ErrStorageKeyNotFound{})
|
||||||
}
|
}
|
||||||
|
@ -19,6 +19,7 @@ package watcher_test
|
|||||||
import (
|
import (
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
. "github.com/onsi/ginkgo"
|
. "github.com/onsi/ginkgo"
|
||||||
@ -46,78 +47,214 @@ var _ = Describe("Storage Watcher", func() {
|
|||||||
|
|
||||||
Describe("executing watcher", func() {
|
Describe("executing watcher", func() {
|
||||||
var (
|
var (
|
||||||
|
errs chan error
|
||||||
mockFetcher *mocks.MockStorageFetcher
|
mockFetcher *mocks.MockStorageFetcher
|
||||||
mockQueue *mocks.MockStorageQueue
|
mockQueue *mocks.MockStorageQueue
|
||||||
mockTransformer *mocks.MockStorageTransformer
|
mockTransformer *mocks.MockStorageTransformer
|
||||||
row utils.StorageDiffRow
|
row utils.StorageDiffRow
|
||||||
|
rows chan utils.StorageDiffRow
|
||||||
storageWatcher watcher.StorageWatcher
|
storageWatcher watcher.StorageWatcher
|
||||||
)
|
)
|
||||||
|
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
|
errs = make(chan error)
|
||||||
|
rows = make(chan utils.StorageDiffRow)
|
||||||
address := common.HexToAddress("0x0123456789abcdef")
|
address := common.HexToAddress("0x0123456789abcdef")
|
||||||
mockFetcher = mocks.NewMockStorageFetcher()
|
mockFetcher = mocks.NewMockStorageFetcher()
|
||||||
mockQueue = &mocks.MockStorageQueue{}
|
mockQueue = &mocks.MockStorageQueue{}
|
||||||
mockTransformer = &mocks.MockStorageTransformer{Address: address}
|
mockTransformer = &mocks.MockStorageTransformer{Address: address}
|
||||||
row = utils.StorageDiffRow{
|
row = utils.StorageDiffRow{
|
||||||
|
Id: 1337,
|
||||||
Contract: address,
|
Contract: address,
|
||||||
BlockHash: common.HexToHash("0xfedcba9876543210"),
|
BlockHash: common.HexToHash("0xfedcba9876543210"),
|
||||||
BlockHeight: 0,
|
BlockHeight: 0,
|
||||||
StorageKey: common.HexToHash("0xabcdef1234567890"),
|
StorageKey: common.HexToHash("0xabcdef1234567890"),
|
||||||
StorageValue: common.HexToHash("0x9876543210abcdef"),
|
StorageValue: common.HexToHash("0x9876543210abcdef"),
|
||||||
}
|
}
|
||||||
mockFetcher.RowsToReturn = []utils.StorageDiffRow{row}
|
})
|
||||||
|
|
||||||
|
It("logs error if fetching storage diffs fails", func(done Done) {
|
||||||
|
mockFetcher.ErrsToReturn = []error{fakes.FakeError}
|
||||||
storageWatcher = watcher.NewStorageWatcher(mockFetcher, test_config.NewTestDB(test_config.NewTestNode()))
|
storageWatcher = watcher.NewStorageWatcher(mockFetcher, test_config.NewTestDB(test_config.NewTestNode()))
|
||||||
storageWatcher.Queue = mockQueue
|
storageWatcher.Queue = mockQueue
|
||||||
storageWatcher.AddTransformers([]transformer.StorageTransformerInitializer{mockTransformer.FakeTransformerInitializer})
|
storageWatcher.AddTransformers([]transformer.StorageTransformerInitializer{mockTransformer.FakeTransformerInitializer})
|
||||||
})
|
|
||||||
|
|
||||||
It("executes transformer for recognized storage row", func() {
|
|
||||||
err := storageWatcher.Execute()
|
|
||||||
|
|
||||||
Expect(err).NotTo(HaveOccurred())
|
|
||||||
Expect(mockTransformer.PassedRow).To(Equal(row))
|
|
||||||
})
|
|
||||||
|
|
||||||
It("queues row for later processing if row's key not recognized", func() {
|
|
||||||
mockTransformer.ExecuteErr = utils.ErrStorageKeyNotFound{}
|
|
||||||
|
|
||||||
err := storageWatcher.Execute()
|
|
||||||
|
|
||||||
Expect(err).NotTo(HaveOccurred())
|
|
||||||
Expect(mockQueue.AddCalled).To(BeTrue())
|
|
||||||
Expect(mockQueue.PassedRow).To(Equal(row))
|
|
||||||
})
|
|
||||||
|
|
||||||
It("logs error if queueing row fails", func() {
|
|
||||||
mockTransformer.ExecuteErr = utils.ErrStorageKeyNotFound{}
|
|
||||||
mockQueue.AddError = fakes.FakeError
|
|
||||||
tempFile, fileErr := ioutil.TempFile("", "log")
|
tempFile, fileErr := ioutil.TempFile("", "log")
|
||||||
Expect(fileErr).NotTo(HaveOccurred())
|
Expect(fileErr).NotTo(HaveOccurred())
|
||||||
defer os.Remove(tempFile.Name())
|
defer os.Remove(tempFile.Name())
|
||||||
logrus.SetOutput(tempFile)
|
logrus.SetOutput(tempFile)
|
||||||
|
|
||||||
err := storageWatcher.Execute()
|
go storageWatcher.Execute(rows, errs, time.Hour)
|
||||||
|
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Eventually(func() (string, error) {
|
||||||
Expect(mockQueue.AddCalled).To(BeTrue())
|
logContent, err := ioutil.ReadFile(tempFile.Name())
|
||||||
logContent, readErr := ioutil.ReadFile(tempFile.Name())
|
return string(logContent), err
|
||||||
Expect(readErr).NotTo(HaveOccurred())
|
}).Should(ContainSubstring(fakes.FakeError.Error()))
|
||||||
Expect(string(logContent)).To(ContainSubstring(fakes.FakeError.Error()))
|
close(done)
|
||||||
})
|
})
|
||||||
|
|
||||||
It("logs error if transformer execution fails for reason other than key not found", func() {
|
Describe("transforming new storage diffs", func() {
|
||||||
mockTransformer.ExecuteErr = fakes.FakeError
|
BeforeEach(func() {
|
||||||
tempFile, fileErr := ioutil.TempFile("", "log")
|
mockFetcher.RowsToReturn = []utils.StorageDiffRow{row}
|
||||||
Expect(fileErr).NotTo(HaveOccurred())
|
storageWatcher = watcher.NewStorageWatcher(mockFetcher, test_config.NewTestDB(test_config.NewTestNode()))
|
||||||
defer os.Remove(tempFile.Name())
|
storageWatcher.Queue = mockQueue
|
||||||
logrus.SetOutput(tempFile)
|
storageWatcher.AddTransformers([]transformer.StorageTransformerInitializer{mockTransformer.FakeTransformerInitializer})
|
||||||
|
})
|
||||||
|
|
||||||
err := storageWatcher.Execute()
|
It("executes transformer for recognized storage row", func(done Done) {
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Hour)
|
||||||
|
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Eventually(func() utils.StorageDiffRow {
|
||||||
logContent, readErr := ioutil.ReadFile(tempFile.Name())
|
return mockTransformer.PassedRow
|
||||||
Expect(readErr).NotTo(HaveOccurred())
|
}).Should(Equal(row))
|
||||||
Expect(string(logContent)).To(ContainSubstring(fakes.FakeError.Error()))
|
close(done)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("queues row for later processing if row's key not recognized", func(done Done) {
|
||||||
|
mockTransformer.ExecuteErr = utils.ErrStorageKeyNotFound{}
|
||||||
|
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Hour)
|
||||||
|
|
||||||
|
Expect(<-errs).To(BeNil())
|
||||||
|
Eventually(func() bool {
|
||||||
|
return mockQueue.AddCalled
|
||||||
|
}).Should(BeTrue())
|
||||||
|
Eventually(func() utils.StorageDiffRow {
|
||||||
|
return mockQueue.AddPassedRow
|
||||||
|
}).Should(Equal(row))
|
||||||
|
close(done)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("logs error if queueing row fails", func(done Done) {
|
||||||
|
mockTransformer.ExecuteErr = utils.ErrStorageKeyNotFound{}
|
||||||
|
mockQueue.AddError = fakes.FakeError
|
||||||
|
tempFile, fileErr := ioutil.TempFile("", "log")
|
||||||
|
Expect(fileErr).NotTo(HaveOccurred())
|
||||||
|
defer os.Remove(tempFile.Name())
|
||||||
|
logrus.SetOutput(tempFile)
|
||||||
|
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Hour)
|
||||||
|
|
||||||
|
Eventually(func() bool {
|
||||||
|
return mockQueue.AddCalled
|
||||||
|
}).Should(BeTrue())
|
||||||
|
Eventually(func() (string, error) {
|
||||||
|
logContent, err := ioutil.ReadFile(tempFile.Name())
|
||||||
|
return string(logContent), err
|
||||||
|
}).Should(ContainSubstring(fakes.FakeError.Error()))
|
||||||
|
close(done)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("logs error if transformer execution fails for reason other than key not found", func(done Done) {
|
||||||
|
mockTransformer.ExecuteErr = fakes.FakeError
|
||||||
|
tempFile, fileErr := ioutil.TempFile("", "log")
|
||||||
|
Expect(fileErr).NotTo(HaveOccurred())
|
||||||
|
defer os.Remove(tempFile.Name())
|
||||||
|
logrus.SetOutput(tempFile)
|
||||||
|
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Hour)
|
||||||
|
|
||||||
|
Eventually(func() (string, error) {
|
||||||
|
logContent, err := ioutil.ReadFile(tempFile.Name())
|
||||||
|
return string(logContent), err
|
||||||
|
}).Should(ContainSubstring(fakes.FakeError.Error()))
|
||||||
|
close(done)
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Describe("transforming queued storage diffs", func() {
|
||||||
|
BeforeEach(func() {
|
||||||
|
mockQueue.RowsToReturn = []utils.StorageDiffRow{row}
|
||||||
|
storageWatcher = watcher.NewStorageWatcher(mockFetcher, test_config.NewTestDB(test_config.NewTestNode()))
|
||||||
|
storageWatcher.Queue = mockQueue
|
||||||
|
storageWatcher.AddTransformers([]transformer.StorageTransformerInitializer{mockTransformer.FakeTransformerInitializer})
|
||||||
|
})
|
||||||
|
|
||||||
|
It("logs error if getting queued storage fails", func(done Done) {
|
||||||
|
mockQueue.GetAllErr = fakes.FakeError
|
||||||
|
tempFile, fileErr := ioutil.TempFile("", "log")
|
||||||
|
Expect(fileErr).NotTo(HaveOccurred())
|
||||||
|
defer os.Remove(tempFile.Name())
|
||||||
|
logrus.SetOutput(tempFile)
|
||||||
|
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Nanosecond)
|
||||||
|
|
||||||
|
Eventually(func() (string, error) {
|
||||||
|
logContent, err := ioutil.ReadFile(tempFile.Name())
|
||||||
|
return string(logContent), err
|
||||||
|
}).Should(ContainSubstring(fakes.FakeError.Error()))
|
||||||
|
close(done)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("executes transformer for storage row", func(done Done) {
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Nanosecond)
|
||||||
|
|
||||||
|
Eventually(func() utils.StorageDiffRow {
|
||||||
|
return mockTransformer.PassedRow
|
||||||
|
}).Should(Equal(row))
|
||||||
|
close(done)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("deletes row from queue if transformer execution successful", func(done Done) {
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Nanosecond)
|
||||||
|
|
||||||
|
Eventually(func() int {
|
||||||
|
return mockQueue.DeletePassedId
|
||||||
|
}).Should(Equal(row.Id))
|
||||||
|
close(done)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("logs error if deleting persisted row fails", func(done Done) {
|
||||||
|
mockQueue.DeleteErr = fakes.FakeError
|
||||||
|
tempFile, fileErr := ioutil.TempFile("", "log")
|
||||||
|
Expect(fileErr).NotTo(HaveOccurred())
|
||||||
|
defer os.Remove(tempFile.Name())
|
||||||
|
logrus.SetOutput(tempFile)
|
||||||
|
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Nanosecond)
|
||||||
|
|
||||||
|
Eventually(func() (string, error) {
|
||||||
|
logContent, err := ioutil.ReadFile(tempFile.Name())
|
||||||
|
return string(logContent), err
|
||||||
|
}).Should(ContainSubstring(fakes.FakeError.Error()))
|
||||||
|
close(done)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("deletes obsolete row from queue if contract not recognized", func(done Done) {
|
||||||
|
obsoleteRow := utils.StorageDiffRow{
|
||||||
|
Id: row.Id + 1,
|
||||||
|
Contract: common.HexToAddress("0xfedcba9876543210"),
|
||||||
|
}
|
||||||
|
mockQueue.RowsToReturn = []utils.StorageDiffRow{obsoleteRow}
|
||||||
|
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Nanosecond)
|
||||||
|
|
||||||
|
Eventually(func() int {
|
||||||
|
return mockQueue.DeletePassedId
|
||||||
|
}).Should(Equal(obsoleteRow.Id))
|
||||||
|
close(done)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("logs error if deleting obsolete row fails", func(done Done) {
|
||||||
|
obsoleteRow := utils.StorageDiffRow{
|
||||||
|
Id: row.Id + 1,
|
||||||
|
Contract: common.HexToAddress("0xfedcba9876543210"),
|
||||||
|
}
|
||||||
|
mockQueue.RowsToReturn = []utils.StorageDiffRow{obsoleteRow}
|
||||||
|
mockQueue.DeleteErr = fakes.FakeError
|
||||||
|
tempFile, fileErr := ioutil.TempFile("", "log")
|
||||||
|
Expect(fileErr).NotTo(HaveOccurred())
|
||||||
|
defer os.Remove(tempFile.Name())
|
||||||
|
logrus.SetOutput(tempFile)
|
||||||
|
|
||||||
|
go storageWatcher.Execute(rows, errs, time.Nanosecond)
|
||||||
|
|
||||||
|
Eventually(func() (string, error) {
|
||||||
|
logContent, err := ioutil.ReadFile(tempFile.Name())
|
||||||
|
return string(logContent), err
|
||||||
|
}).Should(ContainSubstring(fakes.FakeError.Error()))
|
||||||
|
close(done)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
Loading…
Reference in New Issue
Block a user