1use std::collections::HashMap;
4use std::ops::{Deref, DerefMut};
5
6use async_trait::async_trait;
7use cashu::quote_id::QuoteId;
8use cashu::Amount;
9
10use super::{DbTransactionFinalizer, Error};
11use crate::mint::{
12 self, MeltQuote, MintKeySetInfo, MintQuote as MintMintQuote, Operation, ProofsWithState,
13};
14use crate::nuts::{
15 BlindSignature, BlindedMessage, CurrencyUnit, Id, MeltQuoteState, Proof, Proofs, PublicKey,
16 State,
17};
18use crate::payment::PaymentIdentifier;
19
20mod auth;
21
22#[cfg(feature = "test")]
23pub mod test;
24
25pub use auth::{DynMintAuthDatabase, MintAuthDatabase, MintAuthTransaction};
26
27pub use super::kvstore::{
29 validate_kvstore_params, validate_kvstore_string, KVStore, KVStoreDatabase, KVStoreTransaction,
30 KVSTORE_NAMESPACE_KEY_ALPHABET, KVSTORE_NAMESPACE_KEY_MAX_LEN,
31};
32
33#[derive(Debug)]
62pub struct Acquired<T> {
63 inner: T,
64}
65
66impl<T> From<T> for Acquired<T> {
67 fn from(value: T) -> Self {
72 Acquired { inner: value }
73 }
74}
75
76impl<T> Acquired<T> {
77 pub fn inner(self) -> T {
83 self.inner
84 }
85}
86
87impl<T> Deref for Acquired<T> {
88 type Target = T;
89
90 fn deref(&self) -> &Self::Target {
92 &self.inner
93 }
94}
95
96impl<T> DerefMut for Acquired<T> {
97 fn deref_mut(&mut self) -> &mut Self::Target {
99 &mut self.inner
100 }
101}
102
103#[derive(Debug, Clone, PartialEq, Eq)]
105pub struct MeltRequestInfo {
106 pub inputs_amount: Amount<CurrencyUnit>,
108 pub inputs_fee: Amount<CurrencyUnit>,
110 pub change_outputs: Vec<BlindedMessage>,
112}
113
114#[derive(Debug)]
119pub struct LockedMeltQuotes {
120 pub target: Option<Acquired<MeltQuote>>,
122 pub all_related: Vec<Acquired<MeltQuote>>,
124}
125
126#[async_trait]
128pub trait KeysDatabaseTransaction<'a, Error>: DbTransactionFinalizer<Err = Error> {
129 async fn set_active_keyset(&mut self, unit: CurrencyUnit, id: Id) -> Result<(), Error>;
131
132 async fn add_keyset_info(&mut self, keyset: MintKeySetInfo) -> Result<(), Error>;
134}
135
136#[async_trait]
138pub trait KeysDatabase {
139 type Err: Into<Error> + From<Error>;
141
142 async fn begin_transaction<'a>(
144 &'a self,
145 ) -> Result<Box<dyn KeysDatabaseTransaction<'a, Self::Err> + Send + Sync + 'a>, Error>;
146
147 async fn get_active_keyset_id(&self, unit: &CurrencyUnit) -> Result<Option<Id>, Self::Err>;
149
150 async fn get_active_keysets(&self) -> Result<HashMap<CurrencyUnit, Id>, Self::Err>;
152
153 async fn get_keyset_info(&self, id: &Id) -> Result<Option<MintKeySetInfo>, Self::Err>;
155
156 async fn get_keyset_infos(&self) -> Result<Vec<MintKeySetInfo>, Self::Err>;
158}
159
160#[async_trait]
162pub trait QuotesTransaction {
163 type Err: Into<Error> + From<Error>;
165
166 async fn add_melt_request(
168 &mut self,
169 quote_id: &QuoteId,
170 inputs_amount: Amount<CurrencyUnit>,
171 inputs_fee: Amount<CurrencyUnit>,
172 ) -> Result<(), Self::Err>;
173
174 async fn add_blinded_messages(
176 &mut self,
177 quote_id: Option<&QuoteId>,
178 blinded_messages: &[BlindedMessage],
179 operation: &Operation,
180 ) -> Result<(), Self::Err>;
181
182 async fn delete_blinded_messages(
184 &mut self,
185 blinded_secrets: &[PublicKey],
186 ) -> Result<(), Self::Err>;
187
188 async fn get_melt_request_and_blinded_messages(
190 &mut self,
191 quote_id: &QuoteId,
192 ) -> Result<Option<MeltRequestInfo>, Self::Err>;
193
194 async fn delete_melt_request(&mut self, quote_id: &QuoteId) -> Result<(), Self::Err>;
196
197 async fn get_mint_quote(
199 &mut self,
200 quote_id: &QuoteId,
201 ) -> Result<Option<Acquired<MintMintQuote>>, Self::Err>;
202
203 async fn get_mint_quotes_by_ids(
208 &mut self,
209 quote_ids: &[QuoteId],
210 ) -> Result<Vec<Option<Acquired<MintMintQuote>>>, Self::Err>;
211
212 async fn add_mint_quote(
214 &mut self,
215 quote: MintMintQuote,
216 ) -> Result<Acquired<MintMintQuote>, Self::Err>;
217
218 async fn update_mint_quote(
238 &mut self,
239 quote: &mut Acquired<mint::MintQuote>,
240 ) -> Result<(), Self::Err>;
241
242 async fn get_melt_quote(
244 &mut self,
245 quote_id: &QuoteId,
246 ) -> Result<Option<Acquired<mint::MeltQuote>>, Self::Err>;
247
248 async fn add_melt_quote(&mut self, quote: mint::MeltQuote) -> Result<(), Self::Err>;
250
251 async fn get_melt_quotes_by_request_lookup_id(
267 &mut self,
268 request_lookup_id: &PaymentIdentifier,
269 ) -> Result<Vec<Acquired<MeltQuote>>, Self::Err>;
270
271 async fn lock_melt_quote_and_related(
297 &mut self,
298 quote_id: &QuoteId,
299 ) -> Result<LockedMeltQuotes, Self::Err>;
300
301 async fn update_melt_quote_request_lookup_id(
305 &mut self,
306 quote: &mut Acquired<mint::MeltQuote>,
307 new_request_lookup_id: &PaymentIdentifier,
308 ) -> Result<(), Self::Err>;
309
310 async fn update_melt_quote_state(
315 &mut self,
316 quote: &mut Acquired<mint::MeltQuote>,
317 new_state: MeltQuoteState,
318 payment_proof: Option<String>,
319 ) -> Result<MeltQuoteState, Self::Err>;
320
321 async fn get_mint_quote_by_request(
323 &mut self,
324 request: &str,
325 ) -> Result<Option<Acquired<MintMintQuote>>, Self::Err>;
326
327 async fn get_mint_quote_by_request_lookup_id(
329 &mut self,
330 request_lookup_id: &PaymentIdentifier,
331 ) -> Result<Option<Acquired<MintMintQuote>>, Self::Err>;
332}
333
334#[async_trait]
336pub trait QuotesDatabase {
337 type Err: Into<Error> + From<Error>;
339
340 async fn get_mint_quote(&self, quote_id: &QuoteId) -> Result<Option<MintMintQuote>, Self::Err>;
342
343 async fn get_mint_quotes_by_ids(
347 &self,
348 quote_ids: &[QuoteId],
349 ) -> Result<Vec<Option<MintMintQuote>>, Self::Err>;
350
351 async fn get_mint_quote_by_request(
353 &self,
354 request: &str,
355 ) -> Result<Option<MintMintQuote>, Self::Err>;
356 async fn get_mint_quote_by_request_lookup_id(
358 &self,
359 request_lookup_id: &PaymentIdentifier,
360 ) -> Result<Option<MintMintQuote>, Self::Err>;
361 async fn get_mint_quotes(&self) -> Result<Vec<MintMintQuote>, Self::Err>;
363 async fn get_melt_quote(
365 &self,
366 quote_id: &QuoteId,
367 ) -> Result<Option<mint::MeltQuote>, Self::Err>;
368 async fn get_melt_quotes(&self) -> Result<Vec<mint::MeltQuote>, Self::Err>;
370}
371
372#[async_trait]
374pub trait ProofsTransaction {
375 type Err: Into<Error> + From<Error>;
377
378 async fn add_proofs(
383 &mut self,
384 proof: Proofs,
385 quote_id: Option<QuoteId>,
386 operation: &Operation,
387 ) -> Result<Acquired<ProofsWithState>, Self::Err>;
388
389 async fn update_proofs_state(
394 &mut self,
395 proofs: &mut Acquired<ProofsWithState>,
396 new_state: State,
397 ) -> Result<(), Self::Err>;
398
399 async fn get_proofs(
401 &mut self,
402 ys: &[PublicKey],
403 ) -> Result<Acquired<ProofsWithState>, Self::Err>;
404
405 async fn remove_proofs(
407 &mut self,
408 ys: &[PublicKey],
409 quote_id: Option<QuoteId>,
410 ) -> Result<(), Self::Err>;
411
412 async fn get_proof_ys_by_quote_id(
414 &mut self,
415 quote_id: &QuoteId,
416 ) -> Result<Vec<PublicKey>, Self::Err>;
417
418 async fn get_proof_ys_by_operation_id(
420 &mut self,
421 operation_id: &uuid::Uuid,
422 ) -> Result<Vec<PublicKey>, Self::Err>;
423}
424
425#[async_trait]
427pub trait ProofsDatabase {
428 type Err: Into<Error> + From<Error>;
430
431 async fn get_proofs_by_ys(&self, ys: &[PublicKey]) -> Result<Vec<Option<Proof>>, Self::Err>;
433 async fn get_proof_ys_by_quote_id(
435 &self,
436 quote_id: &QuoteId,
437 ) -> Result<Vec<PublicKey>, Self::Err>;
438 async fn get_proofs_states(&self, ys: &[PublicKey]) -> Result<Vec<Option<State>>, Self::Err>;
440
441 async fn get_proofs_by_keyset_id(
443 &self,
444 keyset_id: &Id,
445 ) -> Result<(Proofs, Vec<Option<State>>), Self::Err>;
446
447 async fn get_total_redeemed(&self) -> Result<HashMap<Id, Amount>, Self::Err>;
449
450 async fn get_proof_ys_by_operation_id(
452 &self,
453 operation_id: &uuid::Uuid,
454 ) -> Result<Vec<PublicKey>, Self::Err>;
455}
456
457#[async_trait]
458pub trait SignaturesTransaction {
460 type Err: Into<Error> + From<Error>;
462
463 async fn add_blind_signatures(
465 &mut self,
466 blinded_messages: &[PublicKey],
467 blind_signatures: &[BlindSignature],
468 quote_id: Option<QuoteId>,
469 ) -> Result<(), Self::Err>;
470
471 async fn get_blind_signatures(
473 &mut self,
474 blinded_messages: &[PublicKey],
475 ) -> Result<Vec<Option<BlindSignature>>, Self::Err>;
476}
477
478#[async_trait]
479pub trait SignaturesDatabase {
481 type Err: Into<Error> + From<Error>;
483
484 async fn get_blind_signatures(
486 &self,
487 blinded_messages: &[PublicKey],
488 ) -> Result<Vec<Option<BlindSignature>>, Self::Err>;
489
490 async fn get_blind_signatures_for_keyset(
492 &self,
493 keyset_id: &Id,
494 ) -> Result<Vec<BlindSignature>, Self::Err>;
495
496 async fn get_blind_signatures_for_quote(
498 &self,
499 quote_id: &QuoteId,
500 ) -> Result<Vec<BlindSignature>, Self::Err>;
501
502 async fn get_total_issued(&self) -> Result<HashMap<Id, Amount>, Self::Err>;
504
505 async fn get_blinded_secrets_by_operation_id(
507 &self,
508 operation_id: &uuid::Uuid,
509 ) -> Result<Vec<PublicKey>, Self::Err>;
510}
511
512#[async_trait]
513pub trait SagaTransaction {
515 type Err: Into<Error> + From<Error>;
517
518 async fn get_saga(
520 &mut self,
521 operation_id: &uuid::Uuid,
522 ) -> Result<Option<mint::Saga>, Self::Err>;
523
524 async fn add_saga(&mut self, saga: &mint::Saga) -> Result<(), Self::Err>;
526
527 async fn update_saga(
529 &mut self,
530 operation_id: &uuid::Uuid,
531 new_state: mint::SagaStateEnum,
532 ) -> Result<(), Self::Err>;
533
534 async fn delete_saga(&mut self, operation_id: &uuid::Uuid) -> Result<(), Self::Err>;
536}
537
538#[async_trait]
539pub trait SagaDatabase {
541 type Err: Into<Error> + From<Error>;
543
544 async fn get_incomplete_sagas(
546 &self,
547 operation_kind: mint::OperationKind,
548 ) -> Result<Vec<mint::Saga>, Self::Err>;
549}
550
551#[async_trait]
552pub trait CompletedOperationsTransaction {
554 type Err: Into<Error> + From<Error>;
556
557 async fn add_completed_operation(
559 &mut self,
560 operation: &mint::Operation,
561 fee_by_keyset: &std::collections::HashMap<crate::nuts::Id, crate::Amount>,
562 ) -> Result<(), Self::Err>;
563}
564
565#[async_trait]
566pub trait CompletedOperationsDatabase {
568 type Err: Into<Error> + From<Error>;
570
571 async fn get_completed_operation(
573 &self,
574 operation_id: &uuid::Uuid,
575 ) -> Result<Option<mint::Operation>, Self::Err>;
576
577 async fn get_completed_operations_by_kind(
579 &self,
580 operation_kind: mint::OperationKind,
581 ) -> Result<Vec<mint::Operation>, Self::Err>;
582
583 async fn get_completed_operations(&self) -> Result<Vec<mint::Operation>, Self::Err>;
585}
586
587pub trait Transaction<Error>:
589 DbTransactionFinalizer<Err = Error>
590 + QuotesTransaction<Err = Error>
591 + SignaturesTransaction<Err = Error>
592 + ProofsTransaction<Err = Error>
593 + KVStoreTransaction<Error>
594 + SagaTransaction<Err = Error>
595 + CompletedOperationsTransaction<Err = Error>
596{
597}
598
599#[async_trait]
601pub trait Database<Error>:
602 KVStoreDatabase<Err = Error>
603 + QuotesDatabase<Err = Error>
604 + ProofsDatabase<Err = Error>
605 + SignaturesDatabase<Err = Error>
606 + SagaDatabase<Err = Error>
607 + CompletedOperationsDatabase<Err = Error>
608{
609 async fn begin_transaction(&self) -> Result<Box<dyn Transaction<Error> + Send + Sync>, Error>;
611}
612
613pub type DynMintDatabase = std::sync::Arc<dyn Database<Error> + Send + Sync>;
615
616pub type DynMintTransaction = Box<dyn Transaction<Error> + Send + Sync>;