package routing import ( "os" "reflect" "testing" "time" "github.com/davecgh/go-spew/spew" "github.com/lightningnetwork/lnd/kvdb" "github.com/lightningnetwork/lnd/lnwire" "github.com/lightningnetwork/lnd/routing/route" "github.com/stretchr/testify/require" ) const testMaxRecords = 2 // 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) { // Set time zone explicitly to keep test deterministic. time.Local = time.UTC file, err := os.CreateTemp("", "*.db") if err != nil { t.Fatal(err) } dbPath := file.Name() t.Cleanup(func() { require.NoError(t, file.Close()) require.NoError(t, os.Remove(dbPath)) }) db, err := kvdb.Create( kvdb.BoltBackendName, dbPath, true, kvdb.DefaultDBTimeout, ) if err != nil { t.Fatal(err) } t.Cleanup(func() { require.NoError(t, db.Close()) }) store, err := newMissionControlStore(db, testMaxRecords, time.Second) if err != nil { t.Fatal(err) } results, err := store.fetchAll() if err != nil { t.Fatal(err) } 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 result1 := paymentResult{ route: &testRoute, failure: lnwire.NewFailIncorrectDetails(100, 1000), failureSourceIdx: &failureSourceIdx, id: 99, timeReply: testTime, timeFwd: testTime.Add(-time.Minute), } result2 := result1 result2.timeReply = result1.timeReply.Add(time.Hour) result2.timeFwd = result1.timeReply.Add(time.Hour) result2.id = 2 // Store result. store.AddResult(&result2) // Store again to test idempotency. store.AddResult(&result2) // Store second result which has an earlier timestamp. store.AddResult(&result1) require.NoError(t, store.storeResults()) results, err = store.fetchAll() if err != nil { t.Fatal(err) } require.Equal(t, 2, len(results)) if len(results) != 2 { t.Fatal("expected two results") } // Check that results are stored in chronological order. if !reflect.DeepEqual(&result1, results[0]) { t.Fatalf("the results differ: %v vs %v", spew.Sdump(&result1), 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. store, err = newMissionControlStore(db, testMaxRecords, time.Second) if err != nil { t.Fatal(err) } // Add a newer result which failed due to mpp timeout. result3 := result1 result3.timeReply = result1.timeReply.Add(2 * time.Hour) result3.timeFwd = result1.timeReply.Add(2 * time.Hour) result3.id = 3 result3.failure = &lnwire.FailMPPTimeout{} store.AddResult(&result3) require.NoError(t, store.storeResults()) // Check that results are pruned. results, err = store.fetchAll() if err != nil { t.Fatal(err) } require.Equal(t, 2, len(results)) if len(results) != 2 { t.Fatal("expected two results") } if !reflect.DeepEqual(&result2, results[0]) { t.Fatalf("the results differ: %v vs %v", spew.Sdump(&result2), spew.Sdump(results[0])) } if !reflect.DeepEqual(&result3, results[1]) { t.Fatalf("the results differ: %v vs %v", spew.Sdump(&result3), spew.Sdump(results[1])) } }