use std::collections::BTreeMap;
use bitcoin::Network;
use fedimint_core::core::ModuleKind;
use fedimint_core::encoding::{Decodable, Encodable};
use fedimint_core::envs::BitcoinRpcConfig;
use fedimint_core::module::__reexports::serde_json;
use fedimint_core::util::SafeUrl;
use fedimint_core::{plugin_types_trait_impl_config, Feerate, PeerId};
use miniscript::descriptor::{Wpkh, Wsh};
use secp256k1::SecretKey;
use serde::{Deserialize, Serialize};
use crate::envs::FM_PORT_ESPLORA_ENV;
use crate::keys::CompressedPublicKey;
use crate::{PegInDescriptor, WalletCommonInit};
const DEFAULT_DEPOSIT_FEE_SATS: u64 = 1000;
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct WalletGenParams {
pub local: WalletGenParamsLocal,
pub consensus: WalletGenParamsConsensus,
}
impl WalletGenParams {
pub fn regtest(bitcoin_rpc: BitcoinRpcConfig) -> WalletGenParams {
WalletGenParams {
local: WalletGenParamsLocal { bitcoin_rpc },
consensus: WalletGenParamsConsensus {
network: Network::Regtest,
finality_delay: 10,
client_default_bitcoin_rpc: BitcoinRpcConfig {
kind: "esplora".to_string(),
url: SafeUrl::parse(&format!(
"http://127.0.0.1:{}/",
std::env::var(FM_PORT_ESPLORA_ENV).unwrap_or(String::from("50002"))
))
.expect("Failed to parse default esplora server"),
},
fee_consensus: Default::default(),
},
}
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct WalletGenParamsLocal {
pub bitcoin_rpc: BitcoinRpcConfig,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct WalletGenParamsConsensus {
pub network: Network,
pub finality_delay: u32,
pub client_default_bitcoin_rpc: BitcoinRpcConfig,
pub fee_consensus: FeeConsensus,
}
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct WalletConfig {
pub local: WalletConfigLocal,
pub private: WalletConfigPrivate,
pub consensus: WalletConfigConsensus,
}
#[derive(Clone, Debug, Serialize, Deserialize, Decodable, Encodable)]
pub struct WalletConfigLocal {
pub bitcoin_rpc: BitcoinRpcConfig,
}
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct WalletConfigPrivate {
pub peg_in_key: SecretKey,
}
#[derive(Clone, Debug, Serialize, Deserialize, Encodable, Decodable)]
pub struct WalletConfigConsensus {
pub network: Network,
pub peg_in_descriptor: PegInDescriptor,
pub peer_peg_in_keys: BTreeMap<PeerId, CompressedPublicKey>,
pub finality_delay: u32,
pub default_fee: Feerate,
pub fee_consensus: FeeConsensus,
pub client_default_bitcoin_rpc: BitcoinRpcConfig,
}
#[derive(Clone, Debug, Eq, PartialEq, Hash, Serialize, Deserialize, Encodable, Decodable)]
pub struct WalletClientConfig {
pub peg_in_descriptor: PegInDescriptor,
pub network: Network,
pub finality_delay: u32,
pub fee_consensus: FeeConsensus,
pub default_bitcoin_rpc: BitcoinRpcConfig,
}
impl std::fmt::Display for WalletClientConfig {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"WalletClientConfig {}",
serde_json::to_string(self).map_err(|_e| std::fmt::Error)?
)
}
}
#[derive(Debug, Clone, Copy, Eq, PartialEq, Hash, Serialize, Deserialize, Encodable, Decodable)]
pub struct FeeConsensus {
pub peg_in_abs: fedimint_core::Amount,
pub peg_out_abs: fedimint_core::Amount,
}
impl Default for FeeConsensus {
fn default() -> Self {
Self {
peg_in_abs: fedimint_core::Amount::from_sats(DEFAULT_DEPOSIT_FEE_SATS),
peg_out_abs: fedimint_core::Amount::ZERO,
}
}
}
impl WalletConfig {
#[allow(clippy::too_many_arguments)]
pub fn new(
pubkeys: BTreeMap<PeerId, CompressedPublicKey>,
sk: SecretKey,
threshold: usize,
network: Network,
finality_delay: u32,
bitcoin_rpc: BitcoinRpcConfig,
client_default_bitcoin_rpc: BitcoinRpcConfig,
fee_consensus: FeeConsensus,
) -> Self {
let peg_in_descriptor = if pubkeys.len() == 1 {
PegInDescriptor::Wpkh(
Wpkh::new(
*pubkeys
.values()
.next()
.expect("there is exactly one pub key"),
)
.expect("Our key type is always compressed"),
)
} else {
PegInDescriptor::Wsh(
Wsh::new_sortedmulti(threshold, pubkeys.values().copied().collect()).unwrap(),
)
};
Self {
local: WalletConfigLocal { bitcoin_rpc },
private: WalletConfigPrivate { peg_in_key: sk },
consensus: WalletConfigConsensus {
network,
peg_in_descriptor,
peer_peg_in_keys: pubkeys,
finality_delay,
default_fee: Feerate { sats_per_kvb: 1000 },
fee_consensus,
client_default_bitcoin_rpc,
},
}
}
}
impl WalletClientConfig {
pub fn new(
peg_in_descriptor: PegInDescriptor,
network: bitcoin::network::constants::Network,
finality_delay: u32,
default_bitcoin_rpc: BitcoinRpcConfig,
) -> Self {
Self {
peg_in_descriptor,
network,
finality_delay,
fee_consensus: Default::default(),
default_bitcoin_rpc,
}
}
}
plugin_types_trait_impl_config!(
WalletCommonInit,
WalletGenParams,
WalletGenParamsLocal,
WalletGenParamsConsensus,
WalletConfig,
WalletConfigLocal,
WalletConfigPrivate,
WalletConfigConsensus,
WalletClientConfig
);