use crate::crypt::encrypt;
use crate::error::{SdkError, SdkResult};
use crate::models::{LspAPI, OpeningFeeParams, OpeningFeeParamsMenu};
use anyhow::{anyhow, Result};
use prost::Message;
use sdk_common::grpc::{
self, LspFullListRequest, LspListRequest, PaymentInformation,
RegisterPaymentNotificationRequest, RegisterPaymentNotificationResponse, RegisterPaymentReply,
RegisterPaymentRequest, RemovePaymentNotificationRequest, RemovePaymentNotificationResponse,
SubscribeNotificationsRequest, UnsubscribeNotificationsRequest,
};
use sdk_common::prelude::BreezServer;
use sdk_common::with_connection_retry;
use serde::{Deserialize, Serialize};
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct LspInformation {
pub id: String,
pub name: String,
pub widget_url: String,
pub pubkey: String,
pub host: String,
pub base_fee_msat: i64,
pub fee_rate: f64,
pub time_lock_delta: u32,
pub min_htlc_msat: i64,
pub lsp_pubkey: Vec<u8>,
pub opening_fee_params_list: OpeningFeeParamsMenu,
}
impl LspInformation {
fn try_from(lsp_id: &str, lsp_info: grpc::LspInformation) -> Result<Self> {
let info = LspInformation {
id: lsp_id.to_string(),
name: lsp_info.name,
widget_url: lsp_info.widget_url,
pubkey: lsp_info.pubkey,
host: lsp_info.host,
base_fee_msat: lsp_info.base_fee_msat,
fee_rate: lsp_info.fee_rate,
time_lock_delta: lsp_info.time_lock_delta,
min_htlc_msat: lsp_info.min_htlc_msat,
lsp_pubkey: lsp_info.lsp_pubkey,
opening_fee_params_list: OpeningFeeParamsMenu::try_from(
lsp_info.opening_fee_params_menu,
)?,
};
Ok(info)
}
pub(crate) fn cheapest_open_channel_fee(&self, expiry: u32) -> Result<&OpeningFeeParams> {
for fee in &self.opening_fee_params_list.values {
match fee.valid_for(expiry) {
Ok(valid) => {
if valid {
return Ok(fee);
}
}
Err(e) => return Err(anyhow!("Failed to calculate open channel fees: {e}")),
}
}
self.opening_fee_params_list
.values
.last()
.ok_or_else(|| anyhow!("Dynamic fees menu contains no values"))
}
}
#[tonic::async_trait]
impl LspAPI for BreezServer {
async fn list_lsps(&self, pubkey: String) -> SdkResult<Vec<LspInformation>> {
let mut client = self.get_channel_opener_client().await?;
let request = LspListRequest { pubkey };
let response = with_connection_retry!(client.lsp_list(request.clone())).await?;
let mut lsp_list: Vec<LspInformation> = Vec::new();
for (lsp_id, lsp_info) in response.into_inner().lsps.into_iter() {
match LspInformation::try_from(&lsp_id, lsp_info) {
Ok(lsp) => lsp_list.push(lsp),
Err(e) => error!("LSP Information validation failed for LSP {lsp_id}: {e}"),
}
}
lsp_list.sort_by(|a, b| a.name.to_lowercase().cmp(&b.name.to_lowercase()));
Ok(lsp_list)
}
async fn list_used_lsps(&self, pubkey: String) -> SdkResult<Vec<LspInformation>> {
let mut client = self.get_channel_opener_client().await?;
let request = LspFullListRequest { pubkey };
let response = with_connection_retry!(client.lsp_full_list(request.clone())).await?;
let mut lsp_list: Vec<LspInformation> = Vec::new();
for grpc_lsp_info in response.into_inner().lsps.into_iter() {
let lsp_id = grpc_lsp_info.id.clone();
match LspInformation::try_from(&lsp_id, grpc_lsp_info) {
Ok(lsp) => lsp_list.push(lsp),
Err(e) => error!("LSP Information validation failed for LSP {lsp_id}: {e}"),
}
}
lsp_list.sort_by(|a, b| a.name.to_lowercase().cmp(&b.name.to_lowercase()));
Ok(lsp_list)
}
async fn register_payment_notifications(
&self,
lsp_id: String,
lsp_pubkey: Vec<u8>,
webhook_url: String,
webhook_url_signature: String,
) -> SdkResult<RegisterPaymentNotificationResponse> {
let subscribe_request = SubscribeNotificationsRequest {
url: webhook_url,
signature: webhook_url_signature,
};
let mut client = self.get_payment_notifier_client().await;
let mut buf = Vec::with_capacity(subscribe_request.encoded_len());
subscribe_request
.encode(&mut buf)
.map_err(|e| SdkError::Generic {
err: format!("(LSP {lsp_id}) Failed to encode subscribe request: {e}"),
})?;
let request = RegisterPaymentNotificationRequest {
lsp_id,
blob: encrypt(lsp_pubkey, buf)?,
};
let response =
with_connection_retry!(client.register_payment_notification(request.clone())).await?;
Ok(response.into_inner())
}
async fn unregister_payment_notifications(
&self,
lsp_id: String,
lsp_pubkey: Vec<u8>,
webhook_url: String,
webhook_url_signature: String,
) -> SdkResult<RemovePaymentNotificationResponse> {
let unsubscribe_request = UnsubscribeNotificationsRequest {
url: webhook_url,
signature: webhook_url_signature,
};
let mut client = self.get_payment_notifier_client().await;
let mut buf = Vec::with_capacity(unsubscribe_request.encoded_len());
unsubscribe_request
.encode(&mut buf)
.map_err(|e| SdkError::Generic {
err: format!("(LSP {lsp_id}) Failed to encode unsubscribe request: {e}"),
})?;
let request = RemovePaymentNotificationRequest {
lsp_id,
blob: encrypt(lsp_pubkey, buf)?,
};
let response =
with_connection_retry!(client.remove_payment_notification(request.clone())).await?;
Ok(response.into_inner())
}
async fn register_payment(
&self,
lsp_id: String,
lsp_pubkey: Vec<u8>,
payment_info: PaymentInformation,
) -> SdkResult<RegisterPaymentReply> {
let mut client = self.get_channel_opener_client().await?;
let mut buf = Vec::with_capacity(payment_info.encoded_len());
payment_info
.encode(&mut buf)
.map_err(|e| SdkError::ServiceConnectivity {
err: format!("(LSP {lsp_id}) Failed to encode payment info: {e}"),
})?;
let request = RegisterPaymentRequest {
lsp_id,
blob: encrypt(lsp_pubkey, buf)?,
};
let response = with_connection_retry!(client.register_payment(request.clone())).await?;
Ok(response.into_inner())
}
}
#[cfg(test)]
mod tests {
use crate::{LspInformation, OpeningFeeParams};
use super::OpeningFeeParamsMenu;
use anyhow::Result;
use chrono::{Duration, Utc};
#[test]
fn test_cheapest_open_channel_fee() -> Result<()> {
let mut tested_fees: Vec<OpeningFeeParams> = vec![];
for i in 1..3 {
tested_fees.push(OpeningFeeParams {
min_msat: i,
proportional: i as u32,
valid_until: std::ops::Add::add(Utc::now(), Duration::seconds((i * 3600) as i64))
.to_rfc3339(),
max_idle_time: i as u32,
max_client_to_self_delay: i as u32,
promise: format!("promise {i}"),
})
}
let mut lsp_info = LspInformation {
id: "id".to_string(),
name: "test lsp".to_string(),
widget_url: "".to_string(),
pubkey: "pubkey".to_string(),
host: "localhost".to_string(),
base_fee_msat: 1,
fee_rate: 1.0,
time_lock_delta: 32,
min_htlc_msat: 1000,
lsp_pubkey: hex::decode("A0").unwrap(),
opening_fee_params_list: OpeningFeeParamsMenu {
values: tested_fees,
},
};
for expiry in 1..3 {
let fee = lsp_info
.cheapest_open_channel_fee(expiry * 3600 - 1000)
.unwrap();
assert_eq!(fee.min_msat, expiry as u64);
}
let fee = lsp_info.cheapest_open_channel_fee(4 * 3600 - 1000).unwrap();
assert_eq!(fee.min_msat, 2);
lsp_info.opening_fee_params_list = OpeningFeeParamsMenu { values: vec![] };
let err = lsp_info.cheapest_open_channel_fee(4 * 3600).err().unwrap();
assert_eq!(err.to_string(), "Dynamic fees menu contains no values");
Ok(())
}
}