mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-07-05 04:54:59 +02:00
missioncontrolstore: add additional tests and benchmarks
These will be useful in the next commits.
This commit is contained in:
@ -1,13 +1,14 @@
|
|||||||
package routing
|
package routing
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"reflect"
|
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/davecgh/go-spew/spew"
|
"github.com/btcsuite/btcwallet/walletdb"
|
||||||
"github.com/lightningnetwork/lnd/kvdb"
|
"github.com/lightningnetwork/lnd/kvdb"
|
||||||
|
"github.com/lightningnetwork/lnd/lntest/wait"
|
||||||
"github.com/lightningnetwork/lnd/lnwire"
|
"github.com/lightningnetwork/lnd/lnwire"
|
||||||
"github.com/lightningnetwork/lnd/routing/route"
|
"github.com/lightningnetwork/lnd/routing/route"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
@ -15,18 +16,33 @@ import (
|
|||||||
|
|
||||||
const testMaxRecords = 2
|
const testMaxRecords = 2
|
||||||
|
|
||||||
// TestMissionControlStore tests the recording of payment failure events
|
var (
|
||||||
// in mission control. It tests encoding and decoding of differing lnwire
|
// mcStoreTestRoute is a test route for the mission control store tests.
|
||||||
// failures (FailIncorrectDetails and FailMppTimeout), pruning of results
|
mcStoreTestRoute = route.Route{
|
||||||
// and idempotent writes.
|
SourcePubKey: route.Vertex{1},
|
||||||
func TestMissionControlStore(t *testing.T) {
|
Hops: []*route.Hop{
|
||||||
|
{
|
||||||
|
PubKeyBytes: route.Vertex{2},
|
||||||
|
LegacyPayload: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// mcStoreTestHarness is the harness for a MissonControlStore test.
|
||||||
|
type mcStoreTestHarness struct {
|
||||||
|
db walletdb.DB
|
||||||
|
store *missionControlStore
|
||||||
|
}
|
||||||
|
|
||||||
|
// newMCStoreTestHarness initializes a test mission control store.
|
||||||
|
func newMCStoreTestHarness(t testing.TB, maxRecords int,
|
||||||
|
flushInterval time.Duration) mcStoreTestHarness {
|
||||||
// Set time zone explicitly to keep test deterministic.
|
// Set time zone explicitly to keep test deterministic.
|
||||||
time.Local = time.UTC
|
time.Local = time.UTC
|
||||||
|
|
||||||
file, err := os.CreateTemp("", "*.db")
|
file, err := os.CreateTemp("", "*.db")
|
||||||
if err != nil {
|
require.NoError(t, err)
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
dbPath := file.Name()
|
dbPath := file.Name()
|
||||||
t.Cleanup(func() {
|
t.Cleanup(func() {
|
||||||
@ -37,40 +53,33 @@ func TestMissionControlStore(t *testing.T) {
|
|||||||
db, err := kvdb.Create(
|
db, err := kvdb.Create(
|
||||||
kvdb.BoltBackendName, dbPath, true, kvdb.DefaultDBTimeout,
|
kvdb.BoltBackendName, dbPath, true, kvdb.DefaultDBTimeout,
|
||||||
)
|
)
|
||||||
if err != nil {
|
require.NoError(t, err)
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
t.Cleanup(func() {
|
t.Cleanup(func() {
|
||||||
require.NoError(t, db.Close())
|
require.NoError(t, db.Close())
|
||||||
})
|
})
|
||||||
|
|
||||||
store, err := newMissionControlStore(db, testMaxRecords, time.Second)
|
store, err := newMissionControlStore(db, maxRecords, flushInterval)
|
||||||
if err != nil {
|
require.NoError(t, err)
|
||||||
t.Fatal(err)
|
|
||||||
}
|
return mcStoreTestHarness{db: db, store: store}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestMissionControlStore tests the recording of payment failure events
|
||||||
|
// in mission control. It tests encoding and decoding of differing lnwire
|
||||||
|
// failures (FailIncorrectDetails and FailMppTimeout), pruning of results
|
||||||
|
// and idempotent writes.
|
||||||
|
func TestMissionControlStore(t *testing.T) {
|
||||||
|
h := newMCStoreTestHarness(t, testMaxRecords, time.Second)
|
||||||
|
db, store := h.db, h.store
|
||||||
|
|
||||||
results, err := store.fetchAll()
|
results, err := store.fetchAll()
|
||||||
if err != nil {
|
require.NoError(t, err)
|
||||||
t.Fatal(err)
|
require.Len(t, results, 0)
|
||||||
}
|
|
||||||
if len(results) != 0 {
|
|
||||||
t.Fatal("expected no results")
|
|
||||||
}
|
|
||||||
|
|
||||||
testRoute := route.Route{
|
|
||||||
SourcePubKey: route.Vertex{1},
|
|
||||||
Hops: []*route.Hop{
|
|
||||||
{
|
|
||||||
PubKeyBytes: route.Vertex{2},
|
|
||||||
LegacyPayload: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
failureSourceIdx := 1
|
failureSourceIdx := 1
|
||||||
|
|
||||||
result1 := paymentResult{
|
result1 := paymentResult{
|
||||||
route: &testRoute,
|
route: &mcStoreTestRoute,
|
||||||
failure: lnwire.NewFailIncorrectDetails(100, 1000),
|
failure: lnwire.NewFailIncorrectDetails(100, 1000),
|
||||||
failureSourceIdx: &failureSourceIdx,
|
failureSourceIdx: &failureSourceIdx,
|
||||||
id: 99,
|
id: 99,
|
||||||
@ -94,30 +103,16 @@ func TestMissionControlStore(t *testing.T) {
|
|||||||
require.NoError(t, store.storeResults())
|
require.NoError(t, store.storeResults())
|
||||||
|
|
||||||
results, err = store.fetchAll()
|
results, err = store.fetchAll()
|
||||||
if err != nil {
|
require.NoError(t, err)
|
||||||
t.Fatal(err)
|
require.Len(t, results, 2)
|
||||||
}
|
|
||||||
require.Equal(t, 2, len(results))
|
|
||||||
|
|
||||||
if len(results) != 2 {
|
|
||||||
t.Fatal("expected two results")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check that results are stored in chronological order.
|
// Check that results are stored in chronological order.
|
||||||
if !reflect.DeepEqual(&result1, results[0]) {
|
require.Equal(t, &result1, results[0])
|
||||||
t.Fatalf("the results differ: %v vs %v", spew.Sdump(&result1),
|
require.Equal(t, &result2, results[1])
|
||||||
spew.Sdump(results[0]))
|
|
||||||
}
|
|
||||||
if !reflect.DeepEqual(&result2, results[1]) {
|
|
||||||
t.Fatalf("the results differ: %v vs %v", spew.Sdump(&result2),
|
|
||||||
spew.Sdump(results[1]))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Recreate store to test pruning.
|
// Recreate store to test pruning.
|
||||||
store, err = newMissionControlStore(db, testMaxRecords, time.Second)
|
store, err = newMissionControlStore(db, testMaxRecords, time.Second)
|
||||||
if err != nil {
|
require.NoError(t, err)
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add a newer result which failed due to mpp timeout.
|
// Add a newer result which failed due to mpp timeout.
|
||||||
result3 := result1
|
result3 := result1
|
||||||
@ -131,20 +126,170 @@ func TestMissionControlStore(t *testing.T) {
|
|||||||
|
|
||||||
// Check that results are pruned.
|
// Check that results are pruned.
|
||||||
results, err = store.fetchAll()
|
results, err = store.fetchAll()
|
||||||
if err != nil {
|
require.NoError(t, err)
|
||||||
t.Fatal(err)
|
require.Len(t, results, 2)
|
||||||
}
|
|
||||||
require.Equal(t, 2, len(results))
|
require.Equal(t, &result2, results[0])
|
||||||
if len(results) != 2 {
|
require.Equal(t, &result3, results[1])
|
||||||
t.Fatal("expected two results")
|
}
|
||||||
|
|
||||||
|
// TestMissionControlStoreFlushing asserts the periodic flushing of the store
|
||||||
|
// works correctly.
|
||||||
|
func TestMissionControlStoreFlushing(t *testing.T) {
|
||||||
|
const flushInterval = 500 * time.Millisecond
|
||||||
|
|
||||||
|
h := newMCStoreTestHarness(t, testMaxRecords, flushInterval)
|
||||||
|
db, store := h.db, h.store
|
||||||
|
|
||||||
|
var (
|
||||||
|
failureSourceIdx = 1
|
||||||
|
failureDetails = lnwire.NewFailIncorrectDetails(100, 1000)
|
||||||
|
lastID uint64
|
||||||
|
)
|
||||||
|
nextResult := func() *paymentResult {
|
||||||
|
lastID += 1
|
||||||
|
return &paymentResult{
|
||||||
|
route: &mcStoreTestRoute,
|
||||||
|
failure: failureDetails,
|
||||||
|
failureSourceIdx: &failureSourceIdx,
|
||||||
|
id: lastID,
|
||||||
|
timeReply: testTime,
|
||||||
|
timeFwd: testTime.Add(-time.Minute),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !reflect.DeepEqual(&result2, results[0]) {
|
// Helper to assert the number of results is correct.
|
||||||
t.Fatalf("the results differ: %v vs %v", spew.Sdump(&result2),
|
assertResults := func(wantCount int) {
|
||||||
spew.Sdump(results[0]))
|
t.Helper()
|
||||||
|
err := wait.NoError(func() error {
|
||||||
|
results, err := store.fetchAll()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if wantCount != len(results) {
|
||||||
|
return fmt.Errorf("wrong nb of results: got "+
|
||||||
|
"%d, want %d", len(results), wantCount)
|
||||||
|
}
|
||||||
|
if len(results) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
gotLastID := results[len(results)-1].id
|
||||||
|
if len(results) > 0 && gotLastID != lastID {
|
||||||
|
return fmt.Errorf("wrong id for last item: "+
|
||||||
|
"got %d, want %d", gotLastID, lastID)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}, flushInterval*5)
|
||||||
|
require.NoError(t, err)
|
||||||
}
|
}
|
||||||
if !reflect.DeepEqual(&result3, results[1]) {
|
|
||||||
t.Fatalf("the results differ: %v vs %v", spew.Sdump(&result3),
|
// Run the store.
|
||||||
spew.Sdump(results[1]))
|
store.run()
|
||||||
|
time.Sleep(flushInterval)
|
||||||
|
|
||||||
|
// Wait for the flush interval. There should be no records.
|
||||||
|
assertResults(0)
|
||||||
|
|
||||||
|
// Store a result and check immediately. There still shouldn't be
|
||||||
|
// any results stored (flush interval has not elapsed).
|
||||||
|
store.AddResult(nextResult())
|
||||||
|
assertResults(0)
|
||||||
|
|
||||||
|
// Assert that eventually the result is stored after being flushed.
|
||||||
|
assertResults(1)
|
||||||
|
|
||||||
|
// Store enough results that fill the max number of results.
|
||||||
|
for i := 0; i < testMaxRecords; i++ {
|
||||||
|
store.AddResult(nextResult())
|
||||||
|
}
|
||||||
|
assertResults(testMaxRecords)
|
||||||
|
|
||||||
|
// Finally, stop the store to recreate it.
|
||||||
|
store.stop()
|
||||||
|
|
||||||
|
// Recreate store.
|
||||||
|
store, err := newMissionControlStore(db, testMaxRecords, flushInterval)
|
||||||
|
require.NoError(t, err)
|
||||||
|
store.run()
|
||||||
|
defer store.stop()
|
||||||
|
time.Sleep(flushInterval)
|
||||||
|
assertResults(testMaxRecords)
|
||||||
|
|
||||||
|
// Fill the store with results again.
|
||||||
|
for i := 0; i < testMaxRecords; i++ {
|
||||||
|
store.AddResult(nextResult())
|
||||||
|
}
|
||||||
|
assertResults(testMaxRecords)
|
||||||
|
}
|
||||||
|
|
||||||
|
// BenchmarkMissionControlStoreFlushing benchmarks the periodic storage of data
|
||||||
|
// from the mission control store when additional results are added between
|
||||||
|
// runs.
|
||||||
|
func BenchmarkMissionControlStoreFlushing(b *testing.B) {
|
||||||
|
var (
|
||||||
|
failureSourceIdx = 1
|
||||||
|
failureDetails = lnwire.NewFailIncorrectDetails(100, 1000)
|
||||||
|
testTimeFwd = testTime.Add(-time.Minute)
|
||||||
|
|
||||||
|
tests = []int{0, 1, 10, 100, 250, 500}
|
||||||
|
)
|
||||||
|
|
||||||
|
const testMaxRecords = 1000
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
tc := tc
|
||||||
|
name := fmt.Sprintf("%v additional results", tc)
|
||||||
|
b.Run(name, func(b *testing.B) {
|
||||||
|
h := newMCStoreTestHarness(
|
||||||
|
b, testMaxRecords, time.Second,
|
||||||
|
)
|
||||||
|
store := h.store
|
||||||
|
|
||||||
|
// Fill the store.
|
||||||
|
var lastID uint64
|
||||||
|
for i := 0; i < testMaxRecords; i++ {
|
||||||
|
lastID++
|
||||||
|
result := &paymentResult{
|
||||||
|
route: &mcStoreTestRoute,
|
||||||
|
failure: failureDetails,
|
||||||
|
failureSourceIdx: &failureSourceIdx,
|
||||||
|
id: lastID,
|
||||||
|
timeReply: testTime,
|
||||||
|
timeFwd: testTimeFwd,
|
||||||
|
}
|
||||||
|
store.AddResult(result)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do the first flush.
|
||||||
|
err := store.storeResults()
|
||||||
|
require.NoError(b, err)
|
||||||
|
|
||||||
|
// Create the additional results.
|
||||||
|
results := make([]*paymentResult, tc)
|
||||||
|
for i := 0; i < len(results); i++ {
|
||||||
|
results[i] = &paymentResult{
|
||||||
|
route: &mcStoreTestRoute,
|
||||||
|
failure: failureDetails,
|
||||||
|
failureSourceIdx: &failureSourceIdx,
|
||||||
|
timeReply: testTime,
|
||||||
|
timeFwd: testTimeFwd,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run the actual benchmark.
|
||||||
|
b.ResetTimer()
|
||||||
|
b.ReportAllocs()
|
||||||
|
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
for j := 0; j < len(results); j++ {
|
||||||
|
lastID++
|
||||||
|
results[j].id = lastID
|
||||||
|
store.AddResult(results[j])
|
||||||
|
}
|
||||||
|
err := store.storeResults()
|
||||||
|
require.NoError(b, err)
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user