#![doc = include_str!("../README.md")]
use std::net::SocketAddr;
use std::pin::Pin;
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::Arc;
use async_trait::async_trait;
use bip39::Mnemonic;
use cdk_common::common::FeeReserve;
use cdk_common::payment::{self, *};
use cdk_common::util::{hex, unix_time};
use cdk_common::{Amount, CurrencyUnit, MeltOptions, MeltQuoteState};
use futures::{Stream, StreamExt};
use ldk_node::bitcoin::hashes::Hash;
use ldk_node::bitcoin::Network;
use ldk_node::lightning::ln::channelmanager::PaymentId;
use ldk_node::lightning::ln::msgs::SocketAddress;
use ldk_node::lightning::routing::router::RouteParametersConfig;
use ldk_node::lightning_invoice::{Bolt11InvoiceDescription, Description};
use ldk_node::lightning_types::payment::PaymentHash;
use ldk_node::logger::{LogLevel, LogWriter};
use ldk_node::payment::{PaymentDirection, PaymentKind, PaymentStatus};
use ldk_node::{Builder, Event, Node};
use tokio_stream::wrappers::BroadcastStream;
use tokio_util::sync::CancellationToken;
use tracing::instrument;
use crate::error::Error;
use crate::log::StdoutLogWriter;
mod error;
mod log;
mod web;
#[derive(Clone)]
pub struct CdkLdkNode {
inner: Arc<Node>,
fee_reserve: FeeReserve,
wait_invoice_cancel_token: CancellationToken,
wait_invoice_is_active: Arc<AtomicBool>,
sender: tokio::sync::broadcast::Sender<WaitPaymentResponse>,
receiver: Arc<tokio::sync::broadcast::Receiver<WaitPaymentResponse>>,
events_cancel_token: CancellationToken,
web_addr: Option<SocketAddr>,
}
impl std::fmt::Debug for CdkLdkNode {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("CdkLdkNode")
.field("fee_reserve", &self.fee_reserve)
.field("web_addr", &self.web_addr)
.finish_non_exhaustive()
}
}
#[derive(Debug, Clone)]
pub struct BitcoinRpcConfig {
pub host: String,
pub port: u16,
pub user: String,
pub password: String,
}
#[derive(Debug, Clone)]
pub enum ChainSource {
Esplora(String),
BitcoinRpc(BitcoinRpcConfig),
}
#[derive(Debug, Clone)]
pub enum GossipSource {
P2P,
RapidGossipSync(String),
}
#[derive(Debug)]
pub struct CdkLdkNodeBuilder {
network: Network,
chain_source: ChainSource,
gossip_source: GossipSource,
log_dir_path: Option<String>,
storage_dir_path: String,
fee_reserve: FeeReserve,
listening_addresses: Vec<SocketAddress>,
seed: Option<Mnemonic>,
announcement_addresses: Option<Vec<SocketAddress>>,
}
impl CdkLdkNodeBuilder {
pub fn new(
network: Network,
chain_source: ChainSource,
gossip_source: GossipSource,
storage_dir_path: String,
fee_reserve: FeeReserve,
listening_addresses: Vec<SocketAddress>,
) -> Self {
Self {
network,
chain_source,
gossip_source,
storage_dir_path,
fee_reserve,
listening_addresses,
seed: None,
announcement_addresses: None,
log_dir_path: None,
}
}
pub fn with_seed(mut self, seed: Mnemonic) -> Self {
self.seed = Some(seed);
self
}
pub fn with_announcement_address(mut self, announcement_addresses: Vec<SocketAddress>) -> Self {
self.announcement_addresses = Some(announcement_addresses);
self
}
pub fn with_log_dir_path(mut self, log_dir_path: String) -> Self {
self.log_dir_path = Some(log_dir_path);
self
}
pub fn build(self) -> Result<CdkLdkNode, Error> {
let mut ldk = Builder::new();
ldk.set_network(self.network);
tracing::info!("Storage dir of node is {}", self.storage_dir_path);
ldk.set_storage_dir_path(self.storage_dir_path);
match self.chain_source {
ChainSource::Esplora(esplora_url) => {
ldk.set_chain_source_esplora(esplora_url, None);
}
ChainSource::BitcoinRpc(BitcoinRpcConfig {
host,
port,
user,
password,
}) => {
ldk.set_chain_source_bitcoind_rpc(host, port, user, password);
}
}
match self.gossip_source {
GossipSource::P2P => {
ldk.set_gossip_source_p2p();
}
GossipSource::RapidGossipSync(rgs_url) => {
ldk.set_gossip_source_rgs(rgs_url);
}
}
ldk.set_listening_addresses(self.listening_addresses)?;
if self.log_dir_path.is_some() {
ldk.set_filesystem_logger(self.log_dir_path, Some(LogLevel::Info));
} else {
ldk.set_custom_logger(Arc::new(StdoutLogWriter));
}
ldk.set_node_alias("cdk-ldk-node".to_string())?;
if let Some(seed) = self.seed {
ldk.set_entropy_bip39_mnemonic(seed, None);
}
if let Some(announcement_addresses) = self.announcement_addresses {
ldk.set_announcement_addresses(announcement_addresses)?;
}
let node = ldk.build()?;
tracing::info!("Creating tokio channel for payment notifications");
let (sender, receiver) = tokio::sync::broadcast::channel(8);
let id = node.node_id();
let adr = node.announcement_addresses();
tracing::info!(
"Created node {} with address {:?} on network {}",
id,
adr,
self.network
);
Ok(CdkLdkNode {
inner: node.into(),
fee_reserve: self.fee_reserve,
wait_invoice_cancel_token: CancellationToken::new(),
wait_invoice_is_active: Arc::new(AtomicBool::new(false)),
sender,
receiver: Arc::new(receiver),
events_cancel_token: CancellationToken::new(),
web_addr: None,
})
}
}
impl CdkLdkNode {
pub fn set_web_addr(&mut self, addr: Option<SocketAddr>) {
self.web_addr = addr;
}
pub fn default_web_addr() -> SocketAddr {
SocketAddr::from(([127, 0, 0, 1], 8091))
}
pub fn start_ldk_node(&self) -> Result<(), Error> {
tracing::info!("Starting cdk-ldk node");
self.inner.start()?;
let node_config = self.inner.config();
tracing::info!("Starting node with network {}", node_config.network);
tracing::info!("Node status: {:?}", self.inner.status());
self.handle_events()?;
Ok(())
}
pub fn start_web_server(&self, web_addr: SocketAddr) -> Result<(), Error> {
let web_server = crate::web::WebServer::new(Arc::new(self.clone()));
tokio::spawn(async move {
if let Err(e) = web_server.serve(web_addr).await {
tracing::error!("Web server error: {}", e);
}
});
Ok(())
}
pub fn stop_ldk_node(&self) -> Result<(), Error> {
tracing::info!("Stopping CdkLdkNode");
tracing::info!("Cancelling event handler");
self.events_cancel_token.cancel();
if self.is_wait_invoice_active() {
tracing::info!("Cancelling wait_invoice stream");
self.wait_invoice_cancel_token.cancel();
}
tracing::info!("Stopping LDK node");
self.inner.stop()?;
tracing::info!("CdkLdkNode stopped successfully");
Ok(())
}
async fn handle_payment_received(
node: &Arc<Node>,
sender: &tokio::sync::broadcast::Sender<WaitPaymentResponse>,
payment_id: Option<PaymentId>,
payment_hash: PaymentHash,
amount_msat: u64,
) {
tracing::info!(
"Received payment for hash={} of amount={} msat",
payment_hash,
amount_msat
);
let payment_id = match payment_id {
Some(id) => id,
None => {
tracing::warn!("Received payment without payment_id");
return;
}
};
let payment_id_hex = hex::encode(payment_id.0);
if amount_msat == 0 {
tracing::warn!("Payment of no amount");
return;
}
tracing::info!(
"Processing payment notification: id={}, amount={} msats",
payment_id_hex,
amount_msat
);
let payment_details = match node.payment(&payment_id) {
Some(details) => details,
None => {
tracing::error!("Could not find payment details for id={}", payment_id_hex);
return;
}
};
let (payment_identifier, payment_id) = match payment_details.kind {
PaymentKind::Bolt11 { hash, .. } => {
(PaymentIdentifier::PaymentHash(hash.0), hash.to_string())
}
PaymentKind::Bolt12Offer { hash, offer_id, .. } => match hash {
Some(h) => (
PaymentIdentifier::OfferId(offer_id.to_string()),
h.to_string(),
),
None => {
tracing::error!("Bolt12 payment missing hash");
return;
}
},
k => {
tracing::warn!("Received payment of kind {:?} which is not supported", k);
return;
}
};
let wait_payment_response = WaitPaymentResponse {
payment_identifier,
payment_amount: Amount::new(amount_msat, CurrencyUnit::Msat),
payment_id,
};
match sender.send(wait_payment_response) {
Ok(_) => tracing::info!("Successfully sent payment notification to stream"),
Err(err) => tracing::error!(
"Could not send payment received notification on channel: {}",
err
),
}
}
pub fn handle_events(&self) -> Result<(), Error> {
let node = self.inner.clone();
let sender = self.sender.clone();
let cancel_token = self.events_cancel_token.clone();
tracing::info!("Starting event handler task");
tokio::spawn(async move {
tracing::info!("Event handler loop started");
loop {
tokio::select! {
_ = cancel_token.cancelled() => {
tracing::info!("Event handler cancelled");
break;
}
event = node.next_event_async() => {
match event {
Event::PaymentReceived {
payment_id,
payment_hash,
amount_msat,
custom_records: _
} => {
Self::handle_payment_received(
&node,
&sender,
payment_id,
payment_hash,
amount_msat
).await;
}
event => {
tracing::debug!("Received other ldk node event: {:?}", event);
}
}
if let Err(err) = node.event_handled() {
tracing::error!("Error handling node event: {}", err);
} else {
tracing::debug!("Successfully handled node event");
}
}
}
}
tracing::info!("Event handler loop terminated");
});
tracing::info!("Event handler task spawned");
Ok(())
}
pub fn node(&self) -> Arc<Node> {
Arc::clone(&self.inner)
}
}
#[async_trait]
impl MintPayment for CdkLdkNode {
type Err = payment::Error;
async fn start(&self) -> Result<(), Self::Err> {
self.start_ldk_node().map_err(|e| {
tracing::error!("Failed to start CdkLdkNode: {}", e);
e
})?;
tracing::info!("CdkLdkNode payment processor started successfully");
if let Some(web_addr) = self.web_addr {
tracing::info!("Starting LDK Node web interface on {}", web_addr);
self.start_web_server(web_addr).map_err(|e| {
tracing::error!("Failed to start web server: {}", e);
e
})?;
} else {
tracing::info!("No web server address configured, skipping web interface");
}
Ok(())
}
async fn stop(&self) -> Result<(), Self::Err> {
self.stop_ldk_node().map_err(|e| {
tracing::error!("Failed to stop CdkLdkNode: {}", e);
e.into()
})
}
async fn get_settings(&self) -> Result<SettingsResponse, Self::Err> {
let settings = SettingsResponse {
unit: CurrencyUnit::Msat.to_string(),
bolt11: Some(payment::Bolt11Settings {
mpp: false,
amountless: true,
invoice_description: true,
}),
bolt12: Some(payment::Bolt12Settings { amountless: true }),
custom: std::collections::HashMap::new(),
};
Ok(settings)
}
#[instrument(skip(self))]
async fn create_incoming_payment_request(
&self,
options: IncomingPaymentOptions,
) -> Result<CreateIncomingPaymentResponse, Self::Err> {
match options {
IncomingPaymentOptions::Bolt11(bolt11_options) => {
let amount_msat: Amount = bolt11_options
.amount
.convert_to(&CurrencyUnit::Msat)?
.into();
let description = bolt11_options.description.unwrap_or_default();
let time = bolt11_options
.unix_expiry
.map(|t| t - unix_time())
.unwrap_or(36000);
let description = Bolt11InvoiceDescription::Direct(
Description::new(description).map_err(|_| Error::InvalidDescription)?,
);
let payment = self
.inner
.bolt11_payment()
.receive(amount_msat.into(), &description, time as u32)
.map_err(Error::LdkNode)?;
let payment_hash = payment.payment_hash().to_string();
let payment_identifier = PaymentIdentifier::PaymentHash(
hex::decode(&payment_hash)?
.try_into()
.map_err(|_| Error::InvalidPaymentHashLength)?,
);
Ok(CreateIncomingPaymentResponse {
request_lookup_id: payment_identifier,
request: payment.to_string(),
expiry: Some(unix_time() + time),
extra_json: None,
})
}
IncomingPaymentOptions::Bolt12(bolt12_options) => {
let Bolt12IncomingPaymentOptions {
description,
amount,
unix_expiry,
} = *bolt12_options;
let time = unix_expiry.map(|t| (t - unix_time()) as u32);
let offer = match amount {
Some(amount) => {
let amount_msat: Amount = amount.convert_to(&CurrencyUnit::Msat)?.into();
self.inner
.bolt12_payment()
.receive(
amount_msat.into(),
&description.unwrap_or("".to_string()),
time,
None,
)
.map_err(Error::LdkNode)?
}
None => self
.inner
.bolt12_payment()
.receive_variable_amount(&description.unwrap_or("".to_string()), time)
.map_err(Error::LdkNode)?,
};
let payment_identifier = PaymentIdentifier::OfferId(offer.id().to_string());
Ok(CreateIncomingPaymentResponse {
request_lookup_id: payment_identifier,
request: offer.to_string(),
expiry: time.map(|a| a as u64),
extra_json: None,
})
}
cdk_common::payment::IncomingPaymentOptions::Custom(_) => {
Err(cdk_common::payment::Error::UnsupportedPaymentOption)
}
}
}
#[instrument(skip_all)]
async fn get_payment_quote(
&self,
unit: &CurrencyUnit,
options: OutgoingPaymentOptions,
) -> Result<PaymentQuoteResponse, Self::Err> {
match options {
cdk_common::payment::OutgoingPaymentOptions::Custom(_) => {
Err(cdk_common::payment::Error::UnsupportedPaymentOption)
}
OutgoingPaymentOptions::Bolt11(bolt11_options) => {
let bolt11 = bolt11_options.bolt11;
let amount_msat = match bolt11_options.melt_options {
Some(melt_options) => melt_options.amount_msat(),
None => bolt11
.amount_milli_satoshis()
.ok_or(Error::UnknownInvoiceAmount)?
.into(),
};
let amount =
Amount::new(amount_msat.into(), CurrencyUnit::Msat).convert_to(unit)?;
let relative_fee_reserve =
(self.fee_reserve.percent_fee_reserve * amount.value() as f32) as u64;
let absolute_fee_reserve: u64 = self.fee_reserve.min_fee_reserve.into();
let fee = match relative_fee_reserve > absolute_fee_reserve {
true => relative_fee_reserve,
false => absolute_fee_reserve,
};
let payment_hash = bolt11.payment_hash().to_string();
let payment_hash_bytes = hex::decode(&payment_hash)?
.try_into()
.map_err(|_| Error::InvalidPaymentHashLength)?;
Ok(PaymentQuoteResponse {
request_lookup_id: Some(PaymentIdentifier::PaymentHash(payment_hash_bytes)),
amount,
fee: Amount::new(fee, unit.clone()),
state: MeltQuoteState::Unpaid,
})
}
OutgoingPaymentOptions::Bolt12(bolt12_options) => {
let offer = bolt12_options.offer;
let amount_msat = match bolt12_options.melt_options {
Some(melt_options) => melt_options.amount_msat(),
None => {
let amount = offer.amount().ok_or(payment::Error::AmountMismatch)?;
match amount {
ldk_node::lightning::offers::offer::Amount::Bitcoin {
amount_msats,
} => amount_msats.into(),
_ => return Err(payment::Error::AmountMismatch),
}
}
};
let amount =
Amount::new(amount_msat.into(), CurrencyUnit::Msat).convert_to(unit)?;
let relative_fee_reserve =
(self.fee_reserve.percent_fee_reserve * amount.value() as f32) as u64;
let absolute_fee_reserve: u64 = self.fee_reserve.min_fee_reserve.into();
let fee = match relative_fee_reserve > absolute_fee_reserve {
true => relative_fee_reserve,
false => absolute_fee_reserve,
};
Ok(PaymentQuoteResponse {
request_lookup_id: None,
amount,
fee: Amount::new(fee, unit.clone()),
state: MeltQuoteState::Unpaid,
})
}
}
}
#[instrument(skip(self, options))]
async fn make_payment(
&self,
unit: &CurrencyUnit,
options: OutgoingPaymentOptions,
) -> Result<MakePaymentResponse, Self::Err> {
match options {
cdk_common::payment::OutgoingPaymentOptions::Custom(_) => {
Err(cdk_common::payment::Error::UnsupportedPaymentOption)
}
OutgoingPaymentOptions::Bolt11(bolt11_options) => {
let bolt11 = bolt11_options.bolt11;
let send_params = match bolt11_options
.max_fee_amount
.map(|f| {
f.convert_to(&CurrencyUnit::Msat)
.map(|amount_msat| RouteParametersConfig {
max_total_routing_fee_msat: Some(amount_msat.value()),
..Default::default()
})
})
.transpose()
{
Ok(params) => params,
Err(err) => {
tracing::error!("Failed to convert fee amount: {}", err);
return Err(payment::Error::Custom(format!("Invalid fee amount: {err}")));
}
};
let payment_id = match bolt11_options.melt_options {
Some(MeltOptions::Amountless { amountless }) => self
.inner
.bolt11_payment()
.send_using_amount(&bolt11, amountless.amount_msat.into(), send_params)
.map_err(|err| {
tracing::error!("Could not send send amountless bolt11: {}", err);
Error::CouldNotSendBolt11WithoutAmount
})?,
None => self
.inner
.bolt11_payment()
.send(&bolt11, send_params)
.map_err(|err| {
tracing::error!("Could not send bolt11 {}", err);
Error::CouldNotSendBolt11
})?,
_ => return Err(payment::Error::UnsupportedPaymentOption),
};
let start = std::time::Instant::now();
let timeout = std::time::Duration::from_secs(10);
let (status, payment_details) = loop {
let details = self
.inner
.payment(&payment_id)
.ok_or(Error::PaymentNotFound)?;
match details.status {
PaymentStatus::Succeeded => break (MeltQuoteState::Paid, details),
PaymentStatus::Failed => {
tracing::error!("Failed to pay bolt11 payment.");
break (MeltQuoteState::Failed, details);
}
PaymentStatus::Pending => {
if start.elapsed() > timeout {
tracing::warn!(
"Paying bolt11 exceeded timeout 10 seconds no longer waitning."
);
break (MeltQuoteState::Pending, details);
}
tokio::time::sleep(std::time::Duration::from_millis(100)).await;
continue;
}
}
};
let payment_proof = match payment_details.kind {
PaymentKind::Bolt11 {
hash: _,
preimage,
secret: _,
} => preimage.map(|p| p.to_string()),
_ => return Err(Error::UnexpectedPaymentKind.into()),
};
let total_spent = payment_details
.amount_msat
.ok_or(Error::CouldNotGetAmountSpent)?
+ payment_details.fee_paid_msat.unwrap_or_default();
let total_spent = Amount::new(total_spent, CurrencyUnit::Msat).convert_to(unit)?;
Ok(MakePaymentResponse {
payment_lookup_id: PaymentIdentifier::PaymentHash(
bolt11.payment_hash().to_byte_array(),
),
payment_proof,
status,
total_spent,
})
}
OutgoingPaymentOptions::Bolt12(bolt12_options) => {
let offer = bolt12_options.offer;
let payment_id = match bolt12_options.melt_options {
Some(MeltOptions::Amountless { amountless }) => self
.inner
.bolt12_payment()
.send_using_amount(&offer, amountless.amount_msat.into(), None, None, None)
.map_err(Error::LdkNode)?,
None => self
.inner
.bolt12_payment()
.send(&offer, None, None, None)
.map_err(Error::LdkNode)?,
_ => return Err(payment::Error::UnsupportedPaymentOption),
};
let start = std::time::Instant::now();
let timeout = std::time::Duration::from_secs(10);
let (status, payment_details) = loop {
let details = self
.inner
.payment(&payment_id)
.ok_or(Error::PaymentNotFound)?;
match details.status {
PaymentStatus::Succeeded => break (MeltQuoteState::Paid, details),
PaymentStatus::Failed => {
tracing::error!("Payment with id {} failed.", payment_id);
break (MeltQuoteState::Failed, details);
}
PaymentStatus::Pending => {
if start.elapsed() > timeout {
tracing::warn!(
"Payment has been being for 10 seconds. No longer waiting"
);
break (MeltQuoteState::Pending, details);
}
tokio::time::sleep(std::time::Duration::from_millis(100)).await;
continue;
}
}
};
let payment_proof = match payment_details.kind {
PaymentKind::Bolt12Offer {
hash: _,
preimage,
secret: _,
offer_id: _,
payer_note: _,
quantity: _,
} => preimage.map(|p| p.to_string()),
_ => return Err(Error::UnexpectedPaymentKind.into()),
};
let total_spent = payment_details
.amount_msat
.ok_or(Error::CouldNotGetAmountSpent)?
+ payment_details.fee_paid_msat.unwrap_or_default();
let total_spent = Amount::new(total_spent, CurrencyUnit::Msat).convert_to(unit)?;
Ok(MakePaymentResponse {
payment_lookup_id: PaymentIdentifier::PaymentId(payment_id.0),
payment_proof,
status,
total_spent,
})
}
}
}
#[instrument(skip(self))]
async fn wait_payment_event(
&self,
) -> Result<Pin<Box<dyn Stream<Item = cdk_common::payment::Event> + Send>>, Self::Err> {
tracing::info!("Starting stream for invoices - wait_any_incoming_payment called");
self.wait_invoice_is_active.store(true, Ordering::SeqCst);
tracing::debug!("wait_invoice_is_active set to true");
let receiver = self.receiver.clone();
tracing::info!("Receiver obtained successfully, creating response stream");
let response_stream = BroadcastStream::new(receiver.resubscribe());
let response_stream = response_stream.filter_map(|result| async move {
match result {
Ok(payment) => Some(cdk_common::payment::Event::PaymentReceived(payment)),
Err(err) => {
tracing::warn!("Error in broadcast stream: {}", err);
None
}
}
});
let cancel_token = self.wait_invoice_cancel_token.clone();
let is_active = self.wait_invoice_is_active.clone();
let stream = Box::pin(response_stream);
tokio::spawn(async move {
cancel_token.cancelled().await;
tracing::info!("wait_invoice stream cancelled");
is_active.store(false, Ordering::SeqCst);
});
tracing::info!("wait_any_incoming_payment returning stream");
Ok(stream)
}
fn is_wait_invoice_active(&self) -> bool {
self.wait_invoice_is_active.load(Ordering::SeqCst)
}
fn cancel_wait_invoice(&self) {
self.wait_invoice_cancel_token.cancel()
}
async fn check_incoming_payment_status(
&self,
payment_identifier: &PaymentIdentifier,
) -> Result<Vec<WaitPaymentResponse>, Self::Err> {
let payment_id_str = match payment_identifier {
PaymentIdentifier::PaymentHash(hash) => hex::encode(hash),
PaymentIdentifier::CustomId(id) => id.clone(),
_ => return Err(Error::UnsupportedPaymentIdentifierType.into()),
};
let payment_id = PaymentId(
hex::decode(&payment_id_str)?
.try_into()
.map_err(|_| Error::InvalidPaymentIdLength)?,
);
let payment_details = self
.inner
.payment(&payment_id)
.ok_or(Error::PaymentNotFound)?;
if payment_details.direction == PaymentDirection::Outbound {
return Err(Error::InvalidPaymentDirection.into());
}
let amount = if payment_details.status == PaymentStatus::Succeeded {
payment_details
.amount_msat
.ok_or(Error::CouldNotGetPaymentAmount)?
} else {
return Ok(vec![]);
};
let response = WaitPaymentResponse {
payment_identifier: payment_identifier.clone(),
payment_amount: Amount::new(amount, CurrencyUnit::Msat),
payment_id: payment_id_str,
};
Ok(vec![response])
}
async fn check_outgoing_payment(
&self,
request_lookup_id: &PaymentIdentifier,
) -> Result<MakePaymentResponse, Self::Err> {
let payment_details = match request_lookup_id {
PaymentIdentifier::PaymentHash(id_hash) => self
.inner
.list_payments_with_filter(
|p| matches!(&p.kind, PaymentKind::Bolt11 { hash, .. } if &hash.0 == id_hash),
)
.first()
.cloned(),
PaymentIdentifier::PaymentId(id) => self.inner.payment(&PaymentId(
hex::decode(id)?
.try_into()
.map_err(|_| payment::Error::Custom("Invalid hex".to_string()))?,
)),
_ => {
return Ok(MakePaymentResponse {
payment_lookup_id: request_lookup_id.clone(),
payment_proof: None,
status: MeltQuoteState::Unknown,
total_spent: Amount::new(0, CurrencyUnit::Msat),
});
}
}
.ok_or(Error::PaymentNotFound)?;
if payment_details.direction != PaymentDirection::Outbound {
return Err(Error::InvalidPaymentDirection.into());
}
let status = match payment_details.status {
PaymentStatus::Pending => MeltQuoteState::Pending,
PaymentStatus::Succeeded => MeltQuoteState::Paid,
PaymentStatus::Failed => MeltQuoteState::Failed,
};
let payment_proof = match payment_details.kind {
PaymentKind::Bolt11 {
hash: _,
preimage,
secret: _,
} => preimage.map(|p| p.to_string()),
_ => return Err(Error::UnexpectedPaymentKind.into()),
};
let total_spent = payment_details
.amount_msat
.ok_or(Error::CouldNotGetAmountSpent)?;
Ok(MakePaymentResponse {
payment_lookup_id: request_lookup_id.clone(),
payment_proof,
status,
total_spent: Amount::new(total_spent, CurrencyUnit::Msat),
})
}
}
impl Drop for CdkLdkNode {
fn drop(&mut self) {
tracing::info!("Drop called on CdkLdkNode");
self.wait_invoice_cancel_token.cancel();
tracing::debug!("Cancelled wait_invoice token in drop");
}
}