2024-12-26 22:46:18 +00:00
|
|
|
package lnurl
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2024-12-26 23:34:56 +00:00
|
|
|
"log"
|
2024-12-26 22:46:18 +00:00
|
|
|
"net/http"
|
2024-12-26 23:34:56 +00:00
|
|
|
"net/url"
|
2024-12-26 22:46:18 +00:00
|
|
|
"strconv"
|
2024-12-26 23:34:56 +00:00
|
|
|
"strings"
|
2024-12-26 22:46:18 +00:00
|
|
|
|
2024-12-26 23:34:56 +00:00
|
|
|
"github.com/btcsuite/btcutil/bech32"
|
|
|
|
"github.com/ekzyis/zaply/env"
|
2024-12-26 22:46:18 +00:00
|
|
|
"github.com/ekzyis/zaply/lightning"
|
|
|
|
"github.com/labstack/echo/v4"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
MIN_SENDABLE_AMOUNT = 1000 // 1 sat
|
|
|
|
MAX_SENDABLE_AMOUNT = 100_000_000_000 // 100m sat
|
|
|
|
MAX_COMMENT_LENGTH = 128
|
|
|
|
)
|
|
|
|
|
|
|
|
func Router(e *echo.Echo, ln lightning.Lightning) {
|
|
|
|
e.GET("/.well-known/lnurlp/:name", payRequest)
|
|
|
|
e.GET("/.well-known/lnurlp/:name/pay", pay(ln))
|
|
|
|
}
|
|
|
|
|
|
|
|
func payRequest(c echo.Context) error {
|
2024-12-27 00:06:33 +00:00
|
|
|
callback, err := url.JoinPath(env.PublicUrl, "/.well-known/lnurlp/", c.Param("name"), "/pay")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-12-26 22:46:18 +00:00
|
|
|
return c.JSON(
|
|
|
|
http.StatusOK,
|
|
|
|
map[string]any{
|
2024-12-27 00:06:33 +00:00
|
|
|
"callback": callback,
|
2024-12-26 22:46:18 +00:00
|
|
|
"minSendable": MIN_SENDABLE_AMOUNT,
|
|
|
|
"maxSendable": MAX_SENDABLE_AMOUNT,
|
2024-12-27 00:15:36 +00:00
|
|
|
"metadata": lnurlMetadata(c),
|
2024-12-26 22:46:18 +00:00
|
|
|
"tag": "payRequest",
|
|
|
|
"commentAllowed": MAX_COMMENT_LENGTH,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func pay(ln lightning.Lightning) echo.HandlerFunc {
|
|
|
|
return func(c echo.Context) error {
|
|
|
|
qAmount := c.QueryParam("amount")
|
|
|
|
if qAmount == "" {
|
|
|
|
return lnurlError(c, http.StatusBadRequest, errors.New("amount required"))
|
|
|
|
}
|
|
|
|
msats, err := strconv.ParseInt(qAmount, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
c.Logger().Error(err)
|
|
|
|
return lnurlError(c, http.StatusBadRequest, errors.New("invalid amount"))
|
|
|
|
}
|
|
|
|
if msats < 1000 {
|
|
|
|
return lnurlError(c, http.StatusBadRequest, errors.New("amount must be at least 1000 msats"))
|
|
|
|
}
|
|
|
|
|
|
|
|
comment := c.QueryParam("comment")
|
|
|
|
|
|
|
|
pr, err := ln.CreateInvoice(msats, comment)
|
|
|
|
if err != nil {
|
|
|
|
c.Logger().Error(err)
|
|
|
|
return lnurlError(c, http.StatusInternalServerError, errors.New("failed to create invoice"))
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.JSON(
|
|
|
|
http.StatusOK,
|
|
|
|
map[string]any{
|
|
|
|
"pr": pr,
|
|
|
|
"routes": []string{},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-27 00:15:36 +00:00
|
|
|
func lnurlMetadata(c echo.Context) string {
|
|
|
|
s := "["
|
|
|
|
s += fmt.Sprintf("[\"text/plain\",\"Paying %s@%s\"]", c.Param("name"), c.Request().Host)
|
|
|
|
s += fmt.Sprintf(",[\"text/identifier\",\"%s@%s\"]", c.Param("name"), c.Request().Host)
|
|
|
|
s += "]"
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2024-12-26 22:46:18 +00:00
|
|
|
func lnurlError(c echo.Context, code int, err error) error {
|
|
|
|
return c.JSON(code, map[string]any{"status": "ERROR", "error": err.Error()})
|
|
|
|
}
|
2024-12-26 23:34:56 +00:00
|
|
|
|
|
|
|
func Encode(base string, parts ...string) string {
|
|
|
|
u, err := url.JoinPath(base, parts...)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
bech32Url, err := bech32.ConvertBits([]byte(u), 8, 5, true)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
lnurl, err := bech32.Encode("lnurl", bech32Url)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.ToUpper(lnurl)
|
|
|
|
}
|