123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317 |
- use std::path::PathBuf;
- use std::sync::Arc;
- use cdk::cdk_database;
- use cdk::nuts::{CurrencyUnit, PublicKey};
- use cdk::Amount;
- use cdk_axum::cache;
- use cdk_redb::MintRedbDatabase;
- use cdk_sqlite::MintSqliteDatabase;
- use config::{Config, ConfigError, File};
- use serde::{Deserialize, Serialize};
- #[derive(Debug, Clone, Serialize, Deserialize, Default)]
- pub struct Info {
- pub url: String,
- pub listen_host: String,
- pub listen_port: u16,
- pub mnemonic: String,
- pub input_fee_ppk: Option<u64>,
- pub http_cache: cache::Config,
- /// When this is set to true, the mint exposes a Swagger UI for it's API at
- /// `[listen_host]:[listen_port]/swagger-ui`
- ///
- /// This requires `mintd` was built with the `swagger` feature flag.
- pub enable_swagger_ui: Option<bool>,
- }
- #[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Default)]
- #[serde(rename_all = "lowercase")]
- pub enum LnBackend {
- #[default]
- None,
- Cln,
- Strike,
- LNbits,
- FakeWallet,
- Phoenixd,
- Lnd,
- }
- impl std::str::FromStr for LnBackend {
- type Err = String;
- fn from_str(s: &str) -> Result<Self, Self::Err> {
- match s.to_lowercase().as_str() {
- "cln" => Ok(LnBackend::Cln),
- "strike" => Ok(LnBackend::Strike),
- "lnbits" => Ok(LnBackend::LNbits),
- "fakewallet" => Ok(LnBackend::FakeWallet),
- "phoenixd" => Ok(LnBackend::Phoenixd),
- "lnd" => Ok(LnBackend::Lnd),
- _ => Err(format!("Unknown Lightning backend: {}", s)),
- }
- }
- }
- #[derive(Debug, Clone, Serialize, Deserialize)]
- pub struct Ln {
- pub ln_backend: LnBackend,
- pub invoice_description: Option<String>,
- pub min_mint: Amount,
- pub max_mint: Amount,
- pub min_melt: Amount,
- pub max_melt: Amount,
- }
- impl Default for Ln {
- fn default() -> Self {
- Ln {
- ln_backend: LnBackend::default(),
- invoice_description: None,
- min_mint: 1.into(),
- max_mint: 500_000.into(),
- min_melt: 1.into(),
- max_melt: 500_000.into(),
- }
- }
- }
- #[derive(Debug, Clone, Serialize, Deserialize, Default)]
- pub struct Strike {
- pub api_key: String,
- pub supported_units: Option<Vec<CurrencyUnit>>,
- }
- #[derive(Debug, Clone, Serialize, Deserialize, Default)]
- pub struct LNbits {
- pub admin_api_key: String,
- pub invoice_api_key: String,
- pub lnbits_api: String,
- pub fee_percent: f32,
- pub reserve_fee_min: Amount,
- }
- #[derive(Debug, Clone, Serialize, Deserialize, Default)]
- pub struct Cln {
- pub rpc_path: PathBuf,
- #[serde(default)]
- pub bolt12: bool,
- pub fee_percent: f32,
- pub reserve_fee_min: Amount,
- }
- #[derive(Debug, Clone, Serialize, Deserialize, Default)]
- pub struct Lnd {
- pub address: String,
- pub cert_file: PathBuf,
- pub macaroon_file: PathBuf,
- pub fee_percent: f32,
- pub reserve_fee_min: Amount,
- }
- #[derive(Debug, Clone, Serialize, Deserialize, Default)]
- pub struct Phoenixd {
- pub api_password: String,
- pub api_url: String,
- pub bolt12: bool,
- pub fee_percent: f32,
- pub reserve_fee_min: Amount,
- }
- #[derive(Debug, Clone, Serialize, Deserialize)]
- pub struct FakeWallet {
- pub supported_units: Vec<CurrencyUnit>,
- pub fee_percent: f32,
- pub reserve_fee_min: Amount,
- #[serde(default = "default_min_delay_time")]
- pub min_delay_time: u64,
- #[serde(default = "default_max_delay_time")]
- pub max_delay_time: u64,
- }
- impl Default for FakeWallet {
- fn default() -> Self {
- Self {
- supported_units: vec![CurrencyUnit::Sat],
- fee_percent: 0.02,
- reserve_fee_min: 2.into(),
- min_delay_time: 1,
- max_delay_time: 3,
- }
- }
- }
- // Helper functions to provide default values
- fn default_min_delay_time() -> u64 {
- 1
- }
- fn default_max_delay_time() -> u64 {
- 3
- }
- #[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Default)]
- #[serde(rename_all = "lowercase")]
- pub enum DatabaseEngine {
- #[default]
- Sqlite,
- Redb,
- }
- impl std::str::FromStr for DatabaseEngine {
- type Err = String;
- fn from_str(s: &str) -> Result<Self, Self::Err> {
- match s.to_lowercase().as_str() {
- "sqlite" => Ok(DatabaseEngine::Sqlite),
- "redb" => Ok(DatabaseEngine::Redb),
- _ => Err(format!("Unknown database engine: {}", s)),
- }
- }
- }
- impl DatabaseEngine {
- /// Convert the database instance into a mint database
- pub async fn mint<P: Into<PathBuf>>(
- self,
- work_dir: P,
- ) -> Result<
- Arc<dyn cdk_database::MintDatabase<Err = cdk_database::Error> + Sync + Send + 'static>,
- cdk_database::Error,
- > {
- match self {
- DatabaseEngine::Sqlite => {
- let sql_db_path = work_dir.into().join("cdk-mintd.sqlite");
- let db = MintSqliteDatabase::new(&sql_db_path).await?;
- db.migrate().await;
- Ok(Arc::new(db))
- }
- DatabaseEngine::Redb => {
- let redb_path = work_dir.into().join("cdk-mintd.redb");
- Ok(Arc::new(MintRedbDatabase::new(&redb_path)?))
- }
- }
- }
- }
- #[derive(Debug, Clone, Serialize, Deserialize, Default)]
- pub struct Database {
- pub engine: DatabaseEngine,
- }
- /// CDK settings, derived from `config.toml`
- #[derive(Debug, Clone, Serialize, Deserialize, Default)]
- pub struct Settings {
- pub info: Info,
- pub mint_info: MintInfo,
- pub ln: Ln,
- pub cln: Option<Cln>,
- pub strike: Option<Strike>,
- pub lnbits: Option<LNbits>,
- pub phoenixd: Option<Phoenixd>,
- pub lnd: Option<Lnd>,
- pub fake_wallet: Option<FakeWallet>,
- pub database: Database,
- pub supported_units: Option<Vec<CurrencyUnit>>,
- pub remote_signatory: Option<String>,
- }
- #[derive(Debug, Clone, Serialize, Deserialize, Default)]
- pub struct MintInfo {
- /// name of the mint and should be recognizable
- pub name: String,
- /// hex pubkey of the mint
- pub pubkey: Option<PublicKey>,
- /// short description of the mint
- pub description: String,
- /// long description
- pub description_long: Option<String>,
- /// url to the mint icon
- pub icon_url: Option<String>,
- /// message of the day that the wallet must display to the user
- pub motd: Option<String>,
- /// Nostr publickey
- pub contact_nostr_public_key: Option<String>,
- /// Contact email
- pub contact_email: Option<String>,
- }
- impl Settings {
- #[must_use]
- pub fn new<P>(config_file_name: Option<P>) -> Self
- where
- P: Into<PathBuf>,
- {
- let default_settings = Self::default();
- // attempt to construct settings with file
- let from_file = Self::new_from_default(&default_settings, config_file_name);
- match from_file {
- Ok(f) => f,
- Err(e) => {
- tracing::warn!("Error reading config file ({:?})", e);
- default_settings
- }
- }
- }
- fn new_from_default<P>(
- default: &Settings,
- config_file_name: Option<P>,
- ) -> Result<Self, ConfigError>
- where
- P: Into<PathBuf>,
- {
- let mut default_config_file_name = home::home_dir()
- .ok_or(ConfigError::NotFound("Config Path".to_string()))?
- .join("cashu-rs-mint");
- default_config_file_name.push("config.toml");
- let config: String = match config_file_name {
- Some(value) => value.into().to_string_lossy().to_string(),
- None => default_config_file_name.to_string_lossy().to_string(),
- };
- let builder = Config::builder();
- let config: Config = builder
- // use defaults
- .add_source(Config::try_from(default)?)
- // override with file contents
- .add_source(File::with_name(&config))
- .build()?;
- let settings: Settings = config.try_deserialize()?;
- match settings.ln.ln_backend {
- LnBackend::None => panic!("Ln backend must be set"),
- LnBackend::Cln => assert!(
- settings.cln.is_some(),
- "CLN backend requires a valid config."
- ),
- LnBackend::Strike => assert!(
- settings.strike.is_some(),
- "Strike backend requires a valid config."
- ),
- LnBackend::LNbits => assert!(
- settings.lnbits.is_some(),
- "LNbits backend requires a valid config"
- ),
- LnBackend::Phoenixd => assert!(
- settings.phoenixd.is_some(),
- "Phoenixd backend requires a valid config"
- ),
- LnBackend::Lnd => {
- assert!(
- settings.lnd.is_some(),
- "LND backend requires a valid config."
- )
- }
- LnBackend::FakeWallet => assert!(
- settings.fake_wallet.is_some(),
- "FakeWallet backend requires a valid config."
- ),
- }
- Ok(settings)
- }
- }
|