config.rs 8.9 KB

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