Phoenixd webhook handler

This commit is contained in:
ekzyis 2024-12-27 01:52:47 +01:00
parent 0a10fcc109
commit aa584d2cb5
3 changed files with 113 additions and 4 deletions

View File

@ -1,7 +1,20 @@
package lightning package lightning
type Bolt11 string import "time"
type PaymentRequest string
type Lightning interface { type Lightning interface {
CreateInvoice(msats int64, description string) (Bolt11, error) CreateInvoice(msats int64, description string) (PaymentRequest, error)
GetInvoice(paymentHash string) (*Invoice, error)
}
type Invoice struct {
PaymentHash string
Preimage string
Msats int64
Description string
PaymentRequest string
CreatedAt time.Time
ConfirmedAt time.Time
} }

View File

@ -9,8 +9,10 @@ import (
"net/url" "net/url"
"strconv" "strconv"
"strings" "strings"
"time"
"github.com/ekzyis/zaply/lightning" "github.com/ekzyis/zaply/lightning"
"github.com/labstack/echo/v4"
) )
type Phoenixd struct { type Phoenixd struct {
@ -53,7 +55,7 @@ func WithPhoenixdWebhookUrl(webhookUrl string) func(*Phoenixd) *Phoenixd {
} }
} }
func (p *Phoenixd) CreateInvoice(msats int64, description string) (lightning.Bolt11, error) { func (p *Phoenixd) CreateInvoice(msats int64, description string) (lightning.PaymentRequest, error) {
values := url.Values{} values := url.Values{}
values.Add("amountSat", strconv.FormatInt(msats/1000, 10)) values.Add("amountSat", strconv.FormatInt(msats/1000, 10))
values.Add("description", description) values.Add("description", description)
@ -94,5 +96,87 @@ func (p *Phoenixd) CreateInvoice(msats int64, description string) (lightning.Bol
return "", err return "", err
} }
return lightning.Bolt11(response.Serialized), nil return lightning.PaymentRequest(response.Serialized), nil
}
func (p *Phoenixd) GetInvoice(paymentHash string) (*lightning.Invoice, error) {
endpoint := p.url.JoinPath("payments/incoming", paymentHash)
req, err := http.NewRequest("GET", endpoint.String(), nil)
if err != nil {
return nil, err
}
if p.limitedAccessToken != "" {
req.SetBasicAuth("", p.limitedAccessToken)
}
resp, err := http.DefaultClient.Do(req)
if err != nil {
return nil, err
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
return nil, err
}
if resp.StatusCode != http.StatusOK {
return nil, fmt.Errorf("phoenixd %s: %s", resp.Status, string(body))
}
var response struct {
PaymentHash string `json:"paymentHash"`
Preimage string `json:"preimage"`
Msats int64 `json:"receivedSat"`
Description string `json:"description"`
CreatedAt int64 `json:"createdAt"`
ConfirmedAt int64 `json:"completedAt"`
}
if err := json.Unmarshal(body, &response); err != nil {
return nil, err
}
createdAt := time.Unix(response.CreatedAt/1000, 0)
var confirmedAt time.Time
if response.ConfirmedAt != 0 {
confirmedAt = time.Unix(response.ConfirmedAt/1000, 0)
}
return &lightning.Invoice{
PaymentHash: response.PaymentHash,
Preimage: response.Preimage,
Msats: response.Msats,
Description: response.Description,
CreatedAt: createdAt,
ConfirmedAt: confirmedAt,
}, nil
}
func (p *Phoenixd) WebhookHandler(c echo.Context) error {
go func() {
var webhook struct {
Type string `json:"type"`
AmountSat int64 `json:"amountSat"`
PaymentHash string `json:"paymentHash"`
}
if err := c.Bind(&webhook); err != nil {
c.Logger().Error(err)
return
}
inv, err := p.GetInvoice(webhook.PaymentHash)
if err != nil {
c.Logger().Error(err)
return
}
log.Printf(
"payment received: %s | %d msats | %s | %s | %s",
inv.PaymentHash, inv.Msats, inv.Description,
inv.CreatedAt.Format(time.RFC3339), inv.ConfirmedAt.Format(time.RFC3339),
)
}()
return c.NoContent(http.StatusOK)
} }

View File

@ -1,6 +1,9 @@
package server package server
import ( import (
"log"
"net/url"
"github.com/ekzyis/zaply/env" "github.com/ekzyis/zaply/env"
"github.com/ekzyis/zaply/lightning/phoenixd" "github.com/ekzyis/zaply/lightning/phoenixd"
"github.com/ekzyis/zaply/lnurl" "github.com/ekzyis/zaply/lnurl"
@ -22,11 +25,20 @@ func NewServer() *Server {
CustomTimeFormat: "2006-01-02 15:04:05.00000-0700", CustomTimeFormat: "2006-01-02 15:04:05.00000-0700",
})) }))
webhookPath := "/overlay/webhook"
webhookUrl, err := url.JoinPath(env.PublicUrl, webhookPath)
if err != nil {
log.Fatal(err)
}
p := phoenixd.NewPhoenixd( p := phoenixd.NewPhoenixd(
phoenixd.WithPhoenixdURL(env.PhoenixdURL), phoenixd.WithPhoenixdURL(env.PhoenixdURL),
phoenixd.WithPhoenixdLimitedAccessToken(env.PhoenixdLimitedAccessToken), phoenixd.WithPhoenixdLimitedAccessToken(env.PhoenixdLimitedAccessToken),
phoenixd.WithPhoenixdWebhookUrl(webhookUrl),
) )
s.POST(webhookPath, p.WebhookHandler)
lnurl.Router(s.Echo, p) lnurl.Router(s.Echo, p)
return s return s