Skip to content

Commit

Permalink
Merge pull request #178 from getAlby/task-pay-keysend
Browse files Browse the repository at this point in the history
feat: add pay keysend method
  • Loading branch information
rolznz authored Dec 14, 2023
2 parents 0b8c1a6 + d18b5d4 commit f1824b1
Show file tree
Hide file tree
Showing 12 changed files with 423 additions and 30 deletions.
9 changes: 5 additions & 4 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -131,15 +131,15 @@ You can also contribute to our [bounty program](https://github.com/getAlby/light

`pay_invoice`

`pay_keysend`

⚠️ `make_invoice`
- ⚠️ invoice in response missing (TODO)

⚠️ `lookup_invoice`
- ⚠️ invoice in response missing (TODO)
- ⚠️ response does not match spec, missing fields

`pay_keysend`

`list_transactions`

`multi_pay_invoice (TBC)`
Expand All @@ -159,6 +159,9 @@ You can also contribute to our [bounty program](https://github.com/getAlby/light

`pay_invoice`

`pay_keysend`
- ⚠️ preimage in request not supported

⚠️ `make_invoice`
- ⚠️ expiry in request not supported
- ⚠️ invoice in response missing (TODO)
Expand All @@ -167,8 +170,6 @@ You can also contribute to our [bounty program](https://github.com/getAlby/light
- ⚠️ invoice in response missing (TODO)
- ⚠️ response does not match spec, missing fields (TODO)

`pay_keysend`

`list_transactions`

`multi_pay_invoice (TBC)`
Expand Down
88 changes: 88 additions & 0 deletions alby.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import (
"errors"
"fmt"
"net/http"
"strconv"

"github.com/labstack/echo-contrib/session"
"github.com/labstack/echo/v4"
Expand Down Expand Up @@ -435,6 +436,93 @@ func (svc *AlbyOAuthService) SendPaymentSync(ctx context.Context, senderPubkey,
return "", errors.New(errorPayload.Message)
}

func (svc *AlbyOAuthService) SendKeysend(ctx context.Context, senderPubkey string, amount int64, destination, preimage string, custom_records []TLVRecord) (preImage string, err error) {
app := App{}
err = svc.db.Preload("User").First(&app, &App{
NostrPubkey: senderPubkey,
}).Error
if err != nil {
svc.Logger.WithFields(logrus.Fields{
"senderPubkey": senderPubkey,
"payeePubkey": destination,
}).Errorf("App not found: %v", err)
return "", err
}
svc.Logger.WithFields(logrus.Fields{
"senderPubkey": senderPubkey,
"payeePubkey": destination,
"appId": app.ID,
"userId": app.User.ID,
}).Info("Processing keysend request")
tok, err := svc.FetchUserToken(ctx, app)
if err != nil {
return "", err
}
client := svc.oauthConf.Client(ctx, tok)

customRecordsMap := make(map[string]string)
for _, record := range custom_records {
customRecordsMap[strconv.FormatUint(record.Type, 10)] = record.Value
}

body := bytes.NewBuffer([]byte{})
payload := &KeysendRequest{
Amount: amount,
Destination: destination,
CustomRecords: customRecordsMap,
}
err = json.NewEncoder(body).Encode(payload)

// here we don't use the preimage from params
req, err := http.NewRequest("POST", fmt.Sprintf("%s/payments/keysend", svc.cfg.AlbyAPIURL), body)
if err != nil {
svc.Logger.WithError(err).Error("Error creating request /payments/keysend")
return "", err
}

req.Header.Set("User-Agent", "NWC")
req.Header.Set("Content-Type", "application/json")

resp, err := client.Do(req)
if err != nil {
svc.Logger.WithFields(logrus.Fields{
"senderPubkey": senderPubkey,
"payeePubkey": destination,
"appId": app.ID,
"userId": app.User.ID,
}).Errorf("Failed to pay keysend: %v", err)
return "", err
}

if resp.StatusCode < 300 {
responsePayload := &PayResponse{}
err = json.NewDecoder(resp.Body).Decode(responsePayload)
if err != nil {
return "", err
}
svc.Logger.WithFields(logrus.Fields{
"senderPubkey": senderPubkey,
"payeePubkey": destination,
"appId": app.ID,
"userId": app.User.ID,
"preimage": responsePayload.Preimage,
"paymentHash": responsePayload.PaymentHash,
}).Info("Keysend payment successful")
return responsePayload.Preimage, nil
}

errorPayload := &ErrorResponse{}
err = json.NewDecoder(resp.Body).Decode(errorPayload)
svc.Logger.WithFields(logrus.Fields{
"senderPubkey": senderPubkey,
"payeePubkey": destination,
"appId": app.ID,
"userId": app.User.ID,
"APIHttpStatus": resp.StatusCode,
}).Errorf("Payment failed %s", string(errorPayload.Message))
return "", errors.New(errorPayload.Message)
}

func (svc *AlbyOAuthService) AuthHandler(c echo.Context) error {
appName := c.QueryParam("c") // c - for client
// clear current session
Expand Down
2 changes: 1 addition & 1 deletion handle_balance_request.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ func (svc *Service) HandleGetBalanceEvent(ctx context.Context, request *Nip47Req
return nil, err
}

hasPermission, code, message := svc.hasPermission(&app, event, request.Method, nil)
hasPermission, code, message := svc.hasPermission(&app, event, request.Method, 0)

if !hasPermission {
svc.Logger.WithFields(logrus.Fields{
Expand Down
2 changes: 1 addition & 1 deletion handle_info_request.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ func (svc *Service) HandleGetInfoEvent(ctx context.Context, request *Nip47Reques
return nil, err
}

hasPermission, code, message := svc.hasPermission(&app, event, request.Method, nil)
hasPermission, code, message := svc.hasPermission(&app, event, request.Method, 0)

if !hasPermission {
svc.Logger.WithFields(logrus.Fields{
Expand Down
2 changes: 1 addition & 1 deletion handle_lookup_invoice_request.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ func (svc *Service) HandleLookupInvoiceEvent(ctx context.Context, request *Nip47
}

// TODO: move to a shared function
hasPermission, code, message := svc.hasPermission(&app, event, request.Method, nil)
hasPermission, code, message := svc.hasPermission(&app, event, request.Method, 0)

if !hasPermission {
svc.Logger.WithFields(logrus.Fields{
Expand Down
2 changes: 1 addition & 1 deletion handle_make_invoice_request.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ func (svc *Service) HandleMakeInvoiceEvent(ctx context.Context, request *Nip47Re
}

// TODO: move to a shared function
hasPermission, code, message := svc.hasPermission(&app, event, request.Method, nil)
hasPermission, code, message := svc.hasPermission(&app, event, request.Method, 0)

if !hasPermission {
svc.Logger.WithFields(logrus.Fields{
Expand Down
96 changes: 96 additions & 0 deletions handle_pay_keysend_request.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,96 @@
package main

import (
"context"
"encoding/json"
"fmt"

"github.com/nbd-wtf/go-nostr"
"github.com/sirupsen/logrus"
)

func (svc *Service) HandlePayKeysendEvent(ctx context.Context, request *Nip47Request, event *nostr.Event, app App, ss []byte) (result *nostr.Event, err error) {

nostrEvent := NostrEvent{App: app, NostrId: event.ID, Content: event.Content, State: "received"}
err = svc.db.Create(&nostrEvent).Error
if err != nil {
svc.Logger.WithFields(logrus.Fields{
"eventId": event.ID,
"eventKind": event.Kind,
"appId": app.ID,
}).Errorf("Failed to save nostr event: %v", err)
return nil, err
}

payParams := &Nip47KeysendParams{}
err = json.Unmarshal(request.Params, payParams)
if err != nil {
svc.Logger.WithFields(logrus.Fields{
"eventId": event.ID,
"eventKind": event.Kind,
"appId": app.ID,
}).Errorf("Failed to decode nostr event: %v", err)
return nil, err
}

// We use pay_invoice permissions for budget and max amount
hasPermission, code, message := svc.hasPermission(&app, event, NIP_47_PAY_INVOICE_METHOD, payParams.Amount)

if !hasPermission {
svc.Logger.WithFields(logrus.Fields{
"eventId": event.ID,
"eventKind": event.Kind,
"appId": app.ID,
"senderPubkey": payParams.Pubkey,
}).Errorf("App does not have permission: %s %s", code, message)

return svc.createResponse(event, Nip47Response{
ResultType: request.Method,
Error: &Nip47Error{
Code: code,
Message: message,
}}, ss)
}

payment := Payment{App: app, NostrEvent: nostrEvent, Amount: uint(payParams.Amount / 1000)}
insertPaymentResult := svc.db.Create(&payment)
if insertPaymentResult.Error != nil {
return nil, insertPaymentResult.Error
}

svc.Logger.WithFields(logrus.Fields{
"eventId": event.ID,
"eventKind": event.Kind,
"appId": app.ID,
"senderPubkey": payParams.Pubkey,
}).Info("Sending payment")

preimage, err := svc.lnClient.SendKeysend(ctx, event.PubKey, payParams.Amount/1000, payParams.Pubkey, payParams.Preimage, payParams.TLVRecords)
if err != nil {
svc.Logger.WithFields(logrus.Fields{
"eventId": event.ID,
"eventKind": event.Kind,
"appId": app.ID,
"senderPubkey": payParams.Pubkey,
}).Infof("Failed to send payment: %v", err)
nostrEvent.State = NOSTR_EVENT_STATE_HANDLER_ERROR
svc.db.Save(&nostrEvent)
return svc.createResponse(event, Nip47Response{
ResultType: request.Method,
Error: &Nip47Error{
Code: NIP_47_ERROR_INTERNAL,
Message: fmt.Sprintf("Something went wrong while paying invoice: %s", err.Error()),
},
}, ss)
}
payment.Preimage = &preimage
nostrEvent.State = NOSTR_EVENT_STATE_HANDLER_EXECUTED
svc.db.Save(&nostrEvent)
svc.db.Save(&payment)
return svc.createResponse(event, Nip47Response{
ResultType: request.Method,
Result: Nip47PayResponse{
Preimage: preimage,
},
}, ss)
}
2 changes: 1 addition & 1 deletion handle_payment_request.go
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ func (svc *Service) HandlePayInvoiceEvent(ctx context.Context, request *Nip47Req
}, ss)
}

hasPermission, code, message := svc.hasPermission(&app, event, request.Method, &paymentRequest)
hasPermission, code, message := svc.hasPermission(&app, event, request.Method, paymentRequest.MSatoshi)

if !hasPermission {
svc.Logger.WithFields(logrus.Fields{
Expand Down
Loading

0 comments on commit f1824b1

Please sign in to comment.