diff --git a/tracker/tracker.go b/tracker/tracker.go index f69ff52..50894c6 100644 --- a/tracker/tracker.go +++ b/tracker/tracker.go @@ -1,3 +1,27 @@ +// This package provides a way to track multiple concurrently running trie iterators, save their +// state to a file on failures or interruptions, and restore them at the positions where they +// stopped. +// +// Example usage: +// +// tr := tracker.New("recovery.txt", 100) +// // ensure the tracker is closed and saves its state +// defer tr.CloseAndSave() +// +// // iterate over the trie, from one or multiple threads +// it := tr.Tracked(tree.NodeIterator(nil)) +// for it.Next(true) { +// // ... do work that could fail or be interrupted +// } +// +// // later, restore the iterators +// tr := tracker.New("recovery.txt", 100) +// defer tr.CloseAndSave() +// +// its, err := tr.Restore(tree.NodeIterator) +// for _, it := range its { +// // ... resume traversal +// } package tracker import ( @@ -12,26 +36,30 @@ import ( iter "github.com/cerc-io/eth-iterator-utils" ) -type Tracker interface { +// IteratorTracker exposes a minimal interface to register and consume iterators. +type IteratorTracker interface { Restore(iter.IteratorConstructor) ([]trie.NodeIterator, error) Tracked(trie.NodeIterator) trie.NodeIterator } -var _ Tracker = &trackerAdaptor{} +var _ IteratorTracker = &Tracker{} -// Wrap the tracker state to only expose NodeIterators -type trackerAdaptor struct { +// Tracker is a trie iterator tracker which saves state to and restores it from a file. +type Tracker struct { *TrackerImpl } // New creates a new tracker which saves state to a given file. bufsize sets the size of the // channel buffers used internally to manage tracking. Note that passing a bufsize smaller than the expected // number of concurrent iterators could lead to deadlock. -func New(file string, bufsize uint) *trackerAdaptor { - return &trackerAdaptor{NewImpl(file, bufsize)} +func New(file string, bufsize uint) *Tracker { + return &Tracker{NewImpl(file, bufsize)} } -func (tr *trackerAdaptor) Restore(makeIterator iter.IteratorConstructor) ([]trie.NodeIterator, error) { +// Restore attempts to read iterator state from the recovery file. +// If the file doesn't exist, returns an empty slice with no error. +// Restored iterators are constructed in the same order they appear in the returned slice. +func (tr *Tracker) Restore(makeIterator iter.IteratorConstructor) ([]trie.NodeIterator, error) { its, err := tr.TrackerImpl.Restore(makeIterator) if err != nil { return nil, err @@ -44,9 +72,10 @@ func (tr *trackerAdaptor) Restore(makeIterator iter.IteratorConstructor) ([]trie return ret, nil } -func (tr *trackerAdaptor) Tracked(it trie.NodeIterator) trie.NodeIterator { - var trick any = tr.TrackerImpl.Tracked(it) - return trick.(trie.NodeIterator) +// Tracked wraps an iterator in a tracked iterator. This should not be called when the tracker can +// potentially be closed. +func (tr *Tracker) Tracked(it trie.NodeIterator) trie.NodeIterator { + return tr.TrackerImpl.Tracked(it) } func NewImpl(file string, bufsize uint) *TrackerImpl { @@ -75,22 +104,12 @@ type Iterator struct { tracker *TrackerImpl } -// Tracked wraps an iterator in a Iterator. This should not be called once halts are possible. func (tr *TrackerImpl) Tracked(it trie.NodeIterator) *Iterator { ret := &Iterator{it, tr} tr.startChan <- ret return ret } -// StopIterator explicitly stops an iterator -func (tr *TrackerImpl) StopIterator(it *Iterator) { - tr.RLock() - defer tr.RUnlock() - if tr.running { - tr.stopChan <- it - } -} - // Save dumps iterator path and bounds to a text file so it can be restored later. func (tr *TrackerImpl) Save() error { log.Debug("Saving recovery state", "to", tr.recoveryFile) @@ -127,9 +146,6 @@ func (tr *TrackerImpl) removeRecoveryFile() error { return err } -// Restore attempts to read iterator state from the recovery file. -// If the file doesn't exist, returns an empty slice with no error. -// Restored iterators are constructed in the same order they appear in the returned slice. func (tr *TrackerImpl) Restore(makeIterator iter.IteratorConstructor) ([]*Iterator, error) { file, err := os.Open(tr.recoveryFile) if err != nil { @@ -179,7 +195,8 @@ func (tr *TrackerImpl) Restore(makeIterator iter.IteratorConstructor) ([]*Iterat } // CloseAndSave stops all tracked iterators and dumps their state to a file. -// This closes the tracker, so adding a new iterator will fail. +// This closes the tracker, so adding a new iterator afterwards will fail. +// A new Tracker must be constructed in order to restore state. func (tr *TrackerImpl) CloseAndSave() error { tr.Lock() tr.running = false @@ -211,7 +228,7 @@ func (it *Iterator) Next(descend bool) bool { if it.tracker.running { it.tracker.stopChan <- it } else { - log.Error("Iterator stopped after tracker halted", "path", it.Path()) + log.Error("Tracker was closed before iterator finished") } } return ret