utxonursery: use mocked sweeper in tests
This commit fixes a test flake caused by a race condition. Using the real sweeper in the nursery test created too complex concurrent behaviour to reliably assert on and made the tests difficult to comprehend.
This commit is contained in:
parent
e29b0f8894
commit
347d1545de
@ -2,8 +2,6 @@ package sweep
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
|
||||||
"runtime/pprof"
|
|
||||||
"sync"
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
@ -58,6 +56,8 @@ func NewMockNotifier(t *testing.T) *MockNotifier {
|
|||||||
|
|
||||||
// NotifyEpoch simulates a new epoch arriving.
|
// NotifyEpoch simulates a new epoch arriving.
|
||||||
func (m *MockNotifier) NotifyEpoch(height int32) {
|
func (m *MockNotifier) NotifyEpoch(height int32) {
|
||||||
|
m.t.Helper()
|
||||||
|
|
||||||
for epochChan, chanHeight := range m.epochChan {
|
for epochChan, chanHeight := range m.epochChan {
|
||||||
// Only send notifications if the height is greater than the
|
// Only send notifications if the height is greater than the
|
||||||
// height the caller passed into the register call.
|
// height the caller passed into the register call.
|
||||||
@ -72,8 +72,6 @@ func (m *MockNotifier) NotifyEpoch(height int32) {
|
|||||||
Height: height,
|
Height: height,
|
||||||
}:
|
}:
|
||||||
case <-time.After(defaultTestTimeout):
|
case <-time.After(defaultTestTimeout):
|
||||||
pprof.Lookup("goroutine").WriteTo(os.Stdout, 1)
|
|
||||||
|
|
||||||
m.t.Fatal("epoch event not consumed")
|
m.t.Fatal("epoch event not consumed")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -403,7 +403,7 @@ type nurseryTestContext struct {
|
|||||||
store *nurseryStoreInterceptor
|
store *nurseryStoreInterceptor
|
||||||
restart func() bool
|
restart func() bool
|
||||||
receiveTx func() wire.MsgTx
|
receiveTx func() wire.MsgTx
|
||||||
sweeper *sweep.UtxoSweeper
|
sweeper *mockSweeper
|
||||||
timeoutChan chan chan time.Time
|
timeoutChan chan chan time.Time
|
||||||
t *testing.T
|
t *testing.T
|
||||||
}
|
}
|
||||||
@ -449,33 +449,7 @@ func createNurseryTestContext(t *testing.T,
|
|||||||
bestHeight: 0,
|
bestHeight: 0,
|
||||||
}
|
}
|
||||||
|
|
||||||
sweeperStore := sweep.NewMockSweeperStore()
|
sweeper := newMockSweeper(t)
|
||||||
|
|
||||||
sweeperCfg := &sweep.UtxoSweeperConfig{
|
|
||||||
GenSweepScript: func() ([]byte, error) {
|
|
||||||
return []byte{}, nil
|
|
||||||
},
|
|
||||||
Estimator: &lnwallet.StaticFeeEstimator{},
|
|
||||||
Signer: &nurseryMockSigner{},
|
|
||||||
Notifier: notifier,
|
|
||||||
PublishTransaction: func(tx *wire.MsgTx) error {
|
|
||||||
return publishFunc(tx, "sweeper")
|
|
||||||
},
|
|
||||||
NewBatchTimer: func() <-chan time.Time {
|
|
||||||
c := make(chan time.Time, 1)
|
|
||||||
timeoutChan <- c
|
|
||||||
return c
|
|
||||||
},
|
|
||||||
ChainIO: chainIO,
|
|
||||||
Store: sweeperStore,
|
|
||||||
MaxInputsPerTx: 10,
|
|
||||||
MaxSweepAttempts: 5,
|
|
||||||
NextAttemptDeltaFunc: func(int) int32 { return 1 },
|
|
||||||
}
|
|
||||||
|
|
||||||
sweeper := sweep.New(sweeperCfg)
|
|
||||||
|
|
||||||
sweeper.Start()
|
|
||||||
|
|
||||||
nurseryCfg := NurseryConfig{
|
nurseryCfg := NurseryConfig{
|
||||||
Notifier: notifier,
|
Notifier: notifier,
|
||||||
@ -491,7 +465,7 @@ func createNurseryTestContext(t *testing.T,
|
|||||||
},
|
},
|
||||||
Store: storeIntercepter,
|
Store: storeIntercepter,
|
||||||
ChainIO: chainIO,
|
ChainIO: chainIO,
|
||||||
SweepInput: sweeper.SweepInput,
|
SweepInput: sweeper.sweepInput,
|
||||||
PublishTransaction: func(tx *wire.MsgTx) error {
|
PublishTransaction: func(tx *wire.MsgTx) error {
|
||||||
return publishFunc(tx, "nursery")
|
return publishFunc(tx, "nursery")
|
||||||
},
|
},
|
||||||
@ -531,33 +505,11 @@ func createNurseryTestContext(t *testing.T,
|
|||||||
// Simulate lnd restart.
|
// Simulate lnd restart.
|
||||||
ctx.nursery.Stop()
|
ctx.nursery.Stop()
|
||||||
|
|
||||||
// Also restart sweeper to test behaviour as one unit.
|
|
||||||
//
|
|
||||||
// TODO(joostjager): Mock sweeper to test nursery in
|
|
||||||
// isolation.
|
|
||||||
ctx.sweeper.Stop()
|
|
||||||
|
|
||||||
// Find out if there is a last tx stored. If so, we
|
|
||||||
// expect it to be republished on startup.
|
|
||||||
hasLastTx, err := sweeperCfg.Store.GetLastPublishedTx()
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Restart sweeper.
|
// Restart sweeper.
|
||||||
ctx.sweeper = sweep.New(sweeperCfg)
|
ctx.sweeper = newMockSweeper(t)
|
||||||
ctx.sweeper.Start()
|
|
||||||
|
|
||||||
// Receive last tx if expected.
|
|
||||||
if hasLastTx != nil {
|
|
||||||
utxnLog.Debugf("Expecting republish")
|
|
||||||
ctx.receiveTx()
|
|
||||||
} else {
|
|
||||||
utxnLog.Debugf("Expecting no republish")
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Restart nursery.
|
/// Restart nursery.
|
||||||
nurseryCfg.SweepInput = ctx.sweeper.SweepInput
|
nurseryCfg.SweepInput = ctx.sweeper.sweepInput
|
||||||
ctx.nursery = newUtxoNursery(&nurseryCfg)
|
ctx.nursery = newUtxoNursery(&nurseryCfg)
|
||||||
ctx.nursery.Start()
|
ctx.nursery.Start()
|
||||||
|
|
||||||
@ -571,6 +523,8 @@ func createNurseryTestContext(t *testing.T,
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (ctx *nurseryTestContext) notifyEpoch(height int32) {
|
func (ctx *nurseryTestContext) notifyEpoch(height int32) {
|
||||||
|
ctx.t.Helper()
|
||||||
|
|
||||||
ctx.chainIO.bestHeight = height
|
ctx.chainIO.bestHeight = height
|
||||||
ctx.notifier.NotifyEpoch(height)
|
ctx.notifier.NotifyEpoch(height)
|
||||||
}
|
}
|
||||||
@ -630,8 +584,6 @@ func (ctx *nurseryTestContext) finish() {
|
|||||||
if len(activeHeights) > 0 {
|
if len(activeHeights) > 0 {
|
||||||
ctx.t.Fatalf("Expected height index to be empty")
|
ctx.t.Fatalf("Expected height index to be empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.sweeper.Stop()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func createOutgoingRes(onLocalCommitment bool) *lnwallet.OutgoingHtlcResolution {
|
func createOutgoingRes(onLocalCommitment bool) *lnwallet.OutgoingHtlcResolution {
|
||||||
@ -956,20 +908,17 @@ func testSweep(t *testing.T, ctx *nurseryTestContext,
|
|||||||
afterPublishAssert func()) {
|
afterPublishAssert func()) {
|
||||||
|
|
||||||
// Wait for nursery to publish the sweep tx.
|
// Wait for nursery to publish the sweep tx.
|
||||||
ctx.tick()
|
ctx.sweeper.expectSweep()
|
||||||
sweepTx := ctx.receiveTx()
|
|
||||||
|
|
||||||
if ctx.restart() {
|
if ctx.restart() {
|
||||||
// Nursery reoffers its input. Sweeper needs a tick to create the sweep
|
// Nursery reoffers its input after a restart.
|
||||||
// tx.
|
ctx.sweeper.expectSweep()
|
||||||
ctx.tick()
|
|
||||||
ctx.receiveTx()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
afterPublishAssert()
|
afterPublishAssert()
|
||||||
|
|
||||||
// Confirm the sweep tx.
|
// Confirm the sweep tx.
|
||||||
ctx.notifier.SpendOutpoint(sweepTx.TxIn[0].PreviousOutPoint, sweepTx)
|
ctx.sweeper.sweepAll()
|
||||||
|
|
||||||
// Wait for output to be promoted in store to GRAD.
|
// Wait for output to be promoted in store to GRAD.
|
||||||
select {
|
select {
|
||||||
@ -986,19 +935,6 @@ func testSweep(t *testing.T, ctx *nurseryTestContext,
|
|||||||
assertNurseryReportUnavailable(t, ctx.nursery)
|
assertNurseryReportUnavailable(t, ctx.nursery)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ctx *nurseryTestContext) tick() {
|
|
||||||
select {
|
|
||||||
case c := <-ctx.timeoutChan:
|
|
||||||
select {
|
|
||||||
case c <- time.Time{}:
|
|
||||||
case <-time.After(defaultTestTimeout):
|
|
||||||
ctx.t.Fatal("tick timeout - tick not consumed")
|
|
||||||
}
|
|
||||||
case <-time.After(defaultTestTimeout):
|
|
||||||
ctx.t.Fatal("tick timeout - no new timer created")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type nurseryStoreInterceptor struct {
|
type nurseryStoreInterceptor struct {
|
||||||
ns NurseryStore
|
ns NurseryStore
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user