#![warn(clippy::pedantic)]
#![allow(clippy::cast_possible_truncation)]
#![allow(clippy::default_trait_access)]
#![allow(clippy::doc_markdown)]
#![allow(clippy::match_same_arms)]
#![allow(clippy::missing_errors_doc)]
#![allow(clippy::missing_panics_doc)]
#![allow(clippy::module_name_repetitions)]
#![allow(clippy::must_use_candidate)]
#![allow(clippy::return_self_not_must_use)]
pub mod backup;
mod cli;
pub mod client_db;
mod input;
mod oob;
pub mod output;
use std::cmp::{min, Ordering};
use std::collections::BTreeMap;
use std::fmt;
use std::fmt::{Display, Formatter};
use std::io::Read;
use std::str::FromStr;
use std::sync::Arc;
use std::time::Duration;
use anyhow::{anyhow, bail, ensure, Context as _};
use async_stream::{stream, try_stream};
use backup::recovery::MintRecovery;
use base64::Engine as _;
use bitcoin_hashes::{sha256, sha256t, Hash, HashEngine as BitcoinHashEngine};
use client_db::{DbKeyPrefix, NoteKeyPrefix, RecoveryFinalizedKey};
use fedimint_client::module::init::{
ClientModuleInit, ClientModuleInitArgs, ClientModuleRecoverArgs,
};
use fedimint_client::module::{ClientContext, ClientModule, IClientModule};
use fedimint_client::oplog::{OperationLogEntry, UpdateStreamOrOutcome};
use fedimint_client::sm::util::MapStateTransitions;
use fedimint_client::sm::{Context, DynState, ModuleNotifier, State, StateTransition};
use fedimint_client::transaction::{ClientInput, ClientOutput, TransactionBuilder};
use fedimint_client::{sm_enum_variant_translation, DynGlobalClientContext};
use fedimint_core::config::{FederationId, FederationIdPrefix};
use fedimint_core::core::{Decoder, IntoDynInstance, ModuleInstanceId, OperationId};
use fedimint_core::db::{
AutocommitError, Database, DatabaseTransaction, DatabaseVersion,
IDatabaseTransactionOpsCoreTyped,
};
use fedimint_core::encoding::{Decodable, DecodeError, Encodable};
use fedimint_core::invite_code::{InviteCode, InviteCodeV2};
use fedimint_core::module::registry::ModuleDecoderRegistry;
use fedimint_core::module::{
ApiVersion, CommonModuleInit, ModuleCommon, ModuleInit, MultiApiVersion,
};
use fedimint_core::util::{BoxFuture, BoxStream, NextOrPending, SafeUrl};
use fedimint_core::{
apply, async_trait_maybe_send, push_db_pair_items, Amount, OutPoint, PeerId, Tiered,
TieredCounts, TieredMulti, TransactionId,
};
use fedimint_derive_secret::{ChildId, DerivableSecret};
use fedimint_logging::LOG_CLIENT_MODULE_MINT;
pub use fedimint_mint_common as common;
use fedimint_mint_common::config::MintClientConfig;
pub use fedimint_mint_common::*;
use futures::{pin_mut, StreamExt};
use hex::ToHex;
use secp256k1_zkp::{All, KeyPair, Secp256k1};
use serde::{Deserialize, Serialize};
use strum::IntoEnumIterator;
use tbs::{AggregatePublicKey, Signature};
use thiserror::Error;
use tracing::{debug, warn};
use crate::backup::EcashBackup;
use crate::client_db::{
CancelledOOBSpendKey, CancelledOOBSpendKeyPrefix, NextECashNoteIndexKey,
NextECashNoteIndexKeyPrefix, NoteKey,
};
use crate::input::{
MintInputCommon, MintInputStateCreated, MintInputStateMachine, MintInputStates,
};
use crate::oob::{MintOOBStateMachine, MintOOBStates, MintOOBStatesCreated};
use crate::output::{
MintOutputCommon, MintOutputStateMachine, MintOutputStates, MintOutputStatesCreated,
NoteIssuanceRequest,
};
const MINT_E_CASH_TYPE_CHILD_ID: ChildId = ChildId(0);
pub const LOG_TARGET: &str = "client::module::mint";
#[derive(Clone, Debug, Encodable, PartialEq, Eq)]
pub struct OOBNotes(Vec<OOBNotesPart>);
#[derive(Clone, Debug, Decodable, Encodable, PartialEq, Eq)]
enum OOBNotesPart {
Notes(TieredMulti<SpendableNote>),
FederationIdPrefix(FederationIdPrefix),
Invite {
peer_apis: Vec<(PeerId, SafeUrl)>,
federation_id: FederationId,
},
ApiSecret(String),
#[encodable_default]
Default {
variant: u64,
bytes: Vec<u8>,
},
}
impl OOBNotes {
pub fn new(
federation_id_prefix: FederationIdPrefix,
notes: TieredMulti<SpendableNote>,
) -> Self {
Self(vec![
OOBNotesPart::FederationIdPrefix(federation_id_prefix),
OOBNotesPart::Notes(notes),
])
}
pub fn new_with_invite(notes: TieredMulti<SpendableNote>, invite: &InviteCode) -> Self {
let mut data = vec![
OOBNotesPart::FederationIdPrefix(invite.federation_id().to_prefix()),
OOBNotesPart::Notes(notes),
OOBNotesPart::Invite {
peer_apis: vec![(invite.peer(), invite.url())],
federation_id: invite.federation_id(),
},
];
if let Some(api_secret) = invite.api_secret() {
data.push(OOBNotesPart::ApiSecret(api_secret));
}
Self(data)
}
pub fn federation_id_prefix(&self) -> FederationIdPrefix {
self.0
.iter()
.find_map(|data| match data {
OOBNotesPart::FederationIdPrefix(prefix) => Some(*prefix),
OOBNotesPart::Invite { federation_id, .. } => Some(federation_id.to_prefix()),
_ => None,
})
.expect("Invariant violated: OOBNotes does not contain a FederationIdPrefix")
}
pub fn notes(&self) -> &TieredMulti<SpendableNote> {
self.0
.iter()
.find_map(|data| match data {
OOBNotesPart::Notes(notes) => Some(notes),
_ => None,
})
.expect("Invariant violated: OOBNotes does not contain any notes")
}
pub fn notes_json(&self) -> Result<serde_json::Value, serde_json::Error> {
let mut notes_map = serde_json::Map::new();
for notes in &self.0 {
match notes {
OOBNotesPart::Notes(notes) => {
let notes_json = serde_json::to_value(notes)?;
notes_map.insert("notes".to_string(), notes_json);
}
OOBNotesPart::FederationIdPrefix(prefix) => {
notes_map.insert(
"federation_id_prefix".to_string(),
serde_json::to_value(prefix.to_string())?,
);
}
OOBNotesPart::Invite {
peer_apis,
federation_id,
} => {
let (peer_id, api) = peer_apis
.first()
.cloned()
.expect("Decoding makes sure peer_apis isn't empty");
notes_map.insert(
"invite".to_string(),
serde_json::to_value(InviteCode::new(
api,
peer_id,
*federation_id,
self.api_secret(),
))?,
);
}
OOBNotesPart::ApiSecret(_) => { }
OOBNotesPart::Default { variant, bytes } => {
notes_map.insert(
format!("default_{variant}"),
serde_json::to_value(bytes.encode_hex::<String>())?,
);
}
}
}
Ok(serde_json::Value::Object(notes_map))
}
pub fn federation_invite(&self) -> Option<InviteCode> {
self.0.iter().find_map(|data| {
let OOBNotesPart::Invite {
peer_apis,
federation_id,
} = data
else {
return None;
};
let (peer_id, api) = peer_apis
.first()
.cloned()
.expect("Decoding makes sure peer_apis isn't empty");
Some(InviteCode::new(
api,
peer_id,
*federation_id,
self.api_secret(),
))
})
}
fn api_secret(&self) -> Option<String> {
self.0.iter().find_map(|data| {
let OOBNotesPart::ApiSecret(api_secret) = data else {
return None;
};
Some(api_secret.clone())
})
}
}
impl Decodable for OOBNotes {
fn consensus_decode<R: Read>(
r: &mut R,
_modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
let inner = Vec::<OOBNotesPart>::consensus_decode(r, &ModuleDecoderRegistry::default())?;
if !inner
.iter()
.any(|data| matches!(data, OOBNotesPart::Notes(_)))
{
return Err(DecodeError::from_str(
"No e-cash notes were found in OOBNotes data",
));
}
let maybe_federation_id_prefix = inner.iter().find_map(|data| match data {
OOBNotesPart::FederationIdPrefix(prefix) => Some(*prefix),
_ => None,
});
let maybe_invite = inner.iter().find_map(|data| match data {
OOBNotesPart::Invite {
federation_id,
peer_apis,
} => Some((federation_id, peer_apis)),
_ => None,
});
match (maybe_federation_id_prefix, maybe_invite) {
(Some(p), Some((ip, _))) => {
if p != ip.to_prefix() {
return Err(DecodeError::from_str(
"Inconsistent Federation ID provided in OOBNotes data",
));
}
}
(None, None) => {
return Err(DecodeError::from_str(
"No Federation ID provided in OOBNotes data",
));
}
_ => {}
}
if let Some((_, invite)) = maybe_invite {
if invite.is_empty() {
return Err(DecodeError::from_str("Invite didn't contain API endpoints"));
}
}
Ok(OOBNotes(inner))
}
}
const BASE64_URL_SAFE: base64::engine::GeneralPurpose = base64::engine::GeneralPurpose::new(
&base64::alphabet::URL_SAFE,
base64::engine::general_purpose::PAD,
);
impl FromStr for OOBNotes {
type Err = anyhow::Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let s: String = s.chars().filter(|&c| !c.is_whitespace()).collect();
if let Ok(notes_v2) = OOBNotesV2::decode_base64(&s) {
return notes_v2.into_v1();
}
let bytes = if let Ok(bytes) = BASE64_URL_SAFE.decode(&s) {
bytes
} else {
base64::engine::general_purpose::STANDARD.decode(&s)?
};
let oob_notes: OOBNotes = Decodable::consensus_decode(
&mut std::io::Cursor::new(bytes),
&ModuleDecoderRegistry::default(),
)?;
ensure!(!oob_notes.notes().is_empty(), "OOBNotes cannot be empty");
Ok(oob_notes)
}
}
impl Display for OOBNotes {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
let mut bytes = Vec::new();
Encodable::consensus_encode(self, &mut bytes).expect("encodes correctly");
if f.alternate() {
f.write_str(&BASE64_URL_SAFE.encode(&bytes))
} else {
f.write_str(&base64::engine::general_purpose::STANDARD.encode(&bytes))
}
}
}
impl Serialize for OOBNotes {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
serializer.serialize_str(&self.to_string())
}
}
impl<'de> Deserialize<'de> for OOBNotes {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
let s = String::deserialize(deserializer)?;
FromStr::from_str(&s).map_err(serde::de::Error::custom)
}
}
impl OOBNotes {
pub fn total_amount(&self) -> Amount {
self.notes().total_amount()
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize, Encodable, Decodable)]
pub struct OOBNoteV2 {
pub amount: Amount,
pub sig: Signature,
pub key: KeyPair,
}
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize, Encodable, Decodable)]
pub struct OOBNotesV2 {
pub mint: InviteCodeV2,
pub notes: Vec<OOBNoteV2>,
pub memo: String,
}
impl OOBNotesV2 {
pub fn into_v1(self) -> anyhow::Result<OOBNotes> {
let notes: TieredMulti<SpendableNote> = self
.notes
.iter()
.map(|n| {
(
n.amount,
SpendableNote {
signature: n.sig,
spend_key: n.key,
},
)
})
.collect();
Ok(OOBNotes::new_with_invite(notes, &self.mint.into_v1()?))
}
pub fn total_amount(&self) -> Amount {
self.notes.iter().map(|note| note.amount).sum()
}
pub fn encode_base64(&self) -> String {
let json = &serde_json::to_string(self).expect("Encoding to JSON cannot fail");
let base_64 = base64_url::encode(json);
format!("fedimintA{base_64}")
}
pub fn decode_base64(s: &str) -> anyhow::Result<Self> {
ensure!(s.starts_with("fedimintA"), "Invalid Prefix");
let notes: Self = serde_json::from_slice(&base64_url::decode(&s[9..])?)?;
ensure!(!notes.mint.peers.is_empty(), "Invite code has no peer");
Ok(notes)
}
}
#[derive(Debug, Clone, Eq, PartialEq, Serialize, Deserialize)]
pub enum ReissueExternalNotesState {
Created,
Issuing,
Done,
Failed(String),
}
#[derive(Debug, Clone, Eq, PartialEq, Serialize, Deserialize)]
pub enum SpendOOBState {
Created,
UserCanceledProcessing,
UserCanceledSuccess,
UserCanceledFailure,
Success,
Refunded,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct MintOperationMeta {
pub variant: MintOperationMetaVariant,
pub amount: Amount,
pub extra_meta: serde_json::Value,
}
#[derive(Debug, Clone, Serialize, Deserialize, Eq, PartialEq)]
#[serde(rename_all = "snake_case")]
pub enum MintOperationMetaVariant {
Reissuance {
#[serde(skip_serializing, default, rename = "out_point")]
legacy_out_point: Option<OutPoint>,
#[serde(default)]
txid: Option<TransactionId>,
#[serde(default)]
out_point_indices: Vec<u64>,
},
SpendOOB {
requested_amount: Amount,
oob_notes: OOBNotes,
},
}
#[derive(Debug, Clone)]
pub struct MintClientInit;
impl ModuleInit for MintClientInit {
type Common = MintCommonInit;
const DATABASE_VERSION: DatabaseVersion = DatabaseVersion(0);
async fn dump_database(
&self,
dbtx: &mut DatabaseTransaction<'_>,
prefix_names: Vec<String>,
) -> Box<dyn Iterator<Item = (String, Box<dyn erased_serde::Serialize + Send>)> + '_> {
let mut mint_client_items: BTreeMap<String, Box<dyn erased_serde::Serialize + Send>> =
BTreeMap::new();
let filtered_prefixes = DbKeyPrefix::iter().filter(|f| {
prefix_names.is_empty() || prefix_names.contains(&f.to_string().to_lowercase())
});
for table in filtered_prefixes {
match table {
DbKeyPrefix::Note => {
push_db_pair_items!(
dbtx,
NoteKeyPrefix,
NoteKey,
SpendableNoteUndecoded,
mint_client_items,
"Notes"
);
}
DbKeyPrefix::NextECashNoteIndex => {
push_db_pair_items!(
dbtx,
NextECashNoteIndexKeyPrefix,
NextECashNoteIndexKey,
u64,
mint_client_items,
"NextECashNoteIndex"
);
}
DbKeyPrefix::CancelledOOBSpend => {
push_db_pair_items!(
dbtx,
CancelledOOBSpendKeyPrefix,
CancelledOOBSpendKey,
(),
mint_client_items,
"CancelledOOBSpendKey"
);
}
DbKeyPrefix::RecoveryFinalized => {
if let Some(val) = dbtx.get_value(&RecoveryFinalizedKey).await {
mint_client_items.insert("RecoveryFinalized".to_string(), Box::new(val));
}
}
DbKeyPrefix::RecoveryState => {}
}
}
Box::new(mint_client_items.into_iter())
}
}
#[apply(async_trait_maybe_send!)]
impl ClientModuleInit for MintClientInit {
type Module = MintClientModule;
fn supported_api_versions(&self) -> MultiApiVersion {
MultiApiVersion::try_from_iter([ApiVersion { major: 0, minor: 0 }])
.expect("no version conflicts")
}
async fn init(&self, args: &ClientModuleInitArgs<Self>) -> anyhow::Result<Self::Module> {
Ok(MintClientModule {
federation_id: *args.federation_id(),
cfg: args.cfg().clone(),
secret: args.module_root_secret().clone(),
secp: Secp256k1::new(),
notifier: args.notifier().clone(),
client_ctx: args.context(),
})
}
async fn recover(
&self,
args: &ClientModuleRecoverArgs<Self>,
snapshot: Option<&<Self::Module as ClientModule>::Backup>,
) -> anyhow::Result<()> {
args.recover_from_history::<MintRecovery>(self, snapshot)
.await
}
}
#[derive(Debug)]
pub struct MintClientModule {
federation_id: FederationId,
cfg: MintClientConfig,
secret: DerivableSecret,
secp: Secp256k1<All>,
notifier: ModuleNotifier<MintClientStateMachines>,
client_ctx: ClientContext<Self>,
}
#[derive(Debug, Clone)]
pub struct MintClientContext {
pub mint_decoder: Decoder,
pub tbs_pks: Tiered<AggregatePublicKey>,
pub peer_tbs_pks: BTreeMap<PeerId, Tiered<tbs::PublicKeyShare>>,
pub secret: DerivableSecret,
pub module_db: Database,
}
impl MintClientContext {
fn await_cancel_oob_payment(&self, operation_id: OperationId) -> BoxFuture<'static, ()> {
let db = self.module_db.clone();
Box::pin(async move {
db.wait_key_exists(&CancelledOOBSpendKey(operation_id))
.await;
})
}
}
impl Context for MintClientContext {}
#[apply(async_trait_maybe_send!)]
impl ClientModule for MintClientModule {
type Init = MintClientInit;
type Common = MintModuleTypes;
type Backup = EcashBackup;
type ModuleStateMachineContext = MintClientContext;
type States = MintClientStateMachines;
fn context(&self) -> Self::ModuleStateMachineContext {
MintClientContext {
mint_decoder: self.decoder(),
tbs_pks: self.cfg.tbs_pks.clone(),
peer_tbs_pks: self.cfg.peer_tbs_pks.clone(),
secret: self.secret.clone(),
module_db: self.client_ctx.module_db().clone(),
}
}
fn input_fee(&self, _input: &<Self::Common as ModuleCommon>::Input) -> Option<Amount> {
Some(self.cfg.fee_consensus.note_spend_abs)
}
fn output_fee(&self, _output: &<Self::Common as ModuleCommon>::Output) -> Option<Amount> {
Some(self.cfg.fee_consensus.note_issuance_abs)
}
async fn handle_cli_command(
&self,
args: &[std::ffi::OsString],
) -> anyhow::Result<serde_json::Value> {
cli::handle_cli_command(self, args).await
}
fn supports_backup(&self) -> bool {
true
}
async fn backup(&self) -> anyhow::Result<EcashBackup> {
self.client_ctx
.module_autocommit(
|dbtx_ctx, _| {
Box::pin(async { self.prepare_plaintext_ecash_backup(dbtx_ctx).await })
},
None,
)
.await
.map_err(|e| match e {
AutocommitError::ClosureError { error, .. } => error,
AutocommitError::CommitFailed { last_error, .. } => {
anyhow!("Commit to DB failed: {last_error}")
}
})
}
fn supports_being_primary(&self) -> bool {
true
}
async fn create_final_inputs_and_outputs(
&self,
dbtx: &mut DatabaseTransaction<'_>,
operation_id: OperationId,
mut input_amount: Amount,
mut output_amount: Amount,
) -> anyhow::Result<(
Vec<ClientInput<MintInput, MintClientStateMachines>>,
Vec<ClientOutput<MintOutput, MintClientStateMachines>>,
)> {
let consolidation_inputs = self.consolidate_notes(dbtx, operation_id).await?;
input_amount += consolidation_inputs.iter().map(|input| input.amount).sum();
output_amount += self
.cfg
.fee_consensus
.note_spend_abs
.mul_u64(consolidation_inputs.len() as u64);
let additional_inputs = self
.create_sufficient_input(
dbtx,
operation_id,
output_amount.saturating_sub(input_amount),
)
.await?;
input_amount += additional_inputs.iter().map(|input| input.amount).sum();
output_amount += self
.cfg
.fee_consensus
.note_spend_abs
.mul_u64(additional_inputs.len() as u64);
let outputs = self
.create_exact_output(dbtx, operation_id, 2, input_amount - output_amount)
.await;
Ok(([consolidation_inputs, additional_inputs].concat(), outputs))
}
async fn await_primary_module_output(
&self,
operation_id: OperationId,
out_point: OutPoint,
) -> anyhow::Result<Amount> {
self.await_output_finalized(operation_id, out_point).await
}
async fn get_balance(&self, dbtx: &mut DatabaseTransaction<'_>) -> Amount {
self.get_notes_tier_counts(dbtx).await.total_amount()
}
async fn subscribe_balance_changes(&self) -> BoxStream<'static, ()> {
Box::pin(
self.notifier
.subscribe_all_operations()
.filter_map(|state| async move {
match state {
MintClientStateMachines::Output(MintOutputStateMachine {
state: MintOutputStates::Succeeded(_),
..
}) => Some(()),
MintClientStateMachines::Input(MintInputStateMachine {
state: MintInputStates::Created(_),
..
}) => Some(()),
MintClientStateMachines::OOB(MintOOBStateMachine {
state: MintOOBStates::Created(_),
..
}) => Some(()),
_ => None,
}
}),
)
}
async fn leave(&self, dbtx: &mut DatabaseTransaction<'_>) -> anyhow::Result<()> {
let balance = ClientModule::get_balance(self, dbtx).await;
if Amount::from_sats(0) < balance {
bail!("Outstanding balance: {balance}");
}
if !self.client_ctx.get_own_active_states().await.is_empty() {
bail!("Pending operations")
}
Ok(())
}
async fn handle_rpc(
&self,
method: String,
request: serde_json::Value,
) -> BoxStream<'_, anyhow::Result<serde_json::Value>> {
Box::pin(try_stream! {
match method.as_str() {
"reissue_external_notes" => {
let req: ReissueExternalNotesRequest = serde_json::from_value(request)?;
let result = self.reissue_external_notes(req.oob_notes, req.extra_meta).await?;
yield serde_json::to_value(result)?;
}
"subscribe_reissue_external_notes" => {
let req: SubscribeReissueExternalNotesRequest = serde_json::from_value(request)?;
let stream = self.subscribe_reissue_external_notes(req.operation_id).await?;
for await state in stream.into_stream() {
yield serde_json::to_value(state)?;
}
}
"spend_notes" => {
let req: SpendNotesRequest = serde_json::from_value(request)?;
let result = self.spend_notes(req.min_amount, req.try_cancel_after, req.include_invite, req.extra_meta).await?;
yield serde_json::to_value(result)?;
}
"validate_notes" => {
let req: ValidateNotesRequest = serde_json::from_value(request)?;
let result = self.validate_notes(&req.oob_notes)?;
yield serde_json::to_value(result)?;
}
"try_cancel_spend_notes" => {
let req: TryCancelSpendNotesRequest = serde_json::from_value(request)?;
let result = self.try_cancel_spend_notes(req.operation_id).await;
yield serde_json::to_value(result)?;
}
"subscribe_spend_notes" => {
let req: SubscribeSpendNotesRequest = serde_json::from_value(request)?;
let stream = self.subscribe_spend_notes(req.operation_id).await?;
for await state in stream.into_stream() {
yield serde_json::to_value(state)?;
}
}
"await_spend_oob_refund" => {
let req: AwaitSpendOobRefundRequest = serde_json::from_value(request)?;
let value = self.await_spend_oob_refund(req.operation_id).await;
yield serde_json::to_value(value)?;
}
_ => {
Err(anyhow::format_err!("Unknown method: {}", method))?;
unreachable!()
},
}
})
}
}
#[derive(Deserialize)]
struct ReissueExternalNotesRequest {
oob_notes: OOBNotes,
extra_meta: serde_json::Value,
}
#[derive(Deserialize)]
struct SubscribeReissueExternalNotesRequest {
operation_id: OperationId,
}
#[derive(Deserialize)]
struct SpendNotesRequest {
min_amount: Amount,
try_cancel_after: Duration,
include_invite: bool,
extra_meta: serde_json::Value,
}
#[derive(Deserialize)]
struct ValidateNotesRequest {
oob_notes: OOBNotes,
}
#[derive(Deserialize)]
struct TryCancelSpendNotesRequest {
operation_id: OperationId,
}
#[derive(Deserialize)]
struct SubscribeSpendNotesRequest {
operation_id: OperationId,
}
#[derive(Deserialize)]
struct AwaitSpendOobRefundRequest {
operation_id: OperationId,
}
#[derive(thiserror::Error, Debug, Clone)]
pub enum ReissueExternalNotesError {
#[error("Federation ID does not match")]
WrongFederationId,
#[error("We already reissued these notes")]
AlreadyReissued,
}
impl MintClientModule {
async fn create_sufficient_input(
&self,
dbtx: &mut DatabaseTransaction<'_>,
operation_id: OperationId,
min_amount: Amount,
) -> anyhow::Result<Vec<ClientInput<MintInput, MintClientStateMachines>>> {
if min_amount == Amount::ZERO {
return Ok(Vec::new());
}
let selected_notes = Self::select_notes(
dbtx,
&SelectNotesWithAtleastAmount,
min_amount,
self.cfg.fee_consensus.note_spend_abs,
)
.await?;
for (amount, note) in selected_notes.iter_items() {
debug!(target: LOG_CLIENT_MODULE_MINT, %amount, %note, "Spending note as sufficient input to fund a tx");
MintClientModule::delete_spendable_note(dbtx, amount, note).await;
}
let inputs = self.create_input_from_notes(operation_id, selected_notes)?;
assert!(!inputs.is_empty());
Ok(inputs)
}
pub async fn get_notes_tier_counts(&self, dbtx: &mut DatabaseTransaction<'_>) -> TieredCounts {
dbtx.find_by_prefix(&NoteKeyPrefix)
.await
.fold(
TieredCounts::default(),
|mut acc, (key, _note)| async move {
acc.inc(key.amount, 1);
acc
},
)
.await
}
pub async fn get_available_notes_by_tier_counts(
&self,
dbtx: &mut DatabaseTransaction<'_>,
counts: TieredCounts,
) -> (TieredMulti<SpendableNoteUndecoded>, TieredCounts) {
dbtx.find_by_prefix(&NoteKeyPrefix)
.await
.fold(
(TieredMulti::<SpendableNoteUndecoded>::default(), counts),
|(mut notes, mut counts), (key, note)| async move {
let amount = key.amount;
if 0 < counts.get(amount) {
counts.dec(amount);
notes.push(amount, note);
}
(notes, counts)
},
)
.await
}
pub async fn create_exact_output(
&self,
dbtx: &mut DatabaseTransaction<'_>,
operation_id: OperationId,
notes_per_denomination: u16,
exact_amount: Amount,
) -> Vec<ClientOutput<MintOutput, MintClientStateMachines>> {
if exact_amount == Amount::ZERO {
return Vec::new();
}
let denominations = represent_amount(
exact_amount,
&self.get_notes_tier_counts(dbtx).await,
&self.cfg.tbs_pks,
notes_per_denomination,
);
let mut outputs = Vec::new();
for (amount, num) in denominations.iter() {
for _ in 0..num {
let (issuance_request, blind_nonce) = self.new_ecash_note(amount, dbtx).await;
let state_generator = Arc::new(move |txid, out_idx| {
vec![MintClientStateMachines::Output(MintOutputStateMachine {
common: MintOutputCommon {
operation_id,
out_point: OutPoint { txid, out_idx },
},
state: MintOutputStates::Created(MintOutputStatesCreated {
amount,
issuance_request,
}),
})]
});
debug!(
%amount,
"Generated issuance request"
);
outputs.push(ClientOutput {
output: MintOutput::new_v0(amount, blind_nonce),
amount,
state_machines: state_generator,
});
}
}
assert!(!outputs.is_empty());
outputs
}
pub async fn get_wallet_summary(&self, dbtx: &mut DatabaseTransaction<'_>) -> TieredCounts {
dbtx.find_by_prefix(&NoteKeyPrefix)
.await
.fold(
TieredCounts::default(),
|mut acc, (key, _note)| async move {
acc.inc(key.amount, 1);
acc
},
)
.await
}
pub async fn await_output_finalized(
&self,
operation_id: OperationId,
out_point: OutPoint,
) -> anyhow::Result<Amount> {
let stream = self
.notifier
.subscribe(operation_id)
.await
.filter_map(|state| async {
let MintClientStateMachines::Output(state) = state else {
return None;
};
if state.common.out_point != out_point {
return None;
}
match state.state {
MintOutputStates::Succeeded(succeeded) => Some(Ok(succeeded.amount)),
MintOutputStates::Aborted(_) => Some(Err(anyhow!("Transaction was rejected"))),
MintOutputStates::Failed(failed) => Some(Err(anyhow!(
"Failed to finalize transaction: {}",
failed.error
))),
MintOutputStates::Created(_) => None,
}
});
pin_mut!(stream);
stream.next_or_pending().await
}
pub async fn consolidate_notes(
&self,
dbtx: &mut DatabaseTransaction<'_>,
operation_id: OperationId,
) -> anyhow::Result<Vec<ClientInput<MintInput, MintClientStateMachines>>> {
const MAX_NOTES_PER_TIER_TRIGGER: usize = 8;
const MIN_NOTES_PER_TIER: usize = 4;
const MAX_NOTES_TO_CONSOLIDATE_IN_TX: usize = 20;
#[allow(clippy::assertions_on_constants)]
{
assert!(MIN_NOTES_PER_TIER <= MAX_NOTES_PER_TIER_TRIGGER);
}
let counts = self.get_notes_tier_counts(dbtx).await;
let should_consolidate = counts
.iter()
.any(|(_, count)| MAX_NOTES_PER_TIER_TRIGGER < count);
if !should_consolidate {
return Ok(vec![]);
}
let mut max_count = MAX_NOTES_TO_CONSOLIDATE_IN_TX;
let excessive_counts: TieredCounts = counts
.iter()
.map(|(amount, count)| {
let take = (count.saturating_sub(MIN_NOTES_PER_TIER)).min(max_count);
max_count -= take;
(amount, take)
})
.collect();
let (selected_notes, unavailable) = self
.get_available_notes_by_tier_counts(dbtx, excessive_counts)
.await;
debug_assert!(
unavailable.is_empty(),
"Can't have unavailable notes on a subset of all notes: {unavailable:?}"
);
if !selected_notes.is_empty() {
debug!(target: LOG_CLIENT_MODULE_MINT, note_num=selected_notes.count_items(), denominations_msats=?selected_notes.iter_items().map(|(amount, _)| amount.msats).collect::<Vec<_>>(), "Will consolidate excessive notes");
}
let mut selected_notes_decoded = vec![];
for (amount, note) in selected_notes.iter_items() {
let spendable_note_decoded = note.decode()?;
debug!(target: LOG_CLIENT_MODULE_MINT, %amount, %note, "Consolidating note");
Self::delete_spendable_note(dbtx, amount, &spendable_note_decoded).await;
selected_notes_decoded.push((amount, spendable_note_decoded));
}
self.create_input_from_notes(operation_id, selected_notes_decoded.into_iter().collect())
}
pub fn create_input_from_notes(
&self,
operation_id: OperationId,
notes: TieredMulti<SpendableNote>,
) -> anyhow::Result<Vec<ClientInput<MintInput, MintClientStateMachines>>> {
let mut inputs = Vec::new();
for (amount, spendable_note) in notes {
let key = self
.cfg
.tbs_pks
.get(amount)
.ok_or(anyhow!("Invalid amount tier: {amount}"))?;
let note = spendable_note.note();
if !note.verify(*key) {
bail!("Invalid note");
}
let sm_gen = Arc::new(move |txid, input_idx| {
vec![MintClientStateMachines::Input(MintInputStateMachine {
common: MintInputCommon {
operation_id,
txid,
input_idx,
},
state: MintInputStates::Created(MintInputStateCreated {
amount,
spendable_note,
}),
})]
});
inputs.push(ClientInput {
input: MintInput::new_v0(amount, note),
keys: vec![spendable_note.spend_key],
amount,
state_machines: sm_gen,
});
}
Ok(inputs)
}
async fn spend_notes_oob(
&self,
dbtx: &mut DatabaseTransaction<'_>,
notes_selector: &impl NotesSelector,
amount: Amount,
try_cancel_after: Duration,
) -> anyhow::Result<(
OperationId,
Vec<MintClientStateMachines>,
TieredMulti<SpendableNote>,
)> {
ensure!(
amount > Amount::ZERO,
"zero-amount out-of-band spends are not supported"
);
let selected_notes = Self::select_notes(dbtx, notes_selector, amount, Amount::ZERO).await?;
let operation_id = spendable_notes_to_operation_id(&selected_notes);
for (amount, note) in selected_notes.iter_items() {
debug!(target: LOG_CLIENT_MODULE_MINT, %amount, %note, "Spending note as oob");
MintClientModule::delete_spendable_note(dbtx, amount, note).await;
}
let mut state_machines = Vec::new();
for (amount, spendable_note) in selected_notes.clone() {
state_machines.push(MintClientStateMachines::OOB(MintOOBStateMachine {
operation_id,
state: MintOOBStates::Created(MintOOBStatesCreated {
amount,
spendable_note,
timeout: fedimint_core::time::now() + try_cancel_after,
}),
}));
}
Ok((operation_id, state_machines, selected_notes))
}
pub async fn await_spend_oob_refund(&self, operation_id: OperationId) -> SpendOOBRefund {
Box::pin(
self.notifier
.subscribe(operation_id)
.await
.filter_map(|state| async {
let MintClientStateMachines::OOB(state) = state else {
return None;
};
match state.state {
MintOOBStates::TimeoutRefund(refund) => Some(SpendOOBRefund {
user_triggered: false,
transaction_id: refund.refund_txid,
}),
MintOOBStates::UserRefund(refund) => Some(SpendOOBRefund {
user_triggered: true,
transaction_id: refund.refund_txid,
}),
MintOOBStates::Created(_) => None,
}
}),
)
.next_or_pending()
.await
}
async fn select_notes(
dbtx: &mut DatabaseTransaction<'_>,
notes_selector: &impl NotesSelector,
requested_amount: Amount,
fee_per_note_input: Amount,
) -> anyhow::Result<TieredMulti<SpendableNote>> {
let note_stream = dbtx
.find_by_prefix_sorted_descending(&NoteKeyPrefix)
.await
.map(|(key, note)| (key.amount, note));
notes_selector
.select_notes(note_stream, requested_amount, fee_per_note_input)
.await?
.into_iter()
.map(|(amt, snote)| Ok((amt, snote.decode()?)))
.collect::<anyhow::Result<TieredMulti<_>>>()
}
async fn get_all_spendable_notes(
dbtx: &mut DatabaseTransaction<'_>,
) -> TieredMulti<SpendableNoteUndecoded> {
(dbtx
.find_by_prefix(&NoteKeyPrefix)
.await
.map(|(key, note)| (key.amount, note))
.collect::<Vec<_>>()
.await)
.into_iter()
.collect()
}
async fn get_next_note_index(
&self,
dbtx: &mut DatabaseTransaction<'_>,
amount: Amount,
) -> NoteIndex {
NoteIndex(
dbtx.get_value(&NextECashNoteIndexKey(amount))
.await
.unwrap_or(0),
)
}
pub fn new_note_secret_static(
secret: &DerivableSecret,
amount: Amount,
note_idx: NoteIndex,
) -> DerivableSecret {
assert_eq!(secret.level(), 2);
debug!(?secret, %amount, %note_idx, "Deriving new mint note");
secret
.child_key(MINT_E_CASH_TYPE_CHILD_ID) .child_key(ChildId(note_idx.as_u64()))
.child_key(ChildId(amount.msats))
}
async fn new_note_secret(
&self,
amount: Amount,
dbtx: &mut DatabaseTransaction<'_>,
) -> DerivableSecret {
let new_idx = self.get_next_note_index(dbtx, amount).await;
dbtx.insert_entry(&NextECashNoteIndexKey(amount), &new_idx.next().as_u64())
.await;
Self::new_note_secret_static(&self.secret, amount, new_idx)
}
pub async fn new_ecash_note(
&self,
amount: Amount,
dbtx: &mut DatabaseTransaction<'_>,
) -> (NoteIssuanceRequest, BlindNonce) {
let secret = self.new_note_secret(amount, dbtx).await;
NoteIssuanceRequest::new(&self.secp, &secret)
}
pub async fn reissue_external_notes<M: Serialize + Send>(
&self,
oob_notes: OOBNotes,
extra_meta: M,
) -> anyhow::Result<OperationId> {
let notes = oob_notes.notes().clone();
let federation_id_prefix = oob_notes.federation_id_prefix();
ensure!(
notes.total_amount() > Amount::ZERO,
"Reissuing zero-amount e-cash isn't supported"
);
if federation_id_prefix != self.federation_id.to_prefix() {
bail!(ReissueExternalNotesError::WrongFederationId);
}
let operation_id = OperationId(
notes
.consensus_hash::<sha256t::Hash<OOBReissueTag>>()
.to_byte_array(),
);
let amount = notes.total_amount();
let mint_input = self.create_input_from_notes(operation_id, notes)?;
let tx =
TransactionBuilder::new().with_inputs(self.client_ctx.map_dyn(mint_input).collect());
let extra_meta = serde_json::to_value(extra_meta)
.expect("MintClientModule::reissue_external_notes extra_meta is serializable");
let operation_meta_gen = |txid, out_points: Vec<OutPoint>| {
assert!(
out_points.iter().all(|out_point| out_point.txid == txid),
"Change outpoints didn't all have consistent transaction id."
);
MintOperationMeta {
variant: MintOperationMetaVariant::Reissuance {
legacy_out_point: None,
txid: Some(txid),
out_point_indices: out_points
.iter()
.map(|out_point| out_point.out_idx)
.collect(),
},
amount,
extra_meta: extra_meta.clone(),
}
};
self.client_ctx
.finalize_and_submit_transaction(
operation_id,
MintCommonInit::KIND.as_str(),
operation_meta_gen,
tx,
)
.await
.context(ReissueExternalNotesError::AlreadyReissued)?;
Ok(operation_id)
}
pub async fn subscribe_reissue_external_notes(
&self,
operation_id: OperationId,
) -> anyhow::Result<UpdateStreamOrOutcome<ReissueExternalNotesState>> {
let operation = self.mint_operation(operation_id).await?;
let (txid, out_points) = match operation.meta::<MintOperationMeta>().variant {
MintOperationMetaVariant::Reissuance {
legacy_out_point,
txid,
out_point_indices,
} => {
let txid = txid
.or(legacy_out_point.map(|out_point| out_point.txid))
.context("Empty reissuance not permitted, this should never happen")?;
let out_points = out_point_indices
.into_iter()
.map(|out_idx| OutPoint { txid, out_idx })
.chain(legacy_out_point)
.collect::<Vec<_>>();
(txid, out_points)
}
MintOperationMetaVariant::SpendOOB { .. } => bail!("Operation is not a reissuance"),
};
let client_ctx = self.client_ctx.clone();
Ok(operation.outcome_or_updates(&self.client_ctx.global_db(), operation_id, || {
stream! {
yield ReissueExternalNotesState::Created;
match client_ctx
.transaction_updates(operation_id)
.await
.await_tx_accepted(txid)
.await
{
Ok(()) => {
yield ReissueExternalNotesState::Issuing;
}
Err(e) => {
yield ReissueExternalNotesState::Failed(format!("Transaction not accepted {e:?}"));
return;
}
}
for out_point in out_points {
if let Err(e) = client_ctx.self_ref().await_output_finalized(operation_id, out_point).await {
yield ReissueExternalNotesState::Failed(e.to_string());
return;
}
}
yield ReissueExternalNotesState::Done;
}}
))
}
pub async fn spend_notes<M: Serialize + Send>(
&self,
min_amount: Amount,
try_cancel_after: Duration,
include_invite: bool,
extra_meta: M,
) -> anyhow::Result<(OperationId, OOBNotes)> {
self.spend_notes_with_selector(
&SelectNotesWithAtleastAmount,
min_amount,
try_cancel_after,
include_invite,
extra_meta,
)
.await
}
pub async fn spend_notes_with_selector<M: Serialize + Send>(
&self,
notes_selector: &impl NotesSelector,
requested_amount: Amount,
try_cancel_after: Duration,
include_invite: bool,
extra_meta: M,
) -> anyhow::Result<(OperationId, OOBNotes)> {
let federation_id_prefix = self.federation_id.to_prefix();
let extra_meta = serde_json::to_value(extra_meta)
.expect("MintClientModule::spend_notes extra_meta is serializable");
self.client_ctx
.module_autocommit(
|dbtx, _| {
let extra_meta = extra_meta.clone();
Box::pin(async {
let (operation_id, states, notes) = self
.spend_notes_oob(
&mut dbtx.module_dbtx(),
notes_selector,
requested_amount,
try_cancel_after,
)
.await?;
let oob_notes = if include_invite {
OOBNotes::new_with_invite(
notes,
&self.client_ctx.get_invite_code().await,
)
} else {
OOBNotes::new(federation_id_prefix, notes)
};
dbtx.add_state_machines(self.client_ctx.map_dyn(states).collect())
.await?;
dbtx.add_operation_log_entry(
operation_id,
MintCommonInit::KIND.as_str(),
MintOperationMeta {
variant: MintOperationMetaVariant::SpendOOB {
requested_amount,
oob_notes: oob_notes.clone(),
},
amount: oob_notes.total_amount(),
extra_meta,
},
)
.await;
Ok((operation_id, oob_notes))
})
},
Some(100),
)
.await
.map_err(|e| match e {
AutocommitError::ClosureError { error, .. } => error,
AutocommitError::CommitFailed { last_error, .. } => {
anyhow!("Commit to DB failed: {last_error}")
}
})
}
pub fn validate_notes(&self, oob_notes: &OOBNotes) -> anyhow::Result<Amount> {
let federation_id_prefix = oob_notes.federation_id_prefix();
let notes = oob_notes.notes().clone();
if federation_id_prefix != self.federation_id.to_prefix() {
bail!("Federation ID does not match");
}
let tbs_pks = &self.cfg.tbs_pks;
for (idx, (amt, snote)) in notes.iter_items().enumerate() {
let key = tbs_pks
.get(amt)
.ok_or_else(|| anyhow!("Note {idx} uses an invalid amount tier {amt}"))?;
let note = snote.note();
if !note.verify(*key) {
bail!("Note {idx} has an invalid federation signature");
}
let expected_nonce = Nonce(snote.spend_key.public_key());
if note.nonce != expected_nonce {
bail!("Note {idx} cannot be spent using the supplied spend key");
}
}
Ok(notes.total_amount())
}
pub async fn try_cancel_spend_notes(&self, operation_id: OperationId) {
let mut dbtx = self.client_ctx.module_db().begin_transaction().await;
dbtx.insert_entry(&CancelledOOBSpendKey(operation_id), &())
.await;
if let Err(e) = dbtx.commit_tx_result().await {
warn!("We tried to cancel the same OOB spend multiple times concurrently: {e}");
}
}
pub async fn subscribe_spend_notes(
&self,
operation_id: OperationId,
) -> anyhow::Result<UpdateStreamOrOutcome<SpendOOBState>> {
let operation = self.mint_operation(operation_id).await?;
if !matches!(
operation.meta::<MintOperationMeta>().variant,
MintOperationMetaVariant::SpendOOB { .. }
) {
bail!("Operation is not a out-of-band spend");
};
let client_ctx = self.client_ctx.clone();
let global_db = self.client_ctx.global_db();
Ok(operation.outcome_or_updates(&global_db, operation_id, || {
stream! {
yield SpendOOBState::Created;
let self_ref = client_ctx.self_ref();
let refund = self_ref
.await_spend_oob_refund(operation_id)
.await;
if refund.user_triggered {
yield SpendOOBState::UserCanceledProcessing;
match client_ctx
.transaction_updates(operation_id)
.await
.await_tx_accepted(refund.transaction_id)
.await
{
Ok(()) => {
yield SpendOOBState::UserCanceledSuccess;
},
Err(_) => {
yield SpendOOBState::UserCanceledFailure;
}
}
} else {
match client_ctx
.transaction_updates(operation_id)
.await
.await_tx_accepted(refund.transaction_id)
.await
{
Ok(()) => {
yield SpendOOBState::Refunded;
},
Err(_) => {
yield SpendOOBState::Success;
}
}
}
}
}))
}
async fn mint_operation(&self, operation_id: OperationId) -> anyhow::Result<OperationLogEntry> {
let operation = self.client_ctx.get_operation(operation_id).await?;
if operation.operation_module_kind() != MintCommonInit::KIND.as_str() {
bail!("Operation is not a mint operation");
}
Ok(operation)
}
async fn delete_spendable_note(
dbtx: &mut DatabaseTransaction<'_>,
amount: Amount,
note: &SpendableNote,
) {
dbtx.remove_entry(&NoteKey {
amount,
nonce: note.nonce(),
})
.await
.expect("Must deleted existing spendable note");
}
}
pub fn spendable_notes_to_operation_id(
spendable_selected_notes: &TieredMulti<SpendableNote>,
) -> OperationId {
OperationId(
spendable_selected_notes
.consensus_hash::<sha256t::Hash<OOBSpendTag>>()
.to_byte_array(),
)
}
#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct SpendOOBRefund {
pub user_triggered: bool,
pub transaction_id: TransactionId,
}
#[apply(async_trait_maybe_send!)]
pub trait NotesSelector<Note = SpendableNoteUndecoded>: Send + Sync {
async fn select_notes(
&self,
#[cfg(not(target_family = "wasm"))] stream: impl futures::Stream<Item = (Amount, Note)> + Send,
#[cfg(target_family = "wasm")] stream: impl futures::Stream<Item = (Amount, Note)>,
requested_amount: Amount,
fee_per_note_input: Amount,
) -> anyhow::Result<TieredMulti<Note>>;
}
pub struct SelectNotesWithAtleastAmount;
#[apply(async_trait_maybe_send!)]
impl<Note: Send> NotesSelector<Note> for SelectNotesWithAtleastAmount {
async fn select_notes(
&self,
#[cfg(not(target_family = "wasm"))] stream: impl futures::Stream<Item = (Amount, Note)> + Send,
#[cfg(target_family = "wasm")] stream: impl futures::Stream<Item = (Amount, Note)>,
requested_amount: Amount,
fee_per_note_input: Amount,
) -> anyhow::Result<TieredMulti<Note>> {
Ok(select_notes_from_stream(stream, requested_amount, fee_per_note_input).await?)
}
}
pub struct SelectNotesWithExactAmount;
#[apply(async_trait_maybe_send!)]
impl<Note: Send> NotesSelector<Note> for SelectNotesWithExactAmount {
async fn select_notes(
&self,
#[cfg(not(target_family = "wasm"))] stream: impl futures::Stream<Item = (Amount, Note)> + Send,
#[cfg(target_family = "wasm")] stream: impl futures::Stream<Item = (Amount, Note)>,
requested_amount: Amount,
note_fee: Amount,
) -> anyhow::Result<TieredMulti<Note>> {
let notes = select_notes_from_stream(stream, requested_amount, note_fee).await?;
if notes.total_amount() != requested_amount {
bail!(
"Could not select notes with exact amount. Requested amount: {}. Selected amount: {}",
requested_amount,
notes.total_amount()
);
}
Ok(notes)
}
}
async fn select_notes_from_stream<Note>(
stream: impl futures::Stream<Item = (Amount, Note)>,
requested_amount: Amount,
fee_per_note_input: Amount,
) -> Result<TieredMulti<Note>, InsufficientBalanceError> {
if requested_amount == Amount::ZERO {
return Ok(TieredMulti::default());
}
let mut stream = Box::pin(stream);
let mut selected = vec![];
let mut last_big_note_checkpoint: Option<(Amount, Note, usize)> = None;
let mut pending_amount = requested_amount;
let mut previous_amount: Option<Amount> = None; loop {
if let Some((note_amount, note)) = stream.next().await {
assert!(
previous_amount.map_or(true, |previous| previous >= note_amount),
"notes are not sorted in descending order"
);
previous_amount = Some(note_amount);
if note_amount <= fee_per_note_input {
continue;
}
match note_amount.cmp(&(pending_amount + fee_per_note_input)) {
Ordering::Less => {
pending_amount += fee_per_note_input;
pending_amount -= note_amount;
selected.push((note_amount, note));
}
Ordering::Greater => {
last_big_note_checkpoint = Some((note_amount, note, selected.len()));
}
Ordering::Equal => {
selected.push((note_amount, note));
let notes: TieredMulti<Note> = selected.into_iter().collect();
assert!(
notes.total_amount().msats
>= requested_amount.msats
+ notes.count_items() as u64 * fee_per_note_input.msats
);
return Ok(notes);
}
}
} else {
assert!(pending_amount > Amount::ZERO);
if let Some((big_note_amount, big_note, checkpoint)) = last_big_note_checkpoint {
selected.truncate(checkpoint);
selected.push((big_note_amount, big_note));
let notes: TieredMulti<Note> = selected.into_iter().collect();
assert!(
notes.total_amount().msats
>= requested_amount.msats
+ notes.count_items() as u64 * fee_per_note_input.msats
);
return Ok(notes);
}
let total_amount = requested_amount - pending_amount;
return Err(InsufficientBalanceError {
requested_amount,
total_amount,
});
}
}
}
#[derive(Debug, Clone, Error)]
pub struct InsufficientBalanceError {
pub requested_amount: Amount,
pub total_amount: Amount,
}
impl std::fmt::Display for InsufficientBalanceError {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
write!(
f,
"Insufficient balance: requested {} but only {} available",
self.requested_amount, self.total_amount
)
}
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Decodable, Encodable)]
enum MintRestoreStates {
#[encodable_default]
Default { variant: u64, bytes: Vec<u8> },
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Decodable, Encodable)]
pub struct MintRestoreStateMachine {
operation_id: OperationId,
state: MintRestoreStates,
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Decodable, Encodable)]
pub enum MintClientStateMachines {
Output(MintOutputStateMachine),
Input(MintInputStateMachine),
OOB(MintOOBStateMachine),
Restore(MintRestoreStateMachine),
}
impl IntoDynInstance for MintClientStateMachines {
type DynType = DynState;
fn into_dyn(self, instance_id: ModuleInstanceId) -> Self::DynType {
DynState::from_typed(instance_id, self)
}
}
impl State for MintClientStateMachines {
type ModuleContext = MintClientContext;
fn transitions(
&self,
context: &Self::ModuleContext,
global_context: &DynGlobalClientContext,
) -> Vec<StateTransition<Self>> {
match self {
MintClientStateMachines::Output(issuance_state) => {
sm_enum_variant_translation!(
issuance_state.transitions(context, global_context),
MintClientStateMachines::Output
)
}
MintClientStateMachines::Input(redemption_state) => {
sm_enum_variant_translation!(
redemption_state.transitions(context, global_context),
MintClientStateMachines::Input
)
}
MintClientStateMachines::OOB(oob_state) => {
sm_enum_variant_translation!(
oob_state.transitions(context, global_context),
MintClientStateMachines::OOB
)
}
MintClientStateMachines::Restore(_) => {
sm_enum_variant_translation!(vec![], MintClientStateMachines::Restore)
}
}
}
fn operation_id(&self) -> OperationId {
match self {
MintClientStateMachines::Output(issuance_state) => issuance_state.operation_id(),
MintClientStateMachines::Input(redemption_state) => redemption_state.operation_id(),
MintClientStateMachines::OOB(oob_state) => oob_state.operation_id(),
MintClientStateMachines::Restore(r) => r.operation_id,
}
}
}
#[derive(Clone, Copy, PartialEq, Eq, Hash, Deserialize, Serialize, Encodable, Decodable)]
pub struct SpendableNote {
pub signature: tbs::Signature,
pub spend_key: KeyPair,
}
impl fmt::Debug for SpendableNote {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
f.debug_struct("SpendableNote")
.field("nonce", &self.nonce())
.field("signature", &self.signature)
.field("spend_key", &self.spend_key)
.finish()
}
}
impl fmt::Display for SpendableNote {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
self.nonce().fmt(f)
}
}
impl SpendableNote {
fn nonce(&self) -> Nonce {
Nonce(self.spend_key.public_key())
}
fn note(&self) -> Note {
Note {
nonce: self.nonce(),
signature: self.signature,
}
}
pub fn to_undecoded(&self) -> SpendableNoteUndecoded {
SpendableNoteUndecoded {
signature: self
.signature
.consensus_encode_to_vec()
.try_into()
.expect("Encoded size always correct"),
spend_key: self.spend_key,
}
}
}
#[derive(Clone, Copy, PartialEq, Eq, Hash, Encodable, Decodable, Serialize)]
pub struct SpendableNoteUndecoded {
#[serde(serialize_with = "serdect::array::serialize_hex_lower_or_bin")]
pub signature: [u8; 48],
pub spend_key: KeyPair,
}
impl fmt::Display for SpendableNoteUndecoded {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
self.nonce().fmt(f)
}
}
impl fmt::Debug for SpendableNoteUndecoded {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
f.debug_struct("SpendableNote")
.field("nonce", &self.nonce())
.field("signature", &"[raw]")
.field("spend_key", &self.spend_key)
.finish()
}
}
impl SpendableNoteUndecoded {
fn nonce(&self) -> Nonce {
Nonce(self.spend_key.public_key())
}
pub fn decode(self) -> anyhow::Result<SpendableNote> {
Ok(SpendableNote {
signature: Decodable::consensus_decode_from_finite_reader(
&mut self.signature.as_slice(),
&Default::default(),
)?,
spend_key: self.spend_key,
})
}
}
#[derive(
Copy,
Clone,
Debug,
Serialize,
Deserialize,
PartialEq,
Eq,
Encodable,
Decodable,
Default,
PartialOrd,
Ord,
)]
pub struct NoteIndex(u64);
impl NoteIndex {
pub fn next(self) -> Self {
Self(self.0 + 1)
}
pub fn as_u64(self) -> u64 {
self.0
}
#[allow(unused)]
pub fn from_u64(v: u64) -> Self {
Self(v)
}
pub fn advance(&mut self) {
*self = self.next();
}
}
impl std::fmt::Display for NoteIndex {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.0.fmt(f)
}
}
struct OOBSpendTag;
impl sha256t::Tag for OOBSpendTag {
fn engine() -> sha256::HashEngine {
let mut engine = sha256::HashEngine::default();
engine.input(b"oob-spend");
engine
}
}
struct OOBReissueTag;
impl sha256t::Tag for OOBReissueTag {
fn engine() -> sha256::HashEngine {
let mut engine = sha256::HashEngine::default();
engine.input(b"oob-reissue");
engine
}
}
pub fn represent_amount<K>(
amount: Amount,
current_denominations: &TieredCounts,
tiers: &Tiered<K>,
denomination_sets: u16,
) -> TieredCounts {
let mut remaining_amount = amount;
let mut denominations = TieredCounts::default();
for tier in tiers.tiers() {
let notes = current_denominations.get(*tier);
let missing_notes = u64::from(denomination_sets).saturating_sub(notes as u64);
let possible_notes = remaining_amount / *tier;
let add_notes = min(possible_notes, missing_notes);
denominations.inc(*tier, add_notes as usize);
remaining_amount -= *tier * add_notes;
}
for tier in tiers.tiers().rev() {
let res = remaining_amount / *tier;
remaining_amount %= *tier;
denominations.inc(*tier, res as usize);
}
let represented: u64 = denominations
.iter()
.map(|(k, v)| k.msats * (v as u64))
.sum();
assert_eq!(represented, amount.msats);
denominations
}
#[cfg(test)]
mod tests {
use std::collections::BTreeMap;
use std::fmt::Display;
use std::iter;
use std::str::FromStr;
use bitcoin_hashes::Hash;
use fedimint_core::config::FederationId;
use fedimint_core::encoding::Decodable;
use fedimint_core::invite_code::{InviteCode, InviteCodeV2};
use fedimint_core::secp256k1::rand::rngs::OsRng;
use fedimint_core::secp256k1::SecretKey;
use fedimint_core::util::SafeUrl;
use fedimint_core::{
secp256k1, Amount, OutPoint, PeerId, Tiered, TieredCounts, TieredMulti, TransactionId,
};
use itertools::Itertools;
use serde_json::json;
use tbs::Signature;
use crate::{
represent_amount, select_notes_from_stream, MintOperationMetaVariant, OOBNoteV2, OOBNotes,
OOBNotesPart, OOBNotesV2, SpendableNote, SpendableNoteUndecoded,
};
#[test]
fn represent_amount_targets_denomination_sets() {
fn tiers(tiers: Vec<u64>) -> Tiered<()> {
tiers
.into_iter()
.map(|tier| (Amount::from_sats(tier), ()))
.collect()
}
fn denominations(denominations: Vec<(Amount, usize)>) -> TieredCounts {
TieredCounts::from_iter(denominations)
}
let starting = notes(vec![
(Amount::from_sats(1), 1),
(Amount::from_sats(2), 3),
(Amount::from_sats(3), 2),
])
.summary();
let tiers = tiers(vec![1, 2, 3, 4]);
assert_eq!(
represent_amount(Amount::from_sats(6), &starting, &tiers, 3),
denominations(vec![(Amount::from_sats(1), 3), (Amount::from_sats(3), 1),])
);
assert_eq!(
represent_amount(Amount::from_sats(6), &starting, &tiers, 2),
denominations(vec![(Amount::from_sats(1), 2), (Amount::from_sats(4), 1)])
);
}
#[test_log::test(tokio::test)]
async fn select_notes_avg_test() {
let max_amount = Amount::from_sats(1_000_000);
let tiers = Tiered::gen_denominations(2, max_amount);
let tiered = represent_amount::<()>(max_amount, &Default::default(), &tiers, 3);
let mut total_notes = 0;
for multiplier in 1..100 {
let stream = reverse_sorted_note_stream(tiered.iter().collect());
let select = select_notes_from_stream(
stream,
Amount::from_sats(multiplier * 1000),
Amount::ZERO,
)
.await;
total_notes += select.unwrap().into_iter_items().count();
}
assert_eq!(total_notes / 100, 10);
}
#[test_log::test(tokio::test)]
async fn select_notes_returns_exact_amount_with_minimum_notes() {
let f = || {
reverse_sorted_note_stream(vec![
(Amount::from_sats(1), 10),
(Amount::from_sats(5), 10),
(Amount::from_sats(20), 10),
])
};
assert_eq!(
select_notes_from_stream(f(), Amount::from_sats(7), Amount::ZERO)
.await
.unwrap(),
notes(vec![(Amount::from_sats(1), 2), (Amount::from_sats(5), 1)])
);
assert_eq!(
select_notes_from_stream(f(), Amount::from_sats(20), Amount::ZERO)
.await
.unwrap(),
notes(vec![(Amount::from_sats(20), 1)])
);
}
#[test_log::test(tokio::test)]
async fn select_notes_returns_next_smallest_amount_if_exact_change_cannot_be_made() {
let stream = reverse_sorted_note_stream(vec![
(Amount::from_sats(1), 1),
(Amount::from_sats(5), 5),
(Amount::from_sats(20), 5),
]);
assert_eq!(
select_notes_from_stream(stream, Amount::from_sats(7), Amount::ZERO)
.await
.unwrap(),
notes(vec![(Amount::from_sats(5), 2)])
);
}
#[test_log::test(tokio::test)]
async fn select_notes_uses_big_note_if_small_amounts_are_not_sufficient() {
let stream = reverse_sorted_note_stream(vec![
(Amount::from_sats(1), 3),
(Amount::from_sats(5), 3),
(Amount::from_sats(20), 2),
]);
assert_eq!(
select_notes_from_stream(stream, Amount::from_sats(39), Amount::ZERO)
.await
.unwrap(),
notes(vec![(Amount::from_sats(20), 2)])
);
}
#[test_log::test(tokio::test)]
async fn select_notes_returns_error_if_amount_is_too_large() {
let stream = reverse_sorted_note_stream(vec![(Amount::from_sats(10), 1)]);
let error = select_notes_from_stream(stream, Amount::from_sats(100), Amount::ZERO)
.await
.unwrap_err();
assert_eq!(error.total_amount, Amount::from_sats(10));
}
fn reverse_sorted_note_stream(
notes: Vec<(Amount, usize)>,
) -> impl futures::Stream<Item = (Amount, String)> {
futures::stream::iter(
notes
.into_iter()
.flat_map(|(amount, number)| vec![(amount, "dummy note".into()); number])
.sorted()
.rev(),
)
}
fn notes(notes: Vec<(Amount, usize)>) -> TieredMulti<String> {
notes
.into_iter()
.flat_map(|(amount, number)| vec![(amount, "dummy note".into()); number])
.collect()
}
#[test]
fn decoding_empty_oob_notes_fails() {
let empty_oob_notes = OOBNotes::new(FederationId::dummy().to_prefix(), Default::default());
let oob_notes_string = empty_oob_notes.to_string();
let res = oob_notes_string.parse::<OOBNotes>();
assert!(res.is_err(), "An empty OOB notes string should not parse");
}
fn test_roundtrip_serialize_str<T, F>(data: T, assertions: F)
where
T: FromStr + Display,
<T as FromStr>::Err: std::fmt::Debug,
F: Fn(T),
{
let data_str = data.to_string();
assertions(data);
let data_parsed = data_str.parse().expect("Deserialization failed");
assertions(data_parsed);
}
#[test]
fn notes_encode_decode() {
let federation_id_1 =
FederationId(bitcoin_hashes::sha256::Hash::from_byte_array([0x21; 32]));
let federation_id_prefix_1 = federation_id_1.to_prefix();
let federation_id_2 =
FederationId(bitcoin_hashes::sha256::Hash::from_byte_array([0x42; 32]));
let federation_id_prefix_2 = federation_id_2.to_prefix();
let notes = vec![(
Amount::from_sats(1),
SpendableNote::consensus_decode_hex("a5dd3ebacad1bc48bd8718eed5a8da1d68f91323bef2848ac4fa2e6f8eed710f3178fd4aef047cc234e6b1127086f33cc408b39818781d9521475360de6b205f3328e490a6d99d5e2553a4553207c8bd", &Default::default()).unwrap(),
)]
.into_iter()
.collect::<TieredMulti<_>>();
let notes_no_invite = OOBNotes::new(federation_id_prefix_1, notes.clone());
test_roundtrip_serialize_str(notes_no_invite, |oob_notes| {
assert_eq!(oob_notes.notes(), ¬es);
assert_eq!(oob_notes.federation_id_prefix(), federation_id_prefix_1);
assert_eq!(oob_notes.federation_invite(), None);
});
let invite = InviteCode::new(
"wss://foo.bar".parse().unwrap(),
PeerId::from(0),
federation_id_1,
None,
);
let notes_invite = OOBNotes::new_with_invite(notes.clone(), &invite);
test_roundtrip_serialize_str(notes_invite, |oob_notes| {
assert_eq!(oob_notes.notes(), ¬es);
assert_eq!(oob_notes.federation_id_prefix(), federation_id_prefix_1);
assert_eq!(oob_notes.federation_invite(), Some(invite.clone()));
});
let notes_no_prefix = OOBNotes(vec![
OOBNotesPart::Notes(notes.clone()),
OOBNotesPart::Invite {
peer_apis: vec![(PeerId::from(0), "wss://foo.bar".parse().unwrap())],
federation_id: federation_id_1,
},
]);
test_roundtrip_serialize_str(notes_no_prefix, |oob_notes| {
assert_eq!(oob_notes.notes(), ¬es);
assert_eq!(oob_notes.federation_id_prefix(), federation_id_prefix_1);
});
let notes_inconsistent = OOBNotes(vec![
OOBNotesPart::Notes(notes),
OOBNotesPart::Invite {
peer_apis: vec![(PeerId::from(0), "wss://foo.bar".parse().unwrap())],
federation_id: federation_id_1,
},
OOBNotesPart::FederationIdPrefix(federation_id_prefix_2),
]);
let notes_inconsistent_str = notes_inconsistent.to_string();
assert!(notes_inconsistent_str.parse::<OOBNotes>().is_err());
}
#[test]
fn oob_notes_v2_encode_base64_roundtrip() {
const NUMBER_OF_NOTES: usize = 5;
let notes = OOBNotesV2 {
mint: InviteCodeV2 {
id: FederationId::dummy(),
peers: BTreeMap::from_iter([(
PeerId::from(0),
SafeUrl::parse("https://mint.com").expect("Url is valid"),
)]),
api_secret: None,
},
notes: iter::repeat(OOBNoteV2 {
amount: Amount::from_msats(1),
sig: Signature(bls12_381::G1Affine::generator()),
key: SecretKey::new(&mut OsRng).keypair(secp256k1::SECP256K1),
})
.take(NUMBER_OF_NOTES)
.collect(),
memo: "Here are your sats!".to_string(),
};
OOBNotes::from_str(¬es.encode_base64()).expect("Failed to decode to legacy OOBNotes");
let encoded = notes.encode_base64();
let decoded = OOBNotesV2::decode_base64(&encoded).unwrap();
assert_eq!(notes, decoded);
}
#[test]
fn spendable_note_undecoded_sanity() {
#[allow(clippy::single_element_loop)]
for note_hex in ["a5dd3ebacad1bc48bd8718eed5a8da1d68f91323bef2848ac4fa2e6f8eed710f3178fd4aef047cc234e6b1127086f33cc408b39818781d9521475360de6b205f3328e490a6d99d5e2553a4553207c8bd"] {
let note = SpendableNote::consensus_decode_hex(note_hex, &Default::default()).unwrap();
let note_undecoded= SpendableNoteUndecoded::consensus_decode_hex(note_hex, &Default::default()).unwrap().decode().unwrap();
assert_eq!(
note,
note_undecoded,
);
assert_eq!(
serde_json::to_string(¬e).unwrap(),
serde_json::to_string(¬e_undecoded).unwrap(),
);
}
}
#[test]
fn reissuance_meta_compatibility_02_03() {
let dummy_outpoint = OutPoint {
txid: TransactionId::all_zeros(),
out_idx: 0,
};
let old_meta_json = json!({
"reissuance": {
"out_point": dummy_outpoint
}
});
let old_meta: MintOperationMetaVariant =
serde_json::from_value(old_meta_json).expect("parsing old reissuance meta failed");
assert_eq!(
old_meta,
MintOperationMetaVariant::Reissuance {
legacy_out_point: Some(dummy_outpoint),
txid: None,
out_point_indices: vec![],
}
);
let new_meta_json = serde_json::to_value(MintOperationMetaVariant::Reissuance {
legacy_out_point: None,
txid: Some(dummy_outpoint.txid),
out_point_indices: vec![0],
})
.expect("serializing always works");
assert_eq!(
new_meta_json,
json!({
"reissuance": {
"txid": dummy_outpoint.txid,
"out_point_indices": [dummy_outpoint.out_idx],
}
})
);
}
}