mirror of
https://github.com/aljazceru/nutshell.git
synced 2025-12-22 19:34:18 +01:00
* settings * fix name settings * management rpc * hook up the RPC server * working * format * update build script fix import error * remove accidental commit of vscode extension data * working ✔ * \n * add get mint quote get melt quote * gRPC cli update quotes commands * update mint melt quotes from cli * comment under get cli command group * keyset rotation not yet implemented * try fix * change back contact info default to be empty list * fix import * add server mTLS * ll * script for generating certificates * rename settings * move generation script * do not save TTL expiry into Cache object, rather always load from settings. * update lightning fees * update auth limits * auth rate limit cli * optional arguemnts * better error messages * tests for db update mint/melt quotes * start mint rpc tests * add tos_url field to get-info grpc response * format checks * add types to click groups where it's needed * tests on updating quotes * fix tests * skip updating mint quote state if on regtest * test edge case * unified test_add_remove_contact * mark pytest-asyncio * fix missing db argument * hopefully no more silly errors * fix test_db_update_mint_quote_state * pass in the quote id string. * add keyset rotation * test for keyset rotation through gRPC command * fix logger warning * remove rotation test because it breaks other tests * use different bolt11 invoices * assert returned melt quote has quote * is_postgres * try different things * skip if deprecated api * format checks * update .gitignore * default location for certificates
128 lines
4.5 KiB
Python
128 lines
4.5 KiB
Python
import json
|
|
import re
|
|
from typing import Any, Dict, List, Optional
|
|
|
|
from pydantic import BaseModel
|
|
|
|
from .base import Method, Unit
|
|
from .models import MintInfoContact, MintInfoProtectedEndpoint, Nut15MppSupport
|
|
from .nuts.nuts import BLIND_AUTH_NUT, CLEAR_AUTH_NUT, MPP_NUT, WEBSOCKETS_NUT
|
|
|
|
|
|
class MintInfo(BaseModel):
|
|
name: Optional[str]
|
|
pubkey: Optional[str]
|
|
version: Optional[str]
|
|
description: Optional[str]
|
|
description_long: Optional[str]
|
|
contact: Optional[List[MintInfoContact]]
|
|
motd: Optional[str]
|
|
icon_url: Optional[str]
|
|
urls: Optional[List[str]]
|
|
tos_url: Optional[str]
|
|
time: Optional[int]
|
|
nuts: Dict[int, Any]
|
|
|
|
def __str__(self):
|
|
return f"{self.name} ({self.description})"
|
|
|
|
@classmethod
|
|
def from_json_str(cls, json_str: str):
|
|
return cls.parse_obj(json.loads(json_str))
|
|
|
|
def supports_nut(self, nut: int) -> bool:
|
|
if self.nuts is None:
|
|
return False
|
|
return nut in self.nuts
|
|
|
|
def supports_mpp(self, method: str, unit: Unit) -> bool:
|
|
if not self.nuts:
|
|
return False
|
|
nut_15 = self.nuts.get(MPP_NUT)
|
|
if not nut_15 or not self.supports_nut(MPP_NUT) or not nut_15.get("methods"):
|
|
return False
|
|
|
|
for entry in nut_15["methods"]:
|
|
entry_obj = Nut15MppSupport.parse_obj(entry)
|
|
if entry_obj.method == method and entry_obj.unit == unit.name:
|
|
return True
|
|
|
|
return False
|
|
|
|
def supports_websocket_mint_quote(self, method: Method, unit: Unit) -> bool:
|
|
if not self.nuts or not self.supports_nut(WEBSOCKETS_NUT):
|
|
return False
|
|
websocket_settings = self.nuts[WEBSOCKETS_NUT]
|
|
if not websocket_settings or "supported" not in websocket_settings:
|
|
return False
|
|
websocket_supported = websocket_settings["supported"]
|
|
for entry in websocket_supported:
|
|
if entry["method"] == method.name and entry["unit"] == unit.name:
|
|
if "bolt11_mint_quote" in entry["commands"]:
|
|
return True
|
|
return False
|
|
|
|
def requires_clear_auth(self) -> bool:
|
|
return self.supports_nut(CLEAR_AUTH_NUT)
|
|
|
|
def oidc_discovery_url(self) -> str:
|
|
if not self.requires_clear_auth():
|
|
raise Exception(
|
|
"Could not get OIDC discovery URL. Mint info does not support clear auth."
|
|
)
|
|
return self.nuts[CLEAR_AUTH_NUT]["openid_discovery"]
|
|
|
|
def oidc_client_id(self) -> str:
|
|
if not self.requires_clear_auth():
|
|
raise Exception(
|
|
"Could not get client_id. Mint info does not support clear auth."
|
|
)
|
|
return self.nuts[CLEAR_AUTH_NUT]["client_id"]
|
|
|
|
def required_clear_auth_endpoints(self) -> List[MintInfoProtectedEndpoint]:
|
|
if not self.requires_clear_auth():
|
|
return []
|
|
return [
|
|
MintInfoProtectedEndpoint.parse_obj(e)
|
|
for e in self.nuts[CLEAR_AUTH_NUT]["protected_endpoints"]
|
|
]
|
|
|
|
def requires_clear_auth_path(self, method: str, path: str) -> bool:
|
|
if not self.requires_clear_auth():
|
|
return False
|
|
path = "/" + path if not path.startswith("/") else path
|
|
for endpoint in self.required_clear_auth_endpoints():
|
|
if method == endpoint.method and re.match(endpoint.path, path):
|
|
return True
|
|
return False
|
|
|
|
def requires_blind_auth(self) -> bool:
|
|
return self.supports_nut(BLIND_AUTH_NUT)
|
|
|
|
@property
|
|
def bat_max_mint(self) -> int:
|
|
if not self.requires_blind_auth():
|
|
raise Exception(
|
|
"Could not get max mint. Mint info does not support blind auth."
|
|
)
|
|
if not self.nuts[BLIND_AUTH_NUT].get("bat_max_mint"):
|
|
raise Exception("Could not get max mint. bat_max_mint not set.")
|
|
return self.nuts[BLIND_AUTH_NUT]["bat_max_mint"]
|
|
|
|
def required_blind_auth_paths(self) -> List[MintInfoProtectedEndpoint]:
|
|
if not self.requires_blind_auth():
|
|
return []
|
|
return [
|
|
MintInfoProtectedEndpoint.parse_obj(e)
|
|
for e in self.nuts[BLIND_AUTH_NUT]["protected_endpoints"]
|
|
]
|
|
|
|
def requires_blind_auth_path(self, method: str, path: str) -> bool:
|
|
if not self.requires_blind_auth():
|
|
return False
|
|
path = "/" + path if not path.startswith("/") else path
|
|
for endpoint in self.required_blind_auth_paths():
|
|
if method == endpoint.method and re.match(endpoint.path, path):
|
|
return True
|
|
return False
|