LNURL-Pay

Usage

Rust
// Endpoint can also be of the form:
// lnurlp://domain.com/lnurl-pay?key=val
// lnurl1dp68gurn8ghj7mr0vdskc6r0wd6z7mrww4excttsv9un7um9wdekjmmw84jxywf5x43rvv35xgmr2enrxanr2cfcvsmnwe3jxcukvde48qukgdec89snwde3vfjxvepjxpjnjvtpxd3kvdnxx5crxwpjvyunsephsz36jf
let lnurl_pay_url = "lightning@address.com";

if let Ok(LnUrlPay { data: pd }) = parse(lnurl_pay_url).await {
    let amount_msat = pd.min_sendable;
    let use_trampoline = true;
    let optional_comment = Some("<comment>".to_string());
    let optional_payment_label = Some("<label>".to_string());

    sdk.lnurl_pay(LnUrlPayRequest {
        data: pd,
        amount_msat,
        use_trampoline,
        comment: optional_comment,
        payment_label: optional_payment_label,
        validate_success_action_url: Some(true),
    })
    .await?;
}
Swift
// Endpoint can also be of the form:
// lnurlp://domain.com/lnurl-pay?key=val
// lnurl1dp68gurn8ghj7mr0vdskc6r0wd6z7mrww4excttsv9un7um9wdekjmmw84jxywf5x43rvv35xgmr2enrxanr2cfcvsmnwe3jxcukvde48qukgdec89snwde3vfjxvepjxpjnjvtpxd3kvdnxx5crxwpjvyunsephsz36jf
var response: LnUrlPayResult?
let lnurlPayUrl = "lightning@address.com"
if let inputType = try? parseInput(s: lnurlPayUrl) {
    if case let .lnUrlPay(data) = inputType {
        let amountMsat = data.minSendable
        let useTrampoline = true
        let optionalComment = "<comment>"
        let optionalPaymentLabel = "<label>"
        let req = LnUrlPayRequest(data: data, amountMsat: amountMsat, useTrampoline: useTrampoline, comment: optionalComment, paymentLabel: optionalPaymentLabel)
        response = try? sdk.payLnurl(req: req)
    }
}
Kotlin
// Endpoint can also be of the form:
// lnurlp://domain.com/lnurl-pay?key=val
// lnurl1dp68gurn8ghj7mr0vdskc6r0wd6z7mrww4excttsv9un7um9wdekjmmw84jxywf5x43rvv35xgmr2enrxanr2cfcvsmnwe3jxcukvde48qukgdec89snwde3vfjxvepjxpjnjvtpxd3kvdnxx5crxwpjvyunsephsz36jf
val lnurlPayUrl = "lightning@address.com";
try {
    val inputType = parseInput(lnurlPayUrl)
    if (inputType is InputType.LnUrlPay) {
        val requestData = inputType.data
        val useTrampoline = true;
        val amountMsat = requestData.minSendable
        val optionalComment = "<comment>";
        val optionalPaymentLabel = "<label>";
        
        val req = LnUrlPayRequest(requestData, amountMsat, useTrampoline, optionalComment, optionalPaymentLabel)
        sdk.payLnurl(req)
    }
} catch (e: Exception) {
    // handle error
}
React Native
// Endpoint can also be of the
// lnurlp://domain.com/lnurl-pay?key=val
// lnurl1dp68gurn8ghj7mr0vdskc6r0wd6z7mrww4excttsv9un7um9wdekjmmw84jxywf5x43rvv35xgmr2enrxanr2cfcvsmnwe3jxcukvde48qukgdec89snwde3vfjxvepjxpjnjvtpxd3kvdnxx5crxwpjvyunsephsz36jf
try {
  const lnurlPayUrl = 'lightning@address.com'

  const input = await parseInput(lnurlPayUrl)
  if (input.type === InputTypeVariant.LN_URL_PAY) {
    const amountMsat = input.data.minSendable
    const useTrampoline = true
    const optionalComment = '<comment>'
    const optionalPaymentLabel = '<label>'
    const lnUrlPayResult = await payLnurl({
      data: input.data,
      amountMsat,
      useTrampoline,
      comment: optionalComment,
      paymentLabel: optionalPaymentLabel
    })
  }
} catch (err) {
  console.error(err)
}
Dart
/// Endpoint can also be of the form:
/// lnurlp://domain.com/lnurl-pay?key=val
/// lnurl1dp68gurn8ghj7mr0vdskc6r0wd6z7mrww4excttsv9un7um9wdekjmmw84jxywf5x43rvv35xgmr2enrxanr2cfcvsmnwe3jxcukvde48qukgdec89snwde3vfjxvepjxpjnjvtpxd3kvdnxx5crxwpjvyunsephsz36jf
String lnurlPayUrl = "lightning@address.com";

InputType inputType = await breezSDK.parseInput(input: lnurlPayUrl);
if (inputType is InputType_LnUrlPay) {
  int amountMsat = inputType.data.minSendable;
  bool useTrampoline = true;
  String optionalComment = "<comment>";
  String optionalPaymentLabel = "<label>";
  LnUrlPayRequest req = LnUrlPayRequest(
    data: inputType.data,
    amountMsat: amountMsat,
    useTrampoline: useTrampoline,
    comment: optionalComment,
    paymentLabel: optionalPaymentLabel,
  );
  LnUrlPayResult result = await breezSDK.lnurlPay(req: req);
  print(result.data);
}
Python
# Endpoint can also be of the form:
# lnurlp://domain.com/lnurl-pay?key=val
# lnurl1dp68gurn8ghj7mr0vdskc6r0wd6z7mrww4excttsv9un7um9wdekjmmw84jxywf5x43rvv35xgmr2enrxanr2cfcvsmnwe3jxcukvde48qukgdec89snwde3vfjxvepjxpjnjvtpxd3kvdnxx5crxwpjvyunsephsz36jf
lnurl_pay_url = "lightning@address.com"
try: 
  parsed_input = breez_sdk.parse_input(lnurl_pay_url)
  if isinstance(parsed_input, breez_sdk.InputType.LN_URL_PAY):
     amount_msat = parsed_input.data.min_sendable
     use_trampoline = True
     optional_comment = "<comment>"
     optional_payment_label = "<label>"
     req = breez_sdk.LnUrlPayRequest(parsed_input.data, amount_msat, use_trampoline, optional_comment, optional_payment_label)
     result = sdk_services.pay_lnurl(req)
     return result
except Exception as error:
    print(error)
    raise 
Go
// Endpoint can also be of the form:
// lnurlp://domain.com/lnurl-pay?key=val
// lnurl1dp68gurn8ghj7mr0vdskc6r0wd6z7mrww4excttsv9un7um9wdekjmmw84jxywf5x43rvv35xgmr2enrxanr2cfcvsmnwe3jxcukvde48qukgdec89snwde3vfjxvepjxpjnjvtpxd3kvdnxx5crxwpjvyunsephsz36jf
lnurlPayUrl := "lightning@address.com"

input, err := breez_sdk.ParseInput(lnurlPayUrl)
if err != nil {
    return err
}

switch inputType := input.(type) {
case breez_sdk.InputTypeLnUrlPay:
    useTrampoline := true
    amountMsat := inputType.Data.MinSendable
    optionalComment := "<comment>"
    optionalPaymentLabel := "<label>"
    lnUrlPayRequest := breez_sdk.LnUrlPayRequest{
        Data:          inputType.Data,
        UseTrampoline: useTrampoline,
        AmountMsat:    amountMsat,
        Comment:       &optionalComment,
        PaymentLabel:  &optionalPaymentLabel,
    }
    result, err := sdk.PayLnurl(lnUrlPayRequest)
    if err != nil {
        return err
    }

    switch result.(type) {
    case breez_sdk.LnUrlPayResultEndpointSuccess:
        log.Printf("Successfully paid")
    default:
        log.Printf("Failed to pay")
    }
}
C#
// Endpoint can also be of the form:
// lnurlp://domain.com/lnurl-pay?key=val
// lnurl1dp68gurn8ghj7mr0vdskc6r0wd6z7mrww4excttsv9un7um9wdekjmmw84jxywf5x43rvv35xgmr2enrxanr2cfcvsmnwe3jxcukvde48qukgdec89snwde3vfjxvepjxpjnjvtpxd3kvdnxx5crxwpjvyunsephsz36jf
var lnurlPayUrl = "lightning@address.com";

try
{
    var input = BreezSdkMethods.ParseInput(lnurlPayUrl);
    if (input is InputType.LnUrlPay lnurlp)
    {
        var amountMsat = lnurlp.data.minSendable;
        var useTrampoline = true;
        var optionalComment = "<comment>";
        var optionalPaymentLabel = "<label>";
        
        var req = new LnUrlPayRequest(lnurlp.data, amountMsat, useTrampoline, optionalComment, optionalPaymentLabel);
        var result = sdk.PayLnurl(req);
    }
}
catch (Exception)
{
    // Handle error
}

Trampoline payments

In the above example, the use_trampoline flag is set to true to enable trampoline payments. See Trampoline payments for more details.

Supported Specs

  • LUD-01 LNURL bech32 encoding
  • LUD-06 payRequest spec
  • LUD-16 LN Address
  • LUD-17 Support for lnurlp prefix with non-bech32-encoded LNURL URLs