Skip to content

Commit

Permalink
Start sending payments page
Browse files Browse the repository at this point in the history
  • Loading branch information
Conor Okus committed Sep 14, 2023
1 parent 63a390e commit c1fef0b
Show file tree
Hide file tree
Showing 2 changed files with 173 additions and 0 deletions.
4 changes: 4 additions & 0 deletions docs/.vuepress/config.js
Original file line number Diff line number Diff line change
Expand Up @@ -148,6 +148,10 @@ const tutorialSidebar = [
"/tutorials/building-a-node-with-ldk/opening-a-channel",
"Opening a Channel",
],
[
"/tutorials/building-a-node-with-ldk/sending-payments",
"Sending Payments",
],
],
},
],
Expand Down
169 changes: 169 additions & 0 deletions docs/tutorials/building-a-node-with-ldk/sending-payments.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,169 @@
# Sending Payments

Lightning payments are used to pay invoices, which are typically encoded as a
string in accordance with BOLT 11. After parsing the invoice, you'll need to
find a route from your node to the recipient and then make the payment using
`ChannelManager`.

<CodeSwitcher :languages="{rust:'Rust', java:'Java', kotlin:'Kotlin'}">
<template v-slot:rust>

```rust
// Parse the invoice.
let invoice = Invoice::from_str(encoded_invoice)
.expect("ERROR: failed to parse invoice");

let amt_pico_btc = invoice.amount_pico_btc()
.expect("ERROR: invalid invoice: must contain amount to pay");
let amt_msat = amt_pico_btc / 10;
let payer_pubkey = channel_manager.get_our_node_id();
let network_graph = router.network_graph.read().unwrap();
let payee_pubkey = invoice.recover_payee_pub_key();
let payee_features = invoice.features().cloned();
let first_hops = channel_manager.list_usable_channels();
let last_hops = invoice.route_hints();
let final_cltv = invoice.min_final_cltv_expiry() as u32;

// Find a route and send the payment.
let route = router::get_route(
&payer_pubkey, &network_graph, &payee_pubkey, payee_features,
Some(&first_hops.iter().collect::<Vec<_>>()), &last_hops,
amt_msat, final_cltv, logger.clone(),
).expect("ERROR: failed to find route");

let payment_hash = PaymentHash(invoice.payment_hash().clone().into_inner());
let payment_secret = invoice.payment_secret().cloned();

channel_manager.send_payment(&route, payment_hash, &payment_secret)
.expect("ERROR: failed to send payment");
```

</template>
<template v-slot:java>

```java
String invoice_str = // get an invoice from the payee
Result_InvoiceNoneZ parsed_invoice = Invoice.from_str(invoice_str);

if (parsed_invoice instanceof Result_InvoiceNoneZ.Result_InvoiceNoneZ_OK) {
Invoice invoice = ((Result_InvoiceNoneZ.Result_InvoiceNoneZ_OK) parsed_invoice).res;
long amt_msat = 0;
if (invoice.amount_pico_btc() instanceof Option_u64Z.Some) {
amt_msat = ((Option_u64Z.Some)invoice.amount_pico_btc()).some / 10;
}
if (amt_msat == 0) {
// <Handle a zero-value invoice>
}

Route route;
try (LockedNetworkGraph netgraph = router.read_locked_graph()) {
NetworkGraph graph = netgraph.graph();
Result_RouteLightningErrorZ route_res = UtilMethods.get_route(
channel_manager.get_our_node_id(),
graph, invoice.recover_payee_pub_key(), invoice.features(),
channel_manager.list_usable_channels(), invoice.route_hints(),
amt_msat, invoice.min_final_cltv_expiry(), logger);
assert route_res instanceof Result_RouteLightningErrorZ.Result_RouteLightningErrorZ_OK;
route = ((Result_RouteLightningErrorZ.Result_RouteLightningErrorZ_OK) route_res).res;
}

Result_NonePaymentSendFailureZ payment_res = channel_manager.send_payment(
route, invoice.payment_hash(), invoice.payment_secret());
assert payment_res instanceof Result_NonePaymentSendFailureZ.Result_NonePaymentSendFailureZ_OK;
}
```

</template>
<template v-slot:kotlin>

```java
// Get an invoice from the recipient/payee
val parsedInvoice = Invoice.from_str(recipientInvoice)
if (!parsedInvoice.is_ok) {
// Unable to parse invoice
}

val invoice = (parsedInvoice as Result_InvoiceParseOrSemanticErrorZ.Result_InvoiceParseOrSemanticErrorZ_OK).res

var amountSats: Long = 0
if (invoice.amount_milli_satoshis() is Option_u64Z.Some) {
amountSats = (invoice.amount_milli_satoshis() as Option_u64Z.Some).some * 1000
}

if (amountSats == 0L) {
// Handle a zero-value invoice
}

val res = channelManagerConstructor.payer.pay_invoice(invoice)



if (parsed_invoice instanceof Result_InvoiceNoneZ.Result_InvoiceNoneZ_OK) {
Invoice invoice = ((Result_InvoiceNoneZ.Result_InvoiceNoneZ_OK) parsed_invoice).res;
long amt_msat = 0;
if (invoice.amount_pico_btc() instanceof Option_u64Z.Some) {
amt_msat = ((Option_u64Z.Some)invoice.amount_pico_btc()).some / 10;
}
if (amt_msat == 0) {
// <Handle a zero-value invoice>
}

Route route;
try (LockedNetworkGraph netgraph = router.read_locked_graph()) {
NetworkGraph graph = netgraph.graph();
Result_RouteLightningErrorZ route_res = UtilMethods.get_route(
channel_manager.get_our_node_id(),
graph, invoice.recover_payee_pub_key(), invoice.features(),
channel_manager.list_usable_channels(), invoice.route_hints(),
amt_msat, invoice.min_final_cltv_expiry(), logger);
assert route_res instanceof Result_RouteLightningErrorZ.Result_RouteLightningErrorZ_OK;
route = ((Result_RouteLightningErrorZ.Result_RouteLightningErrorZ_OK) route_res).res;
}

Result_NonePaymentSendFailureZ payment_res = channel_manager.send_payment(
route, invoice.payment_hash(), invoice.payment_secret());
assert payment_res instanceof Result_NonePaymentSendFailureZ.Result_NonePaymentSendFailureZ_OK;
}
```

</template>
</CodeSwitcher>

An event is generated once a payment has completed. Successful payments result
in a `PaymentSent` event with the preimage of the payment hash. Be sure to look
out for a `PaymentFailed` event, if the payment fails for some reason, and act
accordingly.

<CodeSwitcher :languages="{rust:'Rust', java:'Java'}">
<template v-slot:rust>

```rust
// In the event handler passed to BackgroundProcessor::start
match event {
Event::PaymentSent { payment_preimage } => {
// Handle successful payment
}
Event::PaymentFailed { payment_hash, rejected_by_dest } => {
// Handle failed payment
}
// ...
}
```

</template>
<template v-slot:java>

```java
// In the `handle_event` method of ChannelManagerPersister implementation
else if (e instanceof Event.PaymentSent) {
// Handle successful payment
Event.PaymentSent event = ((Event.PaymentSent) e);
}
else if (e instanceof Event.PaymentFailed) {
// Handle failed payment
Event.PaymentFailed event = ((Event.PaymentFailed) e);
}
```

</template>
</CodeSwitcher>

0 comments on commit c1fef0b

Please sign in to comment.