-
Notifications
You must be signed in to change notification settings - Fork 119
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
tapchannel: add aux invoice manager tests
- Loading branch information
1 parent
ca8f36b
commit ca8d0c8
Showing
1 changed file
with
355 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,355 @@ | ||
package tapchannel | ||
|
||
import ( | ||
"context" | ||
"crypto/sha256" | ||
"fmt" | ||
"math/big" | ||
"testing" | ||
"time" | ||
|
||
"github.com/lightninglabs/lndclient" | ||
"github.com/lightninglabs/taproot-assets/asset" | ||
"github.com/lightninglabs/taproot-assets/fn" | ||
"github.com/lightninglabs/taproot-assets/rfq" | ||
"github.com/lightninglabs/taproot-assets/rfqmath" | ||
"github.com/lightninglabs/taproot-assets/rfqmsg" | ||
"github.com/lightningnetwork/lnd/lnrpc" | ||
"github.com/lightningnetwork/lnd/lnwire" | ||
"github.com/lightningnetwork/lnd/routing/route" | ||
"github.com/stretchr/testify/require" | ||
) | ||
|
||
const ( | ||
// The test channel ID to use across the test cases. | ||
testChanID = 1234 | ||
) | ||
|
||
var ( | ||
// The node ID to be used for the RFQ peer. | ||
testNodeID = route.Vertex{1, 2, 3} | ||
|
||
assetRate = big.NewInt(100_000) | ||
|
||
testAssetRate = rfqmath.FixedPoint[rfqmath.BigInt]{ | ||
Coefficient: rfqmath.NewBigInt(assetRate), | ||
Scale: 0, | ||
} | ||
) | ||
|
||
// mockRfqManager mocks the interface of the rfq manager required by the aux | ||
// invoice manager. It also holds some internal state to return the desired | ||
// quotes. | ||
type mockRfqManager struct { | ||
peerBuyQuotes rfq.BuyAcceptMap | ||
localSellQuotes rfq.SellAcceptMap | ||
} | ||
|
||
func (m *mockRfqManager) PeerAcceptedBuyQuotes() rfq.BuyAcceptMap { | ||
return m.peerBuyQuotes | ||
} | ||
|
||
func (m *mockRfqManager) LocalAcceptedSellQuotes() rfq.SellAcceptMap { | ||
return m.localSellQuotes | ||
} | ||
|
||
// mockHtlcModifier mocks the HtlcModifier interface that is required by the | ||
// AuxInvoiceManager. | ||
type mockHtlcModifier struct { | ||
requestQue []lndclient.InvoiceHtlcModifyRequest | ||
expectedResQue []lndclient.InvoiceHtlcModifyResponse | ||
done chan bool | ||
t *testing.T | ||
} | ||
|
||
// HtlcModifier handles the invoice htlc modification requests que, then checks | ||
// the returned error and response against the expected values. | ||
func (m *mockHtlcModifier) HtlcModifier(ctx context.Context, | ||
handler lndclient.InvoiceHtlcModifyHandler) error { | ||
|
||
// Process the requests that are provided by the test case. | ||
for i, r := range m.requestQue { | ||
res, err := handler(ctx, r) | ||
|
||
if err != nil { | ||
return err | ||
} | ||
|
||
if m.expectedResQue[i].CancelSet { | ||
if !res.CancelSet { | ||
return fmt.Errorf("expected cancel set flag") | ||
} | ||
|
||
continue | ||
} | ||
|
||
// Check if there's a match with the expected outcome. | ||
if res.AmtPaid != m.expectedResQue[i].AmtPaid { | ||
return fmt.Errorf("invoice paid amount does not match "+ | ||
"expected amount, %v != %v", res.AmtPaid, | ||
m.expectedResQue[i].AmtPaid) | ||
} | ||
} | ||
|
||
// Signal that the htlc modifications are completed. | ||
close(m.done) | ||
|
||
return nil | ||
} | ||
|
||
// TestAuxInvoiceManager tests that the htlc modifications of the aux invoice | ||
// manager align with our expectations. | ||
func TestAuxInvoiceManager(t *testing.T) { | ||
testCases := []struct { | ||
name string | ||
buyQuotes rfq.BuyAcceptMap | ||
sellQuotes rfq.SellAcceptMap | ||
requests []lndclient.InvoiceHtlcModifyRequest | ||
responses []lndclient.InvoiceHtlcModifyResponse | ||
containedErrStr string | ||
}{ | ||
{ | ||
name: "non asset invoice", | ||
requests: []lndclient.InvoiceHtlcModifyRequest{ | ||
{ | ||
Invoice: &lnrpc.Invoice{}, | ||
ExitHtlcAmt: 1234, | ||
}, | ||
}, | ||
responses: []lndclient.InvoiceHtlcModifyResponse{ | ||
{ | ||
AmtPaid: 1234, | ||
}, | ||
}, | ||
}, | ||
{ | ||
name: "non asset routing hints", | ||
requests: []lndclient.InvoiceHtlcModifyRequest{ | ||
{ | ||
Invoice: &lnrpc.Invoice{ | ||
RouteHints: testNonAssetHints(), | ||
ValueMsat: 1_000_000, | ||
}, | ||
ExitHtlcAmt: 1234, | ||
}, | ||
}, | ||
responses: []lndclient.InvoiceHtlcModifyResponse{ | ||
{ | ||
AmtPaid: 1234, | ||
}, | ||
}, | ||
buyQuotes: map[rfq.SerialisedScid]rfqmsg.BuyAccept{ | ||
testChanID: { | ||
Peer: testNodeID, | ||
}, | ||
}, | ||
}, | ||
{ | ||
name: "asset invoice, no custom records", | ||
requests: []lndclient.InvoiceHtlcModifyRequest{ | ||
{ | ||
Invoice: &lnrpc.Invoice{ | ||
RouteHints: testRouteHints(), | ||
PaymentAddr: []byte{1, 1, 1}, | ||
}, | ||
ExitHtlcAmt: 1234, | ||
}, | ||
}, | ||
responses: []lndclient.InvoiceHtlcModifyResponse{ | ||
{ | ||
CancelSet: true, | ||
}, | ||
}, | ||
buyQuotes: map[rfq.SerialisedScid]rfqmsg.BuyAccept{ | ||
testChanID: { | ||
Peer: testNodeID, | ||
}, | ||
}, | ||
}, | ||
{ | ||
name: "asset invoice, custom records", | ||
requests: []lndclient.InvoiceHtlcModifyRequest{ | ||
{ | ||
Invoice: &lnrpc.Invoice{ | ||
RouteHints: testRouteHints(), | ||
ValueMsat: 3_000_000, | ||
PaymentAddr: []byte{1, 1, 1}, | ||
}, | ||
WireCustomRecords: newWireCustomRecords( | ||
t, []*rfqmsg.AssetBalance{ | ||
rfqmsg.NewAssetBalance( | ||
dummyAssetID(1), 3, | ||
), | ||
}, fn.Some(dummyRfqID(31)), | ||
), | ||
}, | ||
}, | ||
responses: []lndclient.InvoiceHtlcModifyResponse{ | ||
{ | ||
AmtPaid: 3_000_000, | ||
}, | ||
}, | ||
buyQuotes: rfq.BuyAcceptMap{ | ||
fn.Ptr(dummyRfqID(31)).Scid(): { | ||
Peer: testNodeID, | ||
AssetRate: testAssetRate, | ||
}, | ||
}, | ||
}, | ||
{ | ||
name: "asset invoice, not enough amt", | ||
requests: []lndclient.InvoiceHtlcModifyRequest{ | ||
{ | ||
Invoice: &lnrpc.Invoice{ | ||
RouteHints: testRouteHints(), | ||
ValueMsat: 10_000_000, | ||
PaymentAddr: []byte{1, 1, 1}, | ||
}, | ||
WireCustomRecords: newWireCustomRecords( | ||
t, []*rfqmsg.AssetBalance{ | ||
rfqmsg.NewAssetBalance( | ||
dummyAssetID(1), | ||
4, | ||
), | ||
}, fn.Some(dummyRfqID(31)), | ||
), | ||
ExitHtlcAmt: 1234, | ||
}, | ||
}, | ||
responses: []lndclient.InvoiceHtlcModifyResponse{ | ||
{ | ||
AmtPaid: 4_000_000, | ||
}, | ||
}, | ||
buyQuotes: rfq.BuyAcceptMap{ | ||
fn.Ptr(dummyRfqID(31)).Scid(): { | ||
Peer: testNodeID, | ||
AssetRate: testAssetRate, | ||
}, | ||
}, | ||
}, | ||
} | ||
|
||
for _, testCase := range testCases { | ||
testCase := testCase | ||
|
||
t.Logf("Running AuxInvoiceManager test case: %v", testCase.name) | ||
|
||
// Instantiate mock rfq manager. | ||
mockRfq := &mockRfqManager{ | ||
peerBuyQuotes: testCase.buyQuotes, | ||
localSellQuotes: testCase.sellQuotes, | ||
} | ||
|
||
done := make(chan bool) | ||
|
||
// Instantiate mock htlc modifier. | ||
mockModifier := &mockHtlcModifier{ | ||
requestQue: testCase.requests, | ||
expectedResQue: testCase.responses, | ||
done: done, | ||
t: t, | ||
} | ||
|
||
// Create the manager. | ||
manager := NewAuxInvoiceManager( | ||
&InvoiceManagerConfig{ | ||
ChainParams: testChainParams, | ||
InvoiceHtlcModifier: mockModifier, | ||
RfqManager: mockRfq, | ||
}, | ||
) | ||
|
||
err := manager.Start() | ||
require.NoError(t, err) | ||
|
||
// If the manager is not done processing the htlc modification | ||
// requests within the specified timeout, assume this is a | ||
// failure. | ||
select { | ||
case <-done: | ||
case <-time.After(testTimeout): | ||
t.Fail() | ||
} | ||
} | ||
} | ||
|
||
func newHash(i []byte) []byte { | ||
h := sha256.New() | ||
_, _ = h.Write(i) | ||
|
||
return h.Sum(nil) | ||
} | ||
|
||
func dummyAssetID(i byte) asset.ID { | ||
return asset.ID(newHash([]byte{i})) | ||
} | ||
|
||
func dummyRfqID(value int) rfqmsg.ID { | ||
return rfqmsg.ID(newHash([]byte{byte(value)})) | ||
} | ||
|
||
func testRouteHints() []*lnrpc.RouteHint { | ||
return []*lnrpc.RouteHint{ | ||
{ | ||
HopHints: []*lnrpc.HopHint{ | ||
{ | ||
ChanId: 1111, | ||
NodeId: route.Vertex{1, 1, 1}.String(), | ||
}, | ||
{ | ||
ChanId: 1111, | ||
NodeId: route.Vertex{1, 1, 1}.String(), | ||
}, | ||
}, | ||
}, | ||
{ | ||
HopHints: []*lnrpc.HopHint{ | ||
{ | ||
ChanId: 1233, | ||
NodeId: route.Vertex{1, 1, 1}.String(), | ||
}, | ||
{ | ||
ChanId: 1234, | ||
NodeId: route.Vertex{1, 2, 3}.String(), | ||
}, | ||
}, | ||
}, | ||
} | ||
} | ||
|
||
func testNonAssetHints() []*lnrpc.RouteHint { | ||
return []*lnrpc.RouteHint{ | ||
{ | ||
HopHints: []*lnrpc.HopHint{ | ||
{ | ||
ChanId: 1234, | ||
NodeId: route.Vertex{1, 1, 1}.String(), | ||
}, | ||
{ | ||
ChanId: 1234, | ||
NodeId: route.Vertex{1, 1, 1}.String(), | ||
}, | ||
}, | ||
}, | ||
{ | ||
HopHints: []*lnrpc.HopHint{ | ||
{ | ||
ChanId: 1234, | ||
NodeId: route.Vertex{2, 2, 2}.String(), | ||
}, | ||
}, | ||
}, | ||
} | ||
} | ||
|
||
func newWireCustomRecords(t *testing.T, amounts []*rfqmsg.AssetBalance, | ||
rfqID fn.Option[rfqmsg.ID]) lnwire.CustomRecords { | ||
|
||
htlc := rfqmsg.NewHtlc(amounts, rfqID) | ||
|
||
customRecords, err := lnwire.ParseCustomRecords(htlc.Bytes()) | ||
require.NoError(t, err) | ||
|
||
return customRecords | ||
} |