2019-01-14 10:09:46 +01:00
|
|
|
package invoices
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/btcsuite/btcd/chaincfg"
|
|
|
|
"github.com/lightningnetwork/lnd/channeldb"
|
|
|
|
"github.com/lightningnetwork/lnd/lntypes"
|
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
2019-02-20 11:44:47 +01:00
|
|
|
"github.com/lightningnetwork/lnd/zpay32"
|
2019-01-14 10:09:46 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
testTimeout = 5 * time.Second
|
|
|
|
|
|
|
|
preimage = lntypes.Preimage{
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,
|
|
|
|
}
|
|
|
|
|
|
|
|
hash = preimage.Hash()
|
|
|
|
|
|
|
|
// testPayReq is a dummy payment request that does parse properly. It
|
|
|
|
// has no relation with the real invoice parameters and isn't asserted
|
|
|
|
// on in this test. LookupInvoice requires this to have a valid value.
|
|
|
|
testPayReq = "lnbc500u1pwywxzwpp5nd2u9xzq02t0tuf2654as7vma42lwkcjptx4yzfq0umq4swpa7cqdqqcqzysmlpc9ewnydr8rr8dnltyxphdyf6mcqrsd6dml8zajtyhwe6a45d807kxtmzayuf0hh2d9tn478ecxkecdg7c5g85pntupug5kakm7xcpn63zqk"
|
|
|
|
)
|
|
|
|
|
2019-02-20 11:44:47 +01:00
|
|
|
func decodeExpiry(payReq string) (uint32, error) {
|
|
|
|
invoice, err := zpay32.Decode(payReq, &chaincfg.MainNetParams)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return uint32(invoice.MinFinalCLTVExpiry()), nil
|
|
|
|
}
|
|
|
|
|
2019-02-20 12:11:15 +01:00
|
|
|
var (
|
|
|
|
testInvoice = &channeldb.Invoice{
|
|
|
|
Terms: channeldb.ContractTerm{
|
|
|
|
PaymentPreimage: preimage,
|
|
|
|
Value: lnwire.MilliSatoshi(100000),
|
|
|
|
},
|
|
|
|
PaymentRequest: []byte(testPayReq),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
func newTestContext(t *testing.T) (*InvoiceRegistry, func()) {
|
2019-01-14 10:09:46 +01:00
|
|
|
cdb, cleanup, err := newDB()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Instantiate and start the invoice registry.
|
2019-02-20 11:44:47 +01:00
|
|
|
registry := NewRegistry(cdb, decodeExpiry)
|
2019-01-14 10:09:46 +01:00
|
|
|
|
|
|
|
err = registry.Start()
|
|
|
|
if err != nil {
|
2019-02-20 12:11:15 +01:00
|
|
|
cleanup()
|
2019-01-14 10:09:46 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-02-20 12:11:15 +01:00
|
|
|
|
|
|
|
return registry, func() {
|
|
|
|
registry.Stop()
|
|
|
|
cleanup()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestSettleInvoice tests settling of an invoice and related notifications.
|
|
|
|
func TestSettleInvoice(t *testing.T) {
|
|
|
|
registry, cleanup := newTestContext(t)
|
|
|
|
defer cleanup()
|
2019-01-14 10:09:46 +01:00
|
|
|
|
|
|
|
allSubscriptions := registry.SubscribeNotifications(0, 0)
|
|
|
|
defer allSubscriptions.Cancel()
|
|
|
|
|
|
|
|
// Subscribe to the not yet existing invoice.
|
|
|
|
subscription := registry.SubscribeSingleInvoice(hash)
|
|
|
|
defer subscription.Cancel()
|
|
|
|
|
|
|
|
if subscription.hash != hash {
|
|
|
|
t.Fatalf("expected subscription for provided hash")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the invoice.
|
2019-02-20 12:11:15 +01:00
|
|
|
addIdx, err := registry.AddInvoice(testInvoice, hash)
|
2019-01-14 10:09:46 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if addIdx != 1 {
|
|
|
|
t.Fatalf("expected addIndex to start with 1, but got %v",
|
|
|
|
addIdx)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect the open state to be sent to the single invoice subscriber.
|
|
|
|
select {
|
|
|
|
case update := <-subscription.Updates:
|
|
|
|
if update.Terms.State != channeldb.ContractOpen {
|
|
|
|
t.Fatalf("expected state ContractOpen, but got %v",
|
|
|
|
update.Terms.State)
|
|
|
|
}
|
|
|
|
case <-time.After(testTimeout):
|
|
|
|
t.Fatal("no update received")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect a new invoice notification to be sent out.
|
|
|
|
select {
|
|
|
|
case newInvoice := <-allSubscriptions.NewInvoices:
|
|
|
|
if newInvoice.Terms.State != channeldb.ContractOpen {
|
|
|
|
t.Fatalf("expected state ContractOpen, but got %v",
|
|
|
|
newInvoice.Terms.State)
|
|
|
|
}
|
|
|
|
case <-time.After(testTimeout):
|
|
|
|
t.Fatal("no update received")
|
|
|
|
}
|
|
|
|
|
2019-02-11 12:01:05 +01:00
|
|
|
hodlChan := make(chan interface{}, 1)
|
|
|
|
|
2019-01-14 10:09:46 +01:00
|
|
|
// Settle invoice with a slightly higher amount.
|
|
|
|
amtPaid := lnwire.MilliSatoshi(100500)
|
2019-02-11 12:01:05 +01:00
|
|
|
_, err = registry.NotifyExitHopHtlc(hash, amtPaid, hodlChan)
|
2019-01-14 10:09:46 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect the settled state to be sent to the single invoice
|
|
|
|
// subscriber.
|
|
|
|
select {
|
|
|
|
case update := <-subscription.Updates:
|
|
|
|
if update.Terms.State != channeldb.ContractSettled {
|
|
|
|
t.Fatalf("expected state ContractOpen, but got %v",
|
|
|
|
update.Terms.State)
|
|
|
|
}
|
|
|
|
if update.AmtPaid != amtPaid {
|
|
|
|
t.Fatal("invoice AmtPaid incorrect")
|
|
|
|
}
|
|
|
|
case <-time.After(testTimeout):
|
|
|
|
t.Fatal("no update received")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect a settled notification to be sent out.
|
|
|
|
select {
|
|
|
|
case settledInvoice := <-allSubscriptions.SettledInvoices:
|
|
|
|
if settledInvoice.Terms.State != channeldb.ContractSettled {
|
|
|
|
t.Fatalf("expected state ContractOpen, but got %v",
|
|
|
|
settledInvoice.Terms.State)
|
|
|
|
}
|
|
|
|
case <-time.After(testTimeout):
|
|
|
|
t.Fatal("no update received")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to settle again.
|
2019-02-11 12:01:05 +01:00
|
|
|
_, err = registry.NotifyExitHopHtlc(hash, amtPaid, hodlChan)
|
2019-01-14 10:09:46 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal("expected duplicate settle to succeed")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to settle again with a different amount.
|
2019-02-11 12:01:05 +01:00
|
|
|
_, err = registry.NotifyExitHopHtlc(hash, amtPaid+600, hodlChan)
|
2019-01-14 10:09:46 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal("expected duplicate settle to succeed")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that settled amount remains unchanged.
|
|
|
|
inv, _, err := registry.LookupInvoice(hash)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if inv.AmtPaid != amtPaid {
|
|
|
|
t.Fatal("expected amount to be unchanged")
|
|
|
|
}
|
2019-01-11 11:19:16 +01:00
|
|
|
|
|
|
|
// Try to cancel.
|
|
|
|
err = registry.CancelInvoice(hash)
|
|
|
|
if err != channeldb.ErrInvoiceAlreadySettled {
|
|
|
|
t.Fatal("expected cancelation of a settled invoice to fail")
|
|
|
|
}
|
2019-02-11 12:01:05 +01:00
|
|
|
|
|
|
|
// As this is a direct sette, we expect nothing on the hodl chan.
|
|
|
|
select {
|
|
|
|
case <-hodlChan:
|
|
|
|
t.Fatal("unexpected event")
|
|
|
|
default:
|
|
|
|
}
|
2019-01-11 11:19:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// TestCancelInvoice tests cancelation of an invoice and related notifications.
|
|
|
|
func TestCancelInvoice(t *testing.T) {
|
2019-02-20 12:11:15 +01:00
|
|
|
registry, cleanup := newTestContext(t)
|
2019-01-11 11:19:16 +01:00
|
|
|
defer cleanup()
|
|
|
|
|
|
|
|
allSubscriptions := registry.SubscribeNotifications(0, 0)
|
|
|
|
defer allSubscriptions.Cancel()
|
|
|
|
|
|
|
|
// Try to cancel the not yet existing invoice. This should fail.
|
2019-02-20 12:11:15 +01:00
|
|
|
err := registry.CancelInvoice(hash)
|
2019-01-11 11:19:16 +01:00
|
|
|
if err != channeldb.ErrInvoiceNotFound {
|
|
|
|
t.Fatalf("expected ErrInvoiceNotFound, but got %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Subscribe to the not yet existing invoice.
|
|
|
|
subscription := registry.SubscribeSingleInvoice(hash)
|
|
|
|
defer subscription.Cancel()
|
|
|
|
|
|
|
|
if subscription.hash != hash {
|
|
|
|
t.Fatalf("expected subscription for provided hash")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the invoice.
|
|
|
|
amt := lnwire.MilliSatoshi(100000)
|
2019-02-20 12:11:15 +01:00
|
|
|
_, err = registry.AddInvoice(testInvoice, hash)
|
2019-01-11 11:19:16 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect the open state to be sent to the single invoice subscriber.
|
|
|
|
select {
|
|
|
|
case update := <-subscription.Updates:
|
|
|
|
if update.Terms.State != channeldb.ContractOpen {
|
|
|
|
t.Fatalf(
|
|
|
|
"expected state ContractOpen, but got %v",
|
|
|
|
update.Terms.State,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
case <-time.After(testTimeout):
|
|
|
|
t.Fatal("no update received")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect a new invoice notification to be sent out.
|
|
|
|
select {
|
|
|
|
case newInvoice := <-allSubscriptions.NewInvoices:
|
|
|
|
if newInvoice.Terms.State != channeldb.ContractOpen {
|
|
|
|
t.Fatalf(
|
|
|
|
"expected state ContractOpen, but got %v",
|
|
|
|
newInvoice.Terms.State,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
case <-time.After(testTimeout):
|
|
|
|
t.Fatal("no update received")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cancel invoice.
|
|
|
|
err = registry.CancelInvoice(hash)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect the canceled state to be sent to the single invoice
|
|
|
|
// subscriber.
|
|
|
|
select {
|
|
|
|
case update := <-subscription.Updates:
|
|
|
|
if update.Terms.State != channeldb.ContractCanceled {
|
|
|
|
t.Fatalf(
|
|
|
|
"expected state ContractCanceled, but got %v",
|
|
|
|
update.Terms.State,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
case <-time.After(testTimeout):
|
|
|
|
t.Fatal("no update received")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect no cancel notification to be sent to all invoice
|
|
|
|
// subscribers (backwards compatibility).
|
|
|
|
|
|
|
|
// Try to cancel again.
|
|
|
|
err = registry.CancelInvoice(hash)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("expected cancelation of a canceled invoice to succeed")
|
|
|
|
}
|
|
|
|
|
2019-02-20 12:11:15 +01:00
|
|
|
// Notify arrival of a new htlc paying to this invoice. This should
|
|
|
|
// succeed.
|
2019-02-11 12:01:05 +01:00
|
|
|
hodlChan := make(chan interface{})
|
|
|
|
event, err := registry.NotifyExitHopHtlc(hash, amt, hodlChan)
|
2019-02-20 12:11:15 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal("expected settlement of a canceled invoice to succeed")
|
|
|
|
}
|
|
|
|
|
|
|
|
if event.Preimage != nil {
|
|
|
|
t.Fatal("expected cancel hodl event")
|
2019-01-11 11:19:16 +01:00
|
|
|
}
|
2019-01-14 10:09:46 +01:00
|
|
|
}
|
|
|
|
|
2019-02-11 12:01:05 +01:00
|
|
|
// TestHoldInvoice tests settling of a hold invoice and related notifications.
|
|
|
|
func TestHoldInvoice(t *testing.T) {
|
|
|
|
defer timeout(t)()
|
|
|
|
|
|
|
|
cdb, cleanup, err := newDB()
|
|
|
|
defer cleanup()
|
|
|
|
|
|
|
|
// Instantiate and start the invoice registry.
|
|
|
|
registry := NewRegistry(cdb, decodeExpiry)
|
|
|
|
|
|
|
|
err = registry.Start()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer registry.Stop()
|
|
|
|
|
|
|
|
allSubscriptions := registry.SubscribeNotifications(0, 0)
|
|
|
|
defer allSubscriptions.Cancel()
|
|
|
|
|
|
|
|
// Subscribe to the not yet existing invoice.
|
|
|
|
subscription := registry.SubscribeSingleInvoice(hash)
|
|
|
|
defer subscription.Cancel()
|
|
|
|
|
|
|
|
if subscription.hash != hash {
|
|
|
|
t.Fatalf("expected subscription for provided hash")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the invoice.
|
|
|
|
invoice := &channeldb.Invoice{
|
|
|
|
Terms: channeldb.ContractTerm{
|
|
|
|
PaymentPreimage: channeldb.UnknownPreimage,
|
|
|
|
Value: lnwire.MilliSatoshi(100000),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = registry.AddInvoice(invoice, hash)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect the open state to be sent to the single invoice subscriber.
|
|
|
|
update := <-subscription.Updates
|
|
|
|
if update.Terms.State != channeldb.ContractOpen {
|
|
|
|
t.Fatalf("expected state ContractOpen, but got %v",
|
|
|
|
update.Terms.State)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect a new invoice notification to be sent out.
|
|
|
|
newInvoice := <-allSubscriptions.NewInvoices
|
|
|
|
if newInvoice.Terms.State != channeldb.ContractOpen {
|
|
|
|
t.Fatalf("expected state ContractOpen, but got %v",
|
|
|
|
newInvoice.Terms.State)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use slightly higher amount for accept/settle.
|
|
|
|
amtPaid := lnwire.MilliSatoshi(100500)
|
|
|
|
|
|
|
|
hodlChan := make(chan interface{}, 1)
|
|
|
|
|
|
|
|
// NotifyExitHopHtlc without a preimage present in the invoice registry
|
|
|
|
// should be possible.
|
|
|
|
event, err := registry.NotifyExitHopHtlc(hash, amtPaid, hodlChan)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("expected settle to succeed but got %v", err)
|
|
|
|
}
|
|
|
|
if event != nil {
|
|
|
|
t.Fatalf("unexpect direct settle")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test idempotency.
|
|
|
|
event, err = registry.NotifyExitHopHtlc(hash, amtPaid, hodlChan)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("expected settle to succeed but got %v", err)
|
|
|
|
}
|
|
|
|
if event != nil {
|
|
|
|
t.Fatalf("unexpect direct settle")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect the accepted state to be sent to the single invoice
|
|
|
|
// subscriber. For all invoice subscribers, we don't expect an update.
|
|
|
|
// Those only get notified on settle.
|
|
|
|
update = <-subscription.Updates
|
|
|
|
if update.Terms.State != channeldb.ContractAccepted {
|
|
|
|
t.Fatalf("expected state ContractAccepted, but got %v",
|
|
|
|
update.Terms.State)
|
|
|
|
}
|
|
|
|
if update.AmtPaid != amtPaid {
|
|
|
|
t.Fatal("invoice AmtPaid incorrect")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Settling with preimage should succeed.
|
|
|
|
err = registry.SettleHodlInvoice(preimage)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("expected set preimage to succeed")
|
|
|
|
}
|
|
|
|
|
|
|
|
hodlEvent := (<-hodlChan).(HodlEvent)
|
|
|
|
if *hodlEvent.Preimage != preimage {
|
|
|
|
t.Fatal("unexpected preimage in hodl event")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We expect a settled notification to be sent out for both all and
|
|
|
|
// single invoice subscribers.
|
|
|
|
settledInvoice := <-allSubscriptions.SettledInvoices
|
|
|
|
if settledInvoice.Terms.State != channeldb.ContractSettled {
|
|
|
|
t.Fatalf("expected state ContractSettled, but got %v",
|
|
|
|
settledInvoice.Terms.State)
|
|
|
|
}
|
|
|
|
|
|
|
|
update = <-subscription.Updates
|
|
|
|
if update.Terms.State != channeldb.ContractSettled {
|
|
|
|
t.Fatalf("expected state ContractSettled, but got %v",
|
|
|
|
update.Terms.State)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Idempotency.
|
|
|
|
err = registry.SettleHodlInvoice(preimage)
|
|
|
|
if err != channeldb.ErrInvoiceAlreadySettled {
|
|
|
|
t.Fatalf("expected ErrInvoiceAlreadySettled but got %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to cancel.
|
|
|
|
err = registry.CancelInvoice(hash)
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal("expected cancelation of a settled invoice to fail")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-14 10:09:46 +01:00
|
|
|
func newDB() (*channeldb.DB, func(), error) {
|
|
|
|
// First, create a temporary directory to be used for the duration of
|
|
|
|
// this test.
|
|
|
|
tempDirName, err := ioutil.TempDir("", "channeldb")
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next, create channeldb for the first time.
|
|
|
|
cdb, err := channeldb.Open(tempDirName)
|
|
|
|
if err != nil {
|
|
|
|
os.RemoveAll(tempDirName)
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanUp := func() {
|
|
|
|
cdb.Close()
|
|
|
|
os.RemoveAll(tempDirName)
|
|
|
|
}
|
|
|
|
|
|
|
|
return cdb, cleanUp, nil
|
|
|
|
}
|