2023-09-10 15:39:34 +00:00
|
|
|
package lnd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-11-26 22:22:33 +00:00
|
|
|
"database/sql"
|
2023-09-10 15:39:34 +00:00
|
|
|
"log"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"git.ekzyis.com/ekzyis/delphi.market/db"
|
|
|
|
"github.com/lightninglabs/lndclient"
|
|
|
|
"github.com/lightningnetwork/lnd/lnrpc/invoicesrpc"
|
|
|
|
"github.com/lightningnetwork/lnd/lntypes"
|
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
|
|
|
)
|
|
|
|
|
2023-11-26 22:22:33 +00:00
|
|
|
func (lnd *LNDClient) CreateInvoice(tx *sql.Tx, ctx context.Context, d *db.DB, pubkey string, msats int64, description string) (*db.Invoice, error) {
|
2023-09-10 15:39:34 +00:00
|
|
|
var (
|
|
|
|
expiry time.Duration = time.Hour
|
|
|
|
preimage lntypes.Preimage
|
|
|
|
hash lntypes.Hash
|
|
|
|
paymentRequest string
|
|
|
|
lnInvoice *lndclient.Invoice
|
|
|
|
dbInvoice *db.Invoice
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
if preimage, err = generateNewPreimage(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
hash = preimage.Hash()
|
2023-11-26 22:22:33 +00:00
|
|
|
if paymentRequest, err = lnd.Invoices.AddHoldInvoice(ctx, &invoicesrpc.AddInvoiceData{
|
2023-09-10 15:39:34 +00:00
|
|
|
Hash: &hash,
|
|
|
|
Value: lnwire.MilliSatoshi(msats),
|
|
|
|
Expiry: int64(expiry / time.Millisecond),
|
|
|
|
}); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-11-26 22:22:33 +00:00
|
|
|
if lnInvoice, err = lnd.Client.LookupInvoice(ctx, hash); err != nil {
|
2023-09-10 15:39:34 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
dbInvoice = &db.Invoice{
|
|
|
|
Pubkey: pubkey,
|
|
|
|
Msats: msats,
|
|
|
|
Preimage: preimage.String(),
|
|
|
|
PaymentRequest: paymentRequest,
|
|
|
|
Hash: hash.String(),
|
|
|
|
CreatedAt: lnInvoice.CreationDate,
|
|
|
|
ExpiresAt: lnInvoice.CreationDate.Add(expiry),
|
2023-11-09 02:47:24 +00:00
|
|
|
Description: description,
|
2023-09-10 15:39:34 +00:00
|
|
|
}
|
2023-11-26 22:22:33 +00:00
|
|
|
if err := d.CreateInvoice(tx, ctx, dbInvoice); err != nil {
|
2023-09-10 15:39:34 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return dbInvoice, nil
|
|
|
|
}
|
|
|
|
|
2023-10-04 11:36:54 +00:00
|
|
|
func (lnd *LNDClient) CheckInvoice(d *db.DB, hash lntypes.Hash) {
|
2023-09-10 15:39:34 +00:00
|
|
|
var (
|
|
|
|
pollInterval = 5 * time.Second
|
|
|
|
invoice db.Invoice
|
|
|
|
lnInvoice *lndclient.Invoice
|
|
|
|
preimage lntypes.Preimage
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
2023-10-04 11:36:54 +00:00
|
|
|
if err = d.FetchInvoice(&db.FetchInvoiceWhere{Hash: hash.String()}, &invoice); err != nil {
|
2023-09-10 15:39:34 +00:00
|
|
|
log.Println(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
2023-11-27 21:09:51 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.TODO(), 5*time.Second)
|
|
|
|
var tx *sql.Tx
|
|
|
|
if tx, err = d.BeginTx(ctx, nil); err != nil {
|
|
|
|
cancel()
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
handleLoopError := func(err error) {
|
|
|
|
log.Println(err)
|
|
|
|
tx.Rollback()
|
|
|
|
cancel()
|
|
|
|
time.Sleep(pollInterval)
|
|
|
|
}
|
|
|
|
|
2023-09-10 15:39:34 +00:00
|
|
|
log.Printf("lookup invoice: hash=%s", hash)
|
2023-11-27 21:09:51 +00:00
|
|
|
if lnInvoice, err = lnd.Client.LookupInvoice(ctx, hash); err != nil {
|
2023-09-10 15:39:34 +00:00
|
|
|
handleLoopError(err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if time.Now().After(invoice.ExpiresAt) {
|
|
|
|
// cancel invoices after expiration if no matching order found yet
|
2023-11-27 21:09:51 +00:00
|
|
|
if err = lnd.Invoices.CancelInvoice(ctx, hash); err != nil {
|
2023-09-10 15:39:34 +00:00
|
|
|
handleLoopError(err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
log.Printf("invoice expired: hash=%s", hash)
|
2023-11-27 21:09:51 +00:00
|
|
|
tx.Commit()
|
2023-09-10 15:39:34 +00:00
|
|
|
break
|
|
|
|
}
|
2023-11-27 20:58:19 +00:00
|
|
|
if lnInvoice.AmountPaid == lnInvoice.Amount {
|
2023-09-10 15:39:34 +00:00
|
|
|
if preimage, err = lntypes.MakePreimageFromStr(invoice.Preimage); err != nil {
|
|
|
|
handleLoopError(err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// TODO settle invoice after matching order was found
|
2023-11-27 21:09:51 +00:00
|
|
|
if err = lnd.Invoices.SettleInvoice(ctx, preimage); err != nil {
|
2023-09-10 15:39:34 +00:00
|
|
|
handleLoopError(err)
|
|
|
|
continue
|
|
|
|
}
|
2023-11-27 21:09:51 +00:00
|
|
|
if err = d.ConfirmInvoice(tx, ctx, hash.String(), time.Now(), int(lnInvoice.AmountPaid)); err != nil {
|
2023-09-10 15:39:34 +00:00
|
|
|
handleLoopError(err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
log.Printf("invoice confirmed: hash=%s", hash)
|
2023-11-28 20:39:58 +00:00
|
|
|
|
|
|
|
// Run matchmaking if an order was paid
|
|
|
|
var orderId string
|
2023-12-03 22:52:24 +00:00
|
|
|
var deleted bool
|
2023-11-28 20:39:58 +00:00
|
|
|
if err = d.QueryRowContext(ctx,
|
2023-12-03 22:52:24 +00:00
|
|
|
"SELECT o.id, o.deleted_at IS NOT NULL FROM orders o WHERE invoice_id = (SELECT i.id FROM invoices i WHERE hash = $1)",
|
2023-11-28 20:39:58 +00:00
|
|
|
hash.String(),
|
2023-12-03 22:52:24 +00:00
|
|
|
).Scan(&orderId, &deleted); err != nil && err != sql.ErrNoRows {
|
2023-11-28 20:39:58 +00:00
|
|
|
handleLoopError(err)
|
|
|
|
continue
|
|
|
|
}
|
2023-12-03 22:52:24 +00:00
|
|
|
if deleted {
|
|
|
|
// order was canceled before it was paid. refund sats immediately.
|
|
|
|
// this can happen if the market was settled between creating the order and paying the corresponding invoice.
|
|
|
|
if _, err := tx.ExecContext(ctx, "UPDATE users SET msats = msats + $1", int64(lnInvoice.AmountPaid)); err != nil {
|
|
|
|
tx.Rollback()
|
|
|
|
break
|
|
|
|
}
|
|
|
|
log.Printf("order %s canceled. refunded sats to user.", orderId)
|
|
|
|
break
|
|
|
|
}
|
2023-11-28 20:39:58 +00:00
|
|
|
if orderId != "" {
|
|
|
|
go d.RunMatchmaking(orderId)
|
|
|
|
}
|
2023-11-27 21:09:51 +00:00
|
|
|
tx.Commit()
|
2023-11-28 20:39:58 +00:00
|
|
|
|
2023-09-10 15:39:34 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
time.Sleep(pollInterval)
|
|
|
|
}
|
|
|
|
}
|
2023-11-09 03:10:30 +00:00
|
|
|
|
|
|
|
func (lnd *LNDClient) CheckInvoices(d *db.DB) error {
|
|
|
|
var (
|
|
|
|
invoices []db.Invoice
|
|
|
|
err error
|
|
|
|
hash lntypes.Hash
|
|
|
|
)
|
|
|
|
if err = d.FetchInvoices(&db.FetchInvoicesWhere{Unconfirmed: true}, &invoices); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for _, invoice := range invoices {
|
|
|
|
if hash, err = lntypes.MakeHashFromStr(invoice.Hash); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
go lnd.CheckInvoice(d, hash)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|