pub mod ldk;
pub mod lnd;
use std::fmt::Debug;
use std::str::FromStr;
use std::sync::Arc;
use async_trait::async_trait;
use bitcoin::address::NetworkUnchecked;
use bitcoin::hashes::sha256;
use bitcoin::{Address, Network};
use fedimint_core::encoding::{Decodable, Encodable};
use fedimint_core::secp256k1::PublicKey;
use fedimint_core::task::TaskGroup;
use fedimint_core::util::{backoff_util, retry};
use fedimint_core::{secp256k1, Amount, BitcoinAmountOrAll};
use fedimint_ln_common::contracts::Preimage;
use fedimint_ln_common::route_hints::RouteHint;
use fedimint_ln_common::PrunedInvoice;
use fedimint_lnv2_common::contracts::PaymentImage;
use futures::stream::BoxStream;
use lightning_invoice::Bolt11Invoice;
use serde::{Deserialize, Serialize};
use thiserror::Error;
use tracing::{info, warn};
pub const MAX_LIGHTNING_RETRIES: u32 = 10;
pub type RouteHtlcStream<'a> = BoxStream<'a, InterceptPaymentRequest>;
#[derive(
Error, Debug, Serialize, Deserialize, Encodable, Decodable, Clone, Eq, PartialEq, Hash,
)]
pub enum LightningRpcError {
#[error("Failed to connect to Lightning node")]
FailedToConnect,
#[error("Failed to retrieve node info: {failure_reason}")]
FailedToGetNodeInfo { failure_reason: String },
#[error("Failed to retrieve route hints: {failure_reason}")]
FailedToGetRouteHints { failure_reason: String },
#[error("Payment failed: {failure_reason}")]
FailedPayment { failure_reason: String },
#[error("Failed to route HTLCs: {failure_reason}")]
FailedToRouteHtlcs { failure_reason: String },
#[error("Failed to complete HTLC: {failure_reason}")]
FailedToCompleteHtlc { failure_reason: String },
#[error("Failed to open channel: {failure_reason}")]
FailedToOpenChannel { failure_reason: String },
#[error("Failed to close channel: {failure_reason}")]
FailedToCloseChannelsWithPeer { failure_reason: String },
#[error("Failed to get Invoice: {failure_reason}")]
FailedToGetInvoice { failure_reason: String },
#[error("Failed to get funding address: {failure_reason}")]
FailedToGetLnOnchainAddress { failure_reason: String },
#[error("Failed to withdraw funds on-chain: {failure_reason}")]
FailedToWithdrawOnchain { failure_reason: String },
#[error("Failed to connect to peer: {failure_reason}")]
FailedToConnectToPeer { failure_reason: String },
#[error("Failed to list active channels: {failure_reason}")]
FailedToListActiveChannels { failure_reason: String },
#[error("Failed to get balances: {failure_reason}")]
FailedToGetBalances { failure_reason: String },
#[error("Failed to sync to chain: {failure_reason}")]
FailedToSyncToChain { failure_reason: String },
#[error("Invalid metadata: {failure_reason}")]
InvalidMetadata { failure_reason: String },
}
#[derive(Clone, Debug)]
pub struct LightningContext {
pub lnrpc: Arc<dyn ILnRpcClient>,
pub lightning_public_key: PublicKey,
pub lightning_alias: String,
pub lightning_network: Network,
}
#[async_trait]
pub trait ILnRpcClient: Debug + Send + Sync {
async fn info(&self) -> Result<GetNodeInfoResponse, LightningRpcError>;
async fn routehints(
&self,
num_route_hints: usize,
) -> Result<GetRouteHintsResponse, LightningRpcError>;
async fn pay(
&self,
invoice: Bolt11Invoice,
max_delay: u64,
max_fee: Amount,
) -> Result<PayInvoiceResponse, LightningRpcError> {
self.pay_private(
PrunedInvoice::try_from(invoice).map_err(|_| LightningRpcError::FailedPayment {
failure_reason: "Invoice has no amount".to_string(),
})?,
max_delay,
max_fee,
)
.await
}
async fn pay_private(
&self,
_invoice: PrunedInvoice,
_max_delay: u64,
_max_fee: Amount,
) -> Result<PayInvoiceResponse, LightningRpcError> {
Err(LightningRpcError::FailedPayment {
failure_reason: "Private payments not supported".to_string(),
})
}
fn supports_private_payments(&self) -> bool {
false
}
async fn route_htlcs<'a>(
self: Box<Self>,
task_group: &TaskGroup,
) -> Result<(RouteHtlcStream<'a>, Arc<dyn ILnRpcClient>), LightningRpcError>;
async fn complete_htlc(&self, htlc: InterceptPaymentResponse) -> Result<(), LightningRpcError>;
async fn create_invoice(
&self,
create_invoice_request: CreateInvoiceRequest,
) -> Result<CreateInvoiceResponse, LightningRpcError>;
async fn get_ln_onchain_address(
&self,
) -> Result<GetLnOnchainAddressResponse, LightningRpcError>;
async fn send_onchain(
&self,
payload: SendOnchainRequest,
) -> Result<SendOnchainResponse, LightningRpcError>;
async fn open_channel(
&self,
payload: OpenChannelRequest,
) -> Result<OpenChannelResponse, LightningRpcError>;
async fn close_channels_with_peer(
&self,
payload: CloseChannelsWithPeerRequest,
) -> Result<CloseChannelsWithPeerResponse, LightningRpcError>;
async fn list_active_channels(&self) -> Result<ListActiveChannelsResponse, LightningRpcError>;
async fn get_balances(&self) -> Result<GetBalancesResponse, LightningRpcError>;
}
impl dyn ILnRpcClient {
pub async fn parsed_route_hints(&self, num_route_hints: u32) -> Vec<RouteHint> {
if num_route_hints == 0 {
return vec![];
}
let route_hints =
self.routehints(num_route_hints as usize)
.await
.unwrap_or(GetRouteHintsResponse {
route_hints: Vec::new(),
});
route_hints.route_hints
}
pub async fn parsed_node_info(
&self,
) -> std::result::Result<(PublicKey, String, Network, u32, bool), LightningRpcError> {
let GetNodeInfoResponse {
pub_key,
alias,
network,
block_height,
synced_to_chain,
} = self.info().await?;
let network =
Network::from_str(&network).map_err(|e| LightningRpcError::InvalidMetadata {
failure_reason: format!("Invalid network {network}: {e}"),
})?;
Ok((pub_key, alias, network, block_height, synced_to_chain))
}
pub async fn wait_for_chain_sync(&self) -> std::result::Result<(), LightningRpcError> {
retry(
"Wait for chain sync",
backoff_util::background_backoff(),
|| async {
let info = self.info().await?;
let block_height = info.block_height;
if info.synced_to_chain {
Ok(())
} else {
warn!(?block_height, "Lightning node is not synced yet");
Err(anyhow::anyhow!("Not synced yet"))
}
},
)
.await
.map_err(|e| LightningRpcError::FailedToSyncToChain {
failure_reason: format!("Failed to sync to chain: {e:?}"),
})?;
info!("Gateway successfully synced with the chain");
Ok(())
}
}
#[derive(Serialize, Deserialize, Debug, Clone)]
pub struct ChannelInfo {
pub remote_pubkey: secp256k1::PublicKey,
pub channel_size_sats: u64,
pub outbound_liquidity_sats: u64,
pub inbound_liquidity_sats: u64,
pub short_channel_id: u64,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct GetNodeInfoResponse {
pub pub_key: PublicKey,
pub alias: String,
pub network: String,
pub block_height: u32,
pub synced_to_chain: bool,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct InterceptPaymentRequest {
pub payment_hash: sha256::Hash,
pub amount_msat: u64,
pub expiry: u32,
pub incoming_chan_id: u64,
pub short_channel_id: Option<u64>,
pub htlc_id: u64,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct InterceptPaymentResponse {
pub incoming_chan_id: u64,
pub htlc_id: u64,
pub payment_hash: sha256::Hash,
pub action: PaymentAction,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub enum PaymentAction {
Settle(Preimage),
Cancel,
Forward,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct GetRouteHintsResponse {
pub route_hints: Vec<RouteHint>,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct PayInvoiceResponse {
pub preimage: Preimage,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct CreateInvoiceRequest {
pub payment_hash: Option<sha256::Hash>,
pub amount_msat: u64,
pub expiry_secs: u32,
pub description: Option<InvoiceDescription>,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub enum InvoiceDescription {
Direct(String),
Hash(sha256::Hash),
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct CreateInvoiceResponse {
pub invoice: String,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct GetLnOnchainAddressResponse {
pub address: String,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct SendOnchainResponse {
pub txid: String,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct OpenChannelResponse {
pub funding_txid: String,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct CloseChannelsWithPeerResponse {
pub num_channels_closed: u32,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct ListActiveChannelsResponse {
pub channels: Vec<ChannelInfo>,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct GetBalancesResponse {
pub onchain_balance_sats: u64,
pub lightning_balance_msats: u64,
pub inbound_lightning_liquidity_msats: u64,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct OpenChannelRequest {
pub pubkey: secp256k1::PublicKey,
pub host: String,
pub channel_size_sats: u64,
pub push_amount_sats: u64,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct SendOnchainRequest {
pub address: Address<NetworkUnchecked>,
pub amount: BitcoinAmountOrAll,
pub fee_rate_sats_per_vbyte: u64,
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct CloseChannelsWithPeerRequest {
pub pubkey: secp256k1::PublicKey,
}
#[async_trait]
pub trait LightningV2Manager: Debug + Send + Sync {
async fn contains_incoming_contract(&self, payment_image: PaymentImage) -> bool;
}