diff --git a/cashu/mint/ledger.py b/cashu/mint/ledger.py index 3bb0cb8a..cff10b4d 100644 --- a/cashu/mint/ledger.py +++ b/cashu/mint/ledger.py @@ -325,8 +325,11 @@ async def mint_quote(self, quote_request: PostMintQuoteRequest) -> MintQuote: ) if settings.mint_peg_out_only: raise NotAllowedError("Mint does not allow minting new tokens.") - unit = Unit[quote_request.unit] - method = Method.bolt11 + + unit, method = self._verify_and_get_unit_method( + quote_request.unit, Method.bolt11.name + ) + if settings.mint_max_balance: balance = await self.get_balance() if balance + quote_request.amount > settings.mint_max_balance: @@ -387,10 +390,10 @@ async def get_mint_quote(self, quote_id: str) -> MintQuote: MintQuote: Mint quote object. """ quote = await self.crud.get_mint_quote(quote_id=quote_id, db=self.db) - assert quote, "quote not found" - assert quote.method == Method.bolt11.name, "only bolt11 supported" - unit = Unit[quote.unit] - method = Method[quote.method] + if not quote: + raise Exception("quote not found") + + unit, method = self._verify_and_get_unit_method(quote.unit, quote.method) if not quote.paid: assert quote.checking_id, "quote has no checking id" @@ -471,8 +474,10 @@ async def melt_quote( Returns: PostMeltQuoteResponse: Melt quote response. """ - unit = Unit[melt_quote.unit] - method = Method.bolt11 + unit, method = self._verify_and_get_unit_method( + melt_quote.unit, Method.bolt11.name + ) + # NOTE: we normalize the request to lowercase to avoid case sensitivity # This works with Lightning but might not work with other methods request = melt_quote.request.lower() @@ -557,10 +562,12 @@ async def get_melt_quote( MeltQuote: Melt quote object. """ melt_quote = await self.crud.get_melt_quote(quote_id=quote_id, db=self.db) - assert melt_quote, "quote not found" - assert melt_quote.method == Method.bolt11.name, "only bolt11 supported" - unit = Unit[melt_quote.unit] - method = Method[melt_quote.method] + if not melt_quote: + raise Exception("quote not found") + + unit, method = self._verify_and_get_unit_method( + melt_quote.unit, melt_quote.method + ) # we only check the state with the backend if there is no associated internal # mint quote for this melt quote @@ -664,8 +671,11 @@ async def melt( """ # get melt quote and check if it was already paid melt_quote = await self.get_melt_quote(quote_id=quote) - method = Method[melt_quote.method] - unit = Unit[melt_quote.unit] + + unit, method = self._verify_and_get_unit_method( + melt_quote.unit, melt_quote.method + ) + assert not melt_quote.paid, "melt quote already paid" # make sure that the outputs (for fee return) are in the same unit as the quote diff --git a/cashu/mint/protocols.py b/cashu/mint/protocols.py index 47bf618e..04d24c0c 100644 --- a/cashu/mint/protocols.py +++ b/cashu/mint/protocols.py @@ -1,6 +1,6 @@ -from typing import Dict, Protocol +from typing import Dict, Mapping, Protocol -from ..core.base import MintKeyset, Unit +from ..core.base import Method, MintKeyset, Unit from ..core.db import Database from ..lightning.base import LightningBackend from ..mint.crud import LedgerCrud @@ -11,8 +11,8 @@ class SupportsKeysets(Protocol): keysets: Dict[str, MintKeyset] -class SupportLightning(Protocol): - lightning: Dict[Unit, LightningBackend] +class SupportsBackends(Protocol): + backends: Mapping[Method, Mapping[Unit, LightningBackend]] = {} class SupportsDb(Protocol): diff --git a/cashu/mint/verification.py b/cashu/mint/verification.py index 754c8b1c..de38dca3 100644 --- a/cashu/mint/verification.py +++ b/cashu/mint/verification.py @@ -1,12 +1,14 @@ -from typing import Dict, List, Literal, Optional, Union +from typing import Dict, List, Literal, Optional, Tuple, Union from loguru import logger from ..core.base import ( BlindedMessage, BlindedSignature, + Method, MintKeyset, Proof, + Unit, ) from ..core.crypto import b_dhke from ..core.crypto.secp import PublicKey @@ -19,12 +21,15 @@ TransactionError, ) from ..core.settings import settings +from ..lightning.base import LightningBackend from ..mint.crud import LedgerCrud from .conditions import LedgerSpendingConditions -from .protocols import SupportsDb, SupportsKeysets +from .protocols import SupportsBackends, SupportsDb, SupportsKeysets -class LedgerVerification(LedgerSpendingConditions, SupportsKeysets, SupportsDb): +class LedgerVerification( + LedgerSpendingConditions, SupportsKeysets, SupportsDb, SupportsBackends +): """Verification functions for the ledger.""" keyset: MintKeyset @@ -32,6 +37,7 @@ class LedgerVerification(LedgerSpendingConditions, SupportsKeysets, SupportsDb): spent_proofs: Dict[str, Proof] crud: LedgerCrud db: Database + lightning: Dict[Unit, LightningBackend] async def verify_inputs_and_outputs( self, *, proofs: List[Proof], outputs: Optional[List[BlindedMessage]] = None @@ -240,6 +246,22 @@ def _verify_equation_balanced( """ sum_inputs = sum(self._verify_amount(p.amount) for p in proofs) sum_outputs = sum(self._verify_amount(p.amount) for p in outs) - assert sum_outputs - sum_inputs == 0, TransactionError( - "inputs do not have same amount as outputs." - ) + if not sum_outputs - sum_inputs == 0: + raise TransactionError("inputs do not have same amount as outputs.") + + def _verify_and_get_unit_method( + self, unit_str: str, method_str: str + ) -> Tuple[Unit, Method]: + """Verify that the unit is supported by the ledger.""" + method = Method[method_str] + unit = Unit[unit_str] + + if not any([unit == k.unit for k in self.keysets.values()]): + raise NotAllowedError(f"unit '{unit.name}' not supported in any keyset.") + + if not self.backends.get(method) or unit not in self.backends[method]: + raise NotAllowedError( + f"no support for method '{method.name}' with unit '{unit.name}'." + ) + + return unit, method