config.rs 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. use std::path::PathBuf;
  2. use cdk::nuts::{CurrencyUnit, PublicKey};
  3. use cdk::Amount;
  4. use cdk_axum::cache;
  5. use config::{Config, ConfigError, File};
  6. use serde::{Deserialize, Serialize};
  7. #[derive(Debug, Clone, Serialize, Deserialize, Default)]
  8. pub struct Info {
  9. pub url: String,
  10. pub listen_host: String,
  11. pub listen_port: u16,
  12. pub mnemonic: String,
  13. pub input_fee_ppk: Option<u64>,
  14. pub http_cache: cache::Config,
  15. /// When this is set to true, the mint exposes a Swagger UI for it's API at
  16. /// `[listen_host]:[listen_port]/swagger-ui`
  17. ///
  18. /// This requires `mintd` was built with the `swagger` feature flag.
  19. pub enable_swagger_ui: Option<bool>,
  20. }
  21. #[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Default)]
  22. #[serde(rename_all = "lowercase")]
  23. pub enum LnBackend {
  24. #[default]
  25. None,
  26. Cln,
  27. Strike,
  28. LNbits,
  29. FakeWallet,
  30. Phoenixd,
  31. Lnd,
  32. }
  33. impl std::str::FromStr for LnBackend {
  34. type Err = String;
  35. fn from_str(s: &str) -> Result<Self, Self::Err> {
  36. match s.to_lowercase().as_str() {
  37. "cln" => Ok(LnBackend::Cln),
  38. "strike" => Ok(LnBackend::Strike),
  39. "lnbits" => Ok(LnBackend::LNbits),
  40. "fakewallet" => Ok(LnBackend::FakeWallet),
  41. "phoenixd" => Ok(LnBackend::Phoenixd),
  42. "lnd" => Ok(LnBackend::Lnd),
  43. _ => Err(format!("Unknown Lightning backend: {}", s)),
  44. }
  45. }
  46. }
  47. #[derive(Debug, Clone, Serialize, Deserialize)]
  48. pub struct Ln {
  49. pub ln_backend: LnBackend,
  50. pub invoice_description: Option<String>,
  51. pub min_mint: Amount,
  52. pub max_mint: Amount,
  53. pub min_melt: Amount,
  54. pub max_melt: Amount,
  55. }
  56. impl Default for Ln {
  57. fn default() -> Self {
  58. Ln {
  59. ln_backend: LnBackend::default(),
  60. invoice_description: None,
  61. min_mint: 1.into(),
  62. max_mint: 500_000.into(),
  63. min_melt: 1.into(),
  64. max_melt: 500_000.into(),
  65. }
  66. }
  67. }
  68. #[derive(Debug, Clone, Serialize, Deserialize, Default)]
  69. pub struct Strike {
  70. pub api_key: String,
  71. pub supported_units: Option<Vec<CurrencyUnit>>,
  72. }
  73. #[derive(Debug, Clone, Serialize, Deserialize, Default)]
  74. pub struct LNbits {
  75. pub admin_api_key: String,
  76. pub invoice_api_key: String,
  77. pub lnbits_api: String,
  78. pub fee_percent: f32,
  79. pub reserve_fee_min: Amount,
  80. }
  81. #[derive(Debug, Clone, Serialize, Deserialize, Default)]
  82. pub struct Cln {
  83. pub rpc_path: PathBuf,
  84. #[serde(default)]
  85. pub bolt12: bool,
  86. pub fee_percent: f32,
  87. pub reserve_fee_min: Amount,
  88. }
  89. #[derive(Debug, Clone, Serialize, Deserialize, Default)]
  90. pub struct Lnd {
  91. pub address: String,
  92. pub cert_file: PathBuf,
  93. pub macaroon_file: PathBuf,
  94. pub fee_percent: f32,
  95. pub reserve_fee_min: Amount,
  96. }
  97. #[derive(Debug, Clone, Serialize, Deserialize, Default)]
  98. pub struct Phoenixd {
  99. pub api_password: String,
  100. pub api_url: String,
  101. pub bolt12: bool,
  102. pub fee_percent: f32,
  103. pub reserve_fee_min: Amount,
  104. }
  105. #[derive(Debug, Clone, Serialize, Deserialize)]
  106. pub struct FakeWallet {
  107. pub supported_units: Vec<CurrencyUnit>,
  108. pub fee_percent: f32,
  109. pub reserve_fee_min: Amount,
  110. #[serde(default = "default_min_delay_time")]
  111. pub min_delay_time: u64,
  112. #[serde(default = "default_max_delay_time")]
  113. pub max_delay_time: u64,
  114. }
  115. impl Default for FakeWallet {
  116. fn default() -> Self {
  117. Self {
  118. supported_units: vec![CurrencyUnit::Sat],
  119. fee_percent: 0.02,
  120. reserve_fee_min: 2.into(),
  121. min_delay_time: 1,
  122. max_delay_time: 3,
  123. }
  124. }
  125. }
  126. // Helper functions to provide default values
  127. fn default_min_delay_time() -> u64 {
  128. 1
  129. }
  130. fn default_max_delay_time() -> u64 {
  131. 3
  132. }
  133. #[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Default)]
  134. #[serde(rename_all = "lowercase")]
  135. pub enum DatabaseEngine {
  136. #[default]
  137. Sqlite,
  138. Redb,
  139. }
  140. impl std::str::FromStr for DatabaseEngine {
  141. type Err = String;
  142. fn from_str(s: &str) -> Result<Self, Self::Err> {
  143. match s.to_lowercase().as_str() {
  144. "sqlite" => Ok(DatabaseEngine::Sqlite),
  145. "redb" => Ok(DatabaseEngine::Redb),
  146. _ => Err(format!("Unknown database engine: {}", s)),
  147. }
  148. }
  149. }
  150. #[derive(Debug, Clone, Serialize, Deserialize, Default)]
  151. pub struct Database {
  152. pub engine: DatabaseEngine,
  153. }
  154. /// CDK settings, derived from `config.toml`
  155. #[derive(Debug, Clone, Serialize, Deserialize, Default)]
  156. pub struct Settings {
  157. pub info: Info,
  158. pub mint_info: MintInfo,
  159. pub ln: Ln,
  160. pub cln: Option<Cln>,
  161. pub strike: Option<Strike>,
  162. pub lnbits: Option<LNbits>,
  163. pub phoenixd: Option<Phoenixd>,
  164. pub lnd: Option<Lnd>,
  165. pub fake_wallet: Option<FakeWallet>,
  166. pub database: Database,
  167. }
  168. #[derive(Debug, Clone, Serialize, Deserialize, Default)]
  169. pub struct MintInfo {
  170. /// name of the mint and should be recognizable
  171. pub name: String,
  172. /// hex pubkey of the mint
  173. pub pubkey: Option<PublicKey>,
  174. /// short description of the mint
  175. pub description: String,
  176. /// long description
  177. pub description_long: Option<String>,
  178. /// url to the mint icon
  179. pub icon_url: Option<String>,
  180. /// message of the day that the wallet must display to the user
  181. pub motd: Option<String>,
  182. /// Nostr publickey
  183. pub contact_nostr_public_key: Option<String>,
  184. /// Contact email
  185. pub contact_email: Option<String>,
  186. }
  187. impl Settings {
  188. #[must_use]
  189. pub fn new<P>(config_file_name: Option<P>) -> Self
  190. where
  191. P: Into<PathBuf>,
  192. {
  193. let default_settings = Self::default();
  194. // attempt to construct settings with file
  195. let from_file = Self::new_from_default(&default_settings, config_file_name);
  196. match from_file {
  197. Ok(f) => f,
  198. Err(e) => {
  199. tracing::error!(
  200. "Error reading config file, falling back to defaults. Error: {e:?}"
  201. );
  202. default_settings
  203. }
  204. }
  205. }
  206. fn new_from_default<P>(
  207. default: &Settings,
  208. config_file_name: Option<P>,
  209. ) -> Result<Self, ConfigError>
  210. where
  211. P: Into<PathBuf>,
  212. {
  213. let mut default_config_file_name = home::home_dir()
  214. .ok_or(ConfigError::NotFound("Config Path".to_string()))?
  215. .join("cashu-rs-mint");
  216. default_config_file_name.push("config.toml");
  217. let config: String = match config_file_name {
  218. Some(value) => value.into().to_string_lossy().to_string(),
  219. None => default_config_file_name.to_string_lossy().to_string(),
  220. };
  221. let builder = Config::builder();
  222. let config: Config = builder
  223. // use defaults
  224. .add_source(Config::try_from(default)?)
  225. // override with file contents
  226. .add_source(File::with_name(&config))
  227. .build()?;
  228. let settings: Settings = config.try_deserialize()?;
  229. match settings.ln.ln_backend {
  230. LnBackend::None => panic!("Ln backend must be set"),
  231. LnBackend::Cln => assert!(
  232. settings.cln.is_some(),
  233. "CLN backend requires a valid config."
  234. ),
  235. LnBackend::Strike => assert!(
  236. settings.strike.is_some(),
  237. "Strike backend requires a valid config."
  238. ),
  239. LnBackend::LNbits => assert!(
  240. settings.lnbits.is_some(),
  241. "LNbits backend requires a valid config"
  242. ),
  243. LnBackend::Phoenixd => assert!(
  244. settings.phoenixd.is_some(),
  245. "Phoenixd backend requires a valid config"
  246. ),
  247. LnBackend::Lnd => {
  248. assert!(
  249. settings.lnd.is_some(),
  250. "LND backend requires a valid config."
  251. )
  252. }
  253. LnBackend::FakeWallet => assert!(
  254. settings.fake_wallet.is_some(),
  255. "FakeWallet backend requires a valid config."
  256. ),
  257. }
  258. Ok(settings)
  259. }
  260. }