setup.rs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. #[cfg(feature = "fakewallet")]
  2. use std::collections::HashMap;
  3. #[cfg(feature = "fakewallet")]
  4. use std::collections::HashSet;
  5. use std::path::Path;
  6. #[cfg(feature = "ldk-node")]
  7. use std::path::PathBuf;
  8. use std::sync::Arc;
  9. use async_trait::async_trait;
  10. #[cfg(feature = "fakewallet")]
  11. use bip39::rand::{thread_rng, Rng};
  12. use cdk::cdk_database::KVStore;
  13. use cdk::cdk_payment::MintPayment;
  14. use cdk::nuts::CurrencyUnit;
  15. #[cfg(any(
  16. feature = "lnbits",
  17. feature = "cln",
  18. feature = "lnd",
  19. feature = "ldk-node",
  20. feature = "fakewallet"
  21. ))]
  22. use cdk::types::FeeReserve;
  23. use crate::config::{self, Settings};
  24. #[cfg(feature = "cln")]
  25. use crate::expand_path;
  26. #[async_trait]
  27. pub trait LnBackendSetup {
  28. async fn setup(
  29. &self,
  30. settings: &Settings,
  31. unit: CurrencyUnit,
  32. runtime: Option<std::sync::Arc<tokio::runtime::Runtime>>,
  33. work_dir: &Path,
  34. kv_store: Option<Arc<dyn KVStore<Err = cdk::cdk_database::Error> + Send + Sync>>,
  35. ) -> anyhow::Result<impl MintPayment>;
  36. }
  37. #[cfg(feature = "cln")]
  38. #[async_trait]
  39. impl LnBackendSetup for config::Cln {
  40. async fn setup(
  41. &self,
  42. _settings: &Settings,
  43. _unit: CurrencyUnit,
  44. _runtime: Option<std::sync::Arc<tokio::runtime::Runtime>>,
  45. _work_dir: &Path,
  46. kv_store: Option<Arc<dyn KVStore<Err = cdk::cdk_database::Error> + Send + Sync>>,
  47. ) -> anyhow::Result<cdk_cln::Cln> {
  48. // Validate required connection field
  49. if self.rpc_path.as_os_str().is_empty() {
  50. return Err(anyhow::anyhow!(
  51. "CLN rpc_path must be set via config or CDK_MINTD_CLN_RPC_PATH env var"
  52. ));
  53. }
  54. let cln_socket = expand_path(
  55. self.rpc_path
  56. .to_str()
  57. .ok_or(anyhow::anyhow!("cln socket not defined"))?,
  58. )
  59. .ok_or(anyhow::anyhow!("cln socket not defined"))?;
  60. let fee_reserve = FeeReserve {
  61. min_fee_reserve: self.reserve_fee_min,
  62. percent_fee_reserve: self.fee_percent,
  63. };
  64. let cln = cdk_cln::Cln::new(
  65. cln_socket,
  66. fee_reserve,
  67. kv_store.expect("Cln needs kv store"),
  68. )
  69. .await?;
  70. Ok(cln)
  71. }
  72. }
  73. #[cfg(feature = "lnbits")]
  74. #[async_trait]
  75. impl LnBackendSetup for config::LNbits {
  76. async fn setup(
  77. &self,
  78. _settings: &Settings,
  79. _unit: CurrencyUnit,
  80. _runtime: Option<std::sync::Arc<tokio::runtime::Runtime>>,
  81. _work_dir: &Path,
  82. _kv_store: Option<Arc<dyn KVStore<Err = cdk::cdk_database::Error> + Send + Sync>>,
  83. ) -> anyhow::Result<cdk_lnbits::LNbits> {
  84. use anyhow::bail;
  85. // Validate required connection fields
  86. if self.admin_api_key.is_empty() {
  87. bail!("LNbits admin_api_key must be set via config or CDK_MINTD_LNBITS_ADMIN_API_KEY env var");
  88. }
  89. if self.invoice_api_key.is_empty() {
  90. bail!("LNbits invoice_api_key must be set via config or CDK_MINTD_LNBITS_INVOICE_API_KEY env var");
  91. }
  92. if self.lnbits_api.is_empty() {
  93. bail!(
  94. "LNbits lnbits_api must be set via config or CDK_MINTD_LNBITS_LNBITS_API env var"
  95. );
  96. }
  97. let admin_api_key = &self.admin_api_key;
  98. let invoice_api_key = &self.invoice_api_key;
  99. let fee_reserve = FeeReserve {
  100. min_fee_reserve: self.reserve_fee_min,
  101. percent_fee_reserve: self.fee_percent,
  102. };
  103. let lnbits = cdk_lnbits::LNbits::new(
  104. admin_api_key.clone(),
  105. invoice_api_key.clone(),
  106. self.lnbits_api.clone(),
  107. fee_reserve,
  108. )
  109. .await?;
  110. // Use v1 websocket API
  111. lnbits.subscribe_ws().await?;
  112. Ok(lnbits)
  113. }
  114. }
  115. #[cfg(feature = "lnd")]
  116. #[async_trait]
  117. impl LnBackendSetup for config::Lnd {
  118. async fn setup(
  119. &self,
  120. _settings: &Settings,
  121. _unit: CurrencyUnit,
  122. _runtime: Option<std::sync::Arc<tokio::runtime::Runtime>>,
  123. _work_dir: &Path,
  124. kv_store: Option<Arc<dyn KVStore<Err = cdk::cdk_database::Error> + Send + Sync>>,
  125. ) -> anyhow::Result<cdk_lnd::Lnd> {
  126. use anyhow::bail;
  127. // Validate required connection fields
  128. if self.address.is_empty() {
  129. bail!("LND address must be set via config or CDK_MINTD_LND_ADDRESS env var");
  130. }
  131. if self.cert_file.as_os_str().is_empty() {
  132. bail!("LND cert_file must be set via config or CDK_MINTD_LND_CERT_FILE env var");
  133. }
  134. if self.macaroon_file.as_os_str().is_empty() {
  135. bail!(
  136. "LND macaroon_file must be set via config or CDK_MINTD_LND_MACAROON_FILE env var"
  137. );
  138. }
  139. let address = &self.address;
  140. let cert_file = &self.cert_file;
  141. let macaroon_file = &self.macaroon_file;
  142. let fee_reserve = FeeReserve {
  143. min_fee_reserve: self.reserve_fee_min,
  144. percent_fee_reserve: self.fee_percent,
  145. };
  146. let lnd = cdk_lnd::Lnd::new(
  147. address.to_string(),
  148. cert_file.clone(),
  149. macaroon_file.clone(),
  150. fee_reserve,
  151. kv_store.expect("Lnd needs kv store"),
  152. )
  153. .await?;
  154. Ok(lnd)
  155. }
  156. }
  157. #[cfg(feature = "fakewallet")]
  158. #[async_trait]
  159. impl LnBackendSetup for config::FakeWallet {
  160. async fn setup(
  161. &self,
  162. _settings: &Settings,
  163. unit: CurrencyUnit,
  164. _runtime: Option<std::sync::Arc<tokio::runtime::Runtime>>,
  165. _work_dir: &Path,
  166. _kv_store: Option<Arc<dyn KVStore<Err = cdk::cdk_database::Error> + Send + Sync>>,
  167. ) -> anyhow::Result<cdk_fake_wallet::FakeWallet> {
  168. let fee_reserve = FeeReserve {
  169. min_fee_reserve: self.reserve_fee_min,
  170. percent_fee_reserve: self.fee_percent,
  171. };
  172. // calculate random delay time
  173. let mut rng = thread_rng();
  174. let delay_time = rng.gen_range(self.min_delay_time..=self.max_delay_time);
  175. let fake_wallet = cdk_fake_wallet::FakeWallet::new(
  176. fee_reserve,
  177. HashMap::default(),
  178. HashSet::default(),
  179. delay_time,
  180. unit,
  181. );
  182. Ok(fake_wallet)
  183. }
  184. }
  185. #[cfg(feature = "grpc-processor")]
  186. #[async_trait]
  187. impl LnBackendSetup for config::GrpcProcessor {
  188. async fn setup(
  189. &self,
  190. _settings: &Settings,
  191. _unit: CurrencyUnit,
  192. _runtime: Option<std::sync::Arc<tokio::runtime::Runtime>>,
  193. _work_dir: &Path,
  194. _kv_store: Option<Arc<dyn KVStore<Err = cdk::cdk_database::Error> + Send + Sync>>,
  195. ) -> anyhow::Result<cdk_payment_processor::PaymentProcessorClient> {
  196. let payment_processor = cdk_payment_processor::PaymentProcessorClient::new(
  197. &self.addr,
  198. self.port,
  199. self.tls_dir.clone(),
  200. )
  201. .await?;
  202. Ok(payment_processor)
  203. }
  204. }
  205. #[cfg(feature = "ldk-node")]
  206. #[async_trait]
  207. impl LnBackendSetup for config::LdkNode {
  208. async fn setup(
  209. &self,
  210. settings: &Settings,
  211. _unit: CurrencyUnit,
  212. _runtime: Option<std::sync::Arc<tokio::runtime::Runtime>>,
  213. work_dir: &Path,
  214. _kv_store: Option<Arc<dyn KVStore<Err = cdk::cdk_database::Error> + Send + Sync>>,
  215. ) -> anyhow::Result<cdk_ldk_node::CdkLdkNode> {
  216. use std::net::SocketAddr;
  217. use anyhow::bail;
  218. use bip39::Mnemonic;
  219. use bitcoin::Network;
  220. let fee_reserve = FeeReserve {
  221. min_fee_reserve: self.reserve_fee_min,
  222. percent_fee_reserve: self.fee_percent,
  223. };
  224. // Parse network from config
  225. let network = match self
  226. .bitcoin_network
  227. .as_ref()
  228. .map(|n| n.to_lowercase())
  229. .as_deref()
  230. .unwrap_or("regtest")
  231. {
  232. "mainnet" | "bitcoin" => Network::Bitcoin,
  233. "testnet" => Network::Testnet,
  234. "signet" => Network::Signet,
  235. _ => Network::Regtest,
  236. };
  237. // Parse chain source from config
  238. let chain_source = match self
  239. .chain_source_type
  240. .as_ref()
  241. .map(|s| s.to_lowercase())
  242. .as_deref()
  243. .unwrap_or("esplora")
  244. {
  245. "bitcoinrpc" => {
  246. let host = self
  247. .bitcoind_rpc_host
  248. .clone()
  249. .unwrap_or_else(|| "127.0.0.1".to_string());
  250. let port = self.bitcoind_rpc_port.unwrap_or(18443);
  251. let user = self
  252. .bitcoind_rpc_user
  253. .clone()
  254. .unwrap_or_else(|| "testuser".to_string());
  255. let password = self
  256. .bitcoind_rpc_password
  257. .clone()
  258. .unwrap_or_else(|| "testpass".to_string());
  259. cdk_ldk_node::ChainSource::BitcoinRpc(cdk_ldk_node::BitcoinRpcConfig {
  260. host,
  261. port,
  262. user,
  263. password,
  264. })
  265. }
  266. _ => {
  267. let esplora_url = self
  268. .esplora_url
  269. .clone()
  270. .unwrap_or_else(|| "https://mutinynet.com/api".to_string());
  271. cdk_ldk_node::ChainSource::Esplora(esplora_url)
  272. }
  273. };
  274. // Parse gossip source from config
  275. let gossip_source = match self.rgs_url.clone() {
  276. Some(rgs_url) => cdk_ldk_node::GossipSource::RapidGossipSync(rgs_url),
  277. None => cdk_ldk_node::GossipSource::P2P,
  278. };
  279. // Get storage directory path
  280. let storage_dir_path = if let Some(dir_path) = &self.storage_dir_path {
  281. dir_path.clone()
  282. } else {
  283. let mut work_dir = work_dir.to_path_buf();
  284. work_dir.push("ldk-node");
  285. work_dir.to_string_lossy().to_string()
  286. };
  287. // Get LDK node listen address
  288. let host = self
  289. .ldk_node_host
  290. .clone()
  291. .unwrap_or_else(|| "127.0.0.1".to_string());
  292. let port = self.ldk_node_port.unwrap_or(8090);
  293. let socket_addr = SocketAddr::new(host.parse()?, port);
  294. // Parse socket address using ldk_node's SocketAddress
  295. // We need to get the actual socket address struct from ldk_node
  296. // For now, let's construct it manually based on the cdk-ldk-node implementation
  297. let listen_address = vec![socket_addr.into()];
  298. // Check if ldk_node_mnemonic is provided in the ldk_node config
  299. let mnemonic_opt = settings
  300. .clone()
  301. .ldk_node
  302. .as_ref()
  303. .and_then(|ldk_config| ldk_config.ldk_node_mnemonic.clone());
  304. // Only set seed if mnemonic is explicitly provided
  305. // This maintains backward compatibility with existing nodes that use LDK's default seed storage
  306. let seed = if let Some(mnemonic_str) = mnemonic_opt {
  307. Some(
  308. mnemonic_str
  309. .parse::<Mnemonic>()
  310. .map_err(|e| anyhow::anyhow!("invalid ldk_node_mnemonic in config: {e}"))?,
  311. )
  312. } else {
  313. // Check if this is a new node or an existing node
  314. let storage_dir = PathBuf::from(&storage_dir_path);
  315. let keys_seed_file = storage_dir.join("keys_seed");
  316. if !keys_seed_file.exists() {
  317. bail!("ldk_node_mnemonic should be set in the [ldk_node] configuration section.");
  318. }
  319. // Existing node with stored seed, don't set a mnemonic
  320. None
  321. };
  322. let ldk_node_settings = settings
  323. .ldk_node
  324. .as_ref()
  325. .ok_or_else(|| anyhow::anyhow!("ldk_node configuration is required"))?;
  326. let announce_addrs: Vec<_> = ldk_node_settings
  327. .ldk_node_announce_addresses
  328. .as_ref()
  329. .map(|addrs| addrs.iter().filter_map(|addr| addr.parse().ok()).collect())
  330. .unwrap_or_default();
  331. let mut ldk_node_builder = cdk_ldk_node::CdkLdkNodeBuilder::new(
  332. network,
  333. chain_source,
  334. gossip_source,
  335. storage_dir_path,
  336. fee_reserve,
  337. listen_address,
  338. );
  339. // Only set seed if provided
  340. if let Some(mnemonic) = seed {
  341. ldk_node_builder = ldk_node_builder.with_seed(mnemonic);
  342. }
  343. if !announce_addrs.is_empty() {
  344. ldk_node_builder = ldk_node_builder.with_announcement_address(announce_addrs)
  345. }
  346. // Configure webserver address if specified
  347. let webserver_addr = if let Some(host) = &self.webserver_host {
  348. let port = self.webserver_port.unwrap_or(8091);
  349. let socket_addr: SocketAddr = format!("{host}:{port}").parse()?;
  350. Some(socket_addr)
  351. } else if self.webserver_port.is_some() {
  352. // If only port is specified, use default host
  353. let port = self.webserver_port.unwrap_or(8091);
  354. let socket_addr: SocketAddr = format!("127.0.0.1:{port}").parse()?;
  355. Some(socket_addr)
  356. } else {
  357. // Use default webserver address if nothing is configured
  358. Some(cdk_ldk_node::CdkLdkNode::default_web_addr())
  359. };
  360. println!(
  361. "webserver: {}",
  362. webserver_addr.map_or("none".to_string(), |a| a.to_string())
  363. );
  364. if let Some(log_dir_path) = ldk_node_settings.log_dir_path.as_ref() {
  365. ldk_node_builder = ldk_node_builder.with_log_dir_path(log_dir_path.clone());
  366. }
  367. let mut ldk_node = ldk_node_builder.build()?;
  368. ldk_node.set_web_addr(webserver_addr);
  369. Ok(ldk_node)
  370. }
  371. }