123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533 |
- //! CDK lightning backend for CLN
- #![warn(missing_docs)]
- #![warn(rustdoc::bare_urls)]
- use std::path::PathBuf;
- use std::pin::Pin;
- use std::str::FromStr;
- use std::sync::atomic::{AtomicBool, Ordering};
- use std::sync::Arc;
- use std::time::Duration;
- use async_trait::async_trait;
- use cdk::amount::{to_unit, Amount, MSAT_IN_SAT};
- use cdk::cdk_lightning::{
- self, CreateInvoiceResponse, MintLightning, PayInvoiceResponse, PaymentQuoteResponse, Settings,
- };
- use cdk::mint::FeeReserve;
- use cdk::nuts::{CurrencyUnit, MeltQuoteBolt11Request, MeltQuoteState, MintQuoteState};
- use cdk::util::{hex, unix_time};
- use cdk::{mint, Bolt11Invoice};
- use cln_rpc::model::requests::{
- InvoiceRequest, ListinvoicesRequest, ListpaysRequest, PayRequest, WaitanyinvoiceRequest,
- };
- use cln_rpc::model::responses::{
- ListinvoicesInvoices, ListinvoicesInvoicesStatus, ListpaysPaysStatus, PayStatus,
- WaitanyinvoiceResponse, WaitanyinvoiceStatus,
- };
- use cln_rpc::model::Request;
- use cln_rpc::primitives::{Amount as CLN_Amount, AmountOrAny};
- use error::Error;
- use futures::{Stream, StreamExt};
- use tokio::sync::Mutex;
- use tokio_util::sync::CancellationToken;
- use uuid::Uuid;
- pub mod error;
- /// CLN mint backend
- #[derive(Clone)]
- pub struct Cln {
- rpc_socket: PathBuf,
- cln_client: Arc<Mutex<cln_rpc::ClnRpc>>,
- fee_reserve: FeeReserve,
- wait_invoice_cancel_token: CancellationToken,
- wait_invoice_is_active: Arc<AtomicBool>,
- }
- impl Cln {
- /// Create new [`Cln`]
- pub async fn new(rpc_socket: PathBuf, fee_reserve: FeeReserve) -> Result<Self, Error> {
- let cln_client = cln_rpc::ClnRpc::new(&rpc_socket).await?;
- Ok(Self {
- rpc_socket,
- cln_client: Arc::new(Mutex::new(cln_client)),
- fee_reserve,
- wait_invoice_cancel_token: CancellationToken::new(),
- wait_invoice_is_active: Arc::new(AtomicBool::new(false)),
- })
- }
- }
- #[async_trait]
- impl MintLightning for Cln {
- type Err = cdk_lightning::Error;
- fn get_settings(&self) -> Settings {
- Settings {
- mpp: true,
- unit: CurrencyUnit::Msat,
- invoice_description: true,
- }
- }
- /// Is wait invoice active
- fn is_wait_invoice_active(&self) -> bool {
- self.wait_invoice_is_active.load(Ordering::SeqCst)
- }
- /// Cancel wait invoice
- fn cancel_wait_invoice(&self) {
- self.wait_invoice_cancel_token.cancel()
- }
- #[allow(clippy::incompatible_msrv)]
- // Clippy thinks select is not stable but it compiles fine on MSRV (1.63.0)
- async fn wait_any_invoice(
- &self,
- ) -> Result<Pin<Box<dyn Stream<Item = String> + Send>>, Self::Err> {
- let last_pay_index = self.get_last_pay_index().await?;
- let cln_client = cln_rpc::ClnRpc::new(&self.rpc_socket).await?;
- let stream = futures::stream::unfold(
- (
- cln_client,
- last_pay_index,
- self.wait_invoice_cancel_token.clone(),
- Arc::clone(&self.wait_invoice_is_active),
- ),
- |(mut cln_client, mut last_pay_idx, cancel_token, is_active)| async move {
- // Set the stream as active
- is_active.store(true, Ordering::SeqCst);
- loop {
- tokio::select! {
- _ = cancel_token.cancelled() => {
- // Set the stream as inactive
- is_active.store(false, Ordering::SeqCst);
- // End the stream
- return None;
- }
- result = cln_client.call(cln_rpc::Request::WaitAnyInvoice(WaitanyinvoiceRequest {
- timeout: None,
- lastpay_index: last_pay_idx,
- })) => {
- match result {
- Ok(invoice) => {
- // Try to convert the invoice to WaitanyinvoiceResponse
- let wait_any_response_result: Result<WaitanyinvoiceResponse, _> =
- invoice.try_into();
- let wait_any_response = match wait_any_response_result {
- Ok(response) => response,
- Err(e) => {
- tracing::warn!(
- "Failed to parse WaitAnyInvoice response: {:?}",
- e
- );
- // Continue to the next iteration without panicking
- continue;
- }
- };
- // Check the status of the invoice
- // We only want to yield invoices that have been paid
- match wait_any_response.status {
- WaitanyinvoiceStatus::PAID => (),
- WaitanyinvoiceStatus::EXPIRED => continue,
- }
- last_pay_idx = wait_any_response.pay_index;
- let payment_hash = wait_any_response.payment_hash.to_string();
- let request_look_up = match wait_any_response.bolt12 {
- // If it is a bolt12 payment we need to get the offer_id as this is what we use as the request look up.
- // Since this is not returned in the wait any response,
- // we need to do a second query for it.
- Some(_) => {
- match fetch_invoice_by_payment_hash(
- &mut cln_client,
- &payment_hash,
- )
- .await
- {
- Ok(Some(invoice)) => {
- if let Some(local_offer_id) = invoice.local_offer_id {
- local_offer_id.to_string()
- } else {
- continue;
- }
- }
- Ok(None) => continue,
- Err(e) => {
- tracing::warn!(
- "Error fetching invoice by payment hash: {e}"
- );
- continue;
- }
- }
- }
- None => payment_hash,
- };
- return Some((request_look_up, (cln_client, last_pay_idx, cancel_token, is_active)));
- }
- Err(e) => {
- tracing::warn!("Error fetching invoice: {e}");
- tokio::time::sleep(Duration::from_secs(1)).await;
- continue;
- }
- }
- }
- }
- }
- },
- )
- .boxed();
- Ok(stream)
- }
- async fn get_payment_quote(
- &self,
- melt_quote_request: &MeltQuoteBolt11Request,
- ) -> Result<PaymentQuoteResponse, Self::Err> {
- let amount = melt_quote_request.amount_msat()?;
- let amount = amount / MSAT_IN_SAT.into();
- let relative_fee_reserve =
- (self.fee_reserve.percent_fee_reserve * u64::from(amount) 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: melt_quote_request.request.payment_hash().to_string(),
- amount,
- fee: fee.into(),
- state: MeltQuoteState::Unpaid,
- })
- }
- async fn pay_invoice(
- &self,
- melt_quote: mint::MeltQuote,
- partial_amount: Option<Amount>,
- max_fee: Option<Amount>,
- ) -> Result<PayInvoiceResponse, Self::Err> {
- let bolt11 = Bolt11Invoice::from_str(&melt_quote.request)?;
- let pay_state = self
- .check_outgoing_payment(&bolt11.payment_hash().to_string())
- .await?;
- match pay_state.status {
- MeltQuoteState::Unpaid | MeltQuoteState::Unknown | MeltQuoteState::Failed => (),
- MeltQuoteState::Paid => {
- tracing::debug!("Melt attempted on invoice already paid");
- return Err(Self::Err::InvoiceAlreadyPaid);
- }
- MeltQuoteState::Pending => {
- tracing::debug!("Melt attempted on invoice already pending");
- return Err(Self::Err::InvoicePaymentPending);
- }
- }
- let amount_msat = melt_quote
- .msat_to_pay
- .map(|a| CLN_Amount::from_msat(a.into()));
- let mut cln_client = self.cln_client.lock().await;
- let cln_response = cln_client
- .call(Request::Pay(PayRequest {
- bolt11: melt_quote.request.to_string(),
- amount_msat,
- label: None,
- riskfactor: None,
- maxfeepercent: None,
- retry_for: None,
- maxdelay: None,
- exemptfee: None,
- localinvreqid: None,
- exclude: None,
- maxfee: max_fee
- .map(|a| {
- let msat = to_unit(a, &melt_quote.unit, &CurrencyUnit::Msat)?;
- Ok::<CLN_Amount, Self::Err>(CLN_Amount::from_msat(msat.into()))
- })
- .transpose()?,
- description: None,
- partial_msat: partial_amount
- .map(|a| {
- let msat = to_unit(a, &melt_quote.unit, &CurrencyUnit::Msat)?;
- Ok::<cln_rpc::primitives::Amount, Self::Err>(CLN_Amount::from_msat(
- msat.into(),
- ))
- })
- .transpose()?,
- }))
- .await;
- let response = match cln_response {
- Ok(cln_rpc::Response::Pay(pay_response)) => {
- let status = match pay_response.status {
- PayStatus::COMPLETE => MeltQuoteState::Paid,
- PayStatus::PENDING => MeltQuoteState::Pending,
- PayStatus::FAILED => MeltQuoteState::Failed,
- };
- PayInvoiceResponse {
- payment_preimage: Some(hex::encode(pay_response.payment_preimage.to_vec())),
- payment_lookup_id: pay_response.payment_hash.to_string(),
- status,
- total_spent: to_unit(
- pay_response.amount_sent_msat.msat(),
- &CurrencyUnit::Msat,
- &melt_quote.unit,
- )?,
- unit: melt_quote.unit,
- }
- }
- Err(err) => {
- tracing::error!("Could not pay invoice: {}", err);
- return Err(Error::ClnRpc(err).into());
- }
- _ => {
- tracing::error!(
- "Error attempting to pay invoice: {}",
- bolt11.payment_hash().to_string()
- );
- return Err(Error::WrongClnResponse.into());
- }
- };
- Ok(response)
- }
- async fn create_invoice(
- &self,
- amount: Amount,
- unit: &CurrencyUnit,
- description: String,
- unix_expiry: u64,
- ) -> Result<CreateInvoiceResponse, Self::Err> {
- let time_now = unix_time();
- assert!(unix_expiry > time_now);
- let mut cln_client = self.cln_client.lock().await;
- let label = Uuid::new_v4().to_string();
- let amount = to_unit(amount, unit, &CurrencyUnit::Msat)?;
- let amount_msat = AmountOrAny::Amount(CLN_Amount::from_msat(amount.into()));
- let cln_response = cln_client
- .call(cln_rpc::Request::Invoice(InvoiceRequest {
- amount_msat,
- description,
- label: label.clone(),
- expiry: Some(unix_expiry - time_now),
- fallbacks: None,
- preimage: None,
- cltv: None,
- deschashonly: None,
- exposeprivatechannels: None,
- }))
- .await
- .map_err(Error::from)?;
- match cln_response {
- cln_rpc::Response::Invoice(invoice_res) => {
- let request = Bolt11Invoice::from_str(&invoice_res.bolt11)?;
- let expiry = request.expires_at().map(|t| t.as_secs());
- let payment_hash = request.payment_hash();
- Ok(CreateInvoiceResponse {
- request_lookup_id: payment_hash.to_string(),
- request,
- expiry,
- })
- }
- _ => {
- tracing::warn!("CLN returned wrong response kind");
- Err(Error::WrongClnResponse.into())
- }
- }
- }
- async fn check_incoming_invoice_status(
- &self,
- payment_hash: &str,
- ) -> Result<MintQuoteState, Self::Err> {
- let mut cln_client = self.cln_client.lock().await;
- let cln_response = cln_client
- .call(Request::ListInvoices(ListinvoicesRequest {
- payment_hash: Some(payment_hash.to_string()),
- label: None,
- invstring: None,
- offer_id: None,
- index: None,
- limit: None,
- start: None,
- }))
- .await
- .map_err(Error::from)?;
- let status = match cln_response {
- cln_rpc::Response::ListInvoices(invoice_response) => {
- match invoice_response.invoices.first() {
- Some(invoice_response) => {
- cln_invoice_status_to_mint_state(invoice_response.status)
- }
- None => {
- tracing::info!(
- "Check invoice called on unknown look up id: {}",
- payment_hash
- );
- return Err(Error::WrongClnResponse.into());
- }
- }
- }
- _ => {
- tracing::warn!("CLN returned wrong response kind");
- return Err(Error::WrongClnResponse.into());
- }
- };
- Ok(status)
- }
- async fn check_outgoing_payment(
- &self,
- payment_hash: &str,
- ) -> Result<PayInvoiceResponse, Self::Err> {
- let mut cln_client = self.cln_client.lock().await;
- let cln_response = cln_client
- .call(Request::ListPays(ListpaysRequest {
- payment_hash: Some(payment_hash.parse().map_err(|_| Error::InvalidHash)?),
- bolt11: None,
- status: None,
- start: None,
- index: None,
- limit: None,
- }))
- .await
- .map_err(Error::from)?;
- match cln_response {
- cln_rpc::Response::ListPays(pays_response) => match pays_response.pays.first() {
- Some(pays_response) => {
- let status = cln_pays_status_to_mint_state(pays_response.status);
- Ok(PayInvoiceResponse {
- payment_lookup_id: pays_response.payment_hash.to_string(),
- payment_preimage: pays_response.preimage.map(|p| hex::encode(p.to_vec())),
- status,
- total_spent: pays_response
- .amount_sent_msat
- .map_or(Amount::ZERO, |a| a.msat().into()),
- unit: CurrencyUnit::Msat,
- })
- }
- None => Ok(PayInvoiceResponse {
- payment_lookup_id: payment_hash.to_string(),
- payment_preimage: None,
- status: MeltQuoteState::Unknown,
- total_spent: Amount::ZERO,
- unit: CurrencyUnit::Msat,
- }),
- },
- _ => {
- tracing::warn!("CLN returned wrong response kind");
- Err(Error::WrongClnResponse.into())
- }
- }
- }
- }
- impl Cln {
- /// Get last pay index for cln
- async fn get_last_pay_index(&self) -> Result<Option<u64>, Error> {
- let mut cln_client = self.cln_client.lock().await;
- let cln_response = cln_client
- .call(cln_rpc::Request::ListInvoices(ListinvoicesRequest {
- index: None,
- invstring: None,
- label: None,
- limit: None,
- offer_id: None,
- payment_hash: None,
- start: None,
- }))
- .await
- .map_err(Error::from)?;
- match cln_response {
- cln_rpc::Response::ListInvoices(invoice_res) => match invoice_res.invoices.last() {
- Some(last_invoice) => Ok(last_invoice.pay_index),
- None => Ok(None),
- },
- _ => {
- tracing::warn!("CLN returned wrong response kind");
- Err(Error::WrongClnResponse)
- }
- }
- }
- }
- fn cln_invoice_status_to_mint_state(status: ListinvoicesInvoicesStatus) -> MintQuoteState {
- match status {
- ListinvoicesInvoicesStatus::UNPAID => MintQuoteState::Unpaid,
- ListinvoicesInvoicesStatus::PAID => MintQuoteState::Paid,
- ListinvoicesInvoicesStatus::EXPIRED => MintQuoteState::Unpaid,
- }
- }
- fn cln_pays_status_to_mint_state(status: ListpaysPaysStatus) -> MeltQuoteState {
- match status {
- ListpaysPaysStatus::PENDING => MeltQuoteState::Pending,
- ListpaysPaysStatus::COMPLETE => MeltQuoteState::Paid,
- ListpaysPaysStatus::FAILED => MeltQuoteState::Failed,
- }
- }
- async fn fetch_invoice_by_payment_hash(
- cln_client: &mut cln_rpc::ClnRpc,
- payment_hash: &str,
- ) -> Result<Option<ListinvoicesInvoices>, Error> {
- match cln_client
- .call(cln_rpc::Request::ListInvoices(ListinvoicesRequest {
- payment_hash: Some(payment_hash.to_string()),
- index: None,
- invstring: None,
- label: None,
- limit: None,
- offer_id: None,
- start: None,
- }))
- .await
- {
- Ok(cln_rpc::Response::ListInvoices(invoice_response)) => {
- Ok(invoice_response.invoices.first().cloned())
- }
- Ok(_) => {
- tracing::warn!("CLN returned an unexpected response type");
- Err(Error::WrongClnResponse)
- }
- Err(e) => {
- tracing::warn!("Error fetching invoice: {e}");
- Err(Error::from(e))
- }
- }
- }
|