mirror of
https://github.com/lnbits/lnbits-legend.git
synced 2024-11-19 18:11:30 +01:00
eb73daffe9
* [FEAT] Node Managment feat: node dashboard channels and transactions fix: update channel variables better types refactor ui add onchain balances and backend_name mock values for fake wallet remove app tab start implementing peers and channel management peer and channel management implement channel closing add channel states, better errors seperate payments and invoices on transactions tab display total channel balance feat: optional public page feat: show node address fix: port conversion feat: details dialog on transactions fix: peer info without alias fix: rename channel balances small improvements to channels tab feat: pagination on transactions tab test caching transactions refactor: move WALLET into wallets module fix: backwards compatibility refactor: move get_node_class to nodes modules post merge bundle fundle feat: disconnect peer feat: initial lnd support only use filtered channels for total balance adjust closing logic add basic node tests add setting for disabling transactions tab revert unnecessary changes add tests for invoices and payments improve payment and invoice implementations the previously used invoice fixture has a session scope, but a new invoice is required tests and bug fixes for channels api use query instead of body in channel delete delete requests should generally not use a body take node id through path instead of body for delete endpoint add peer management tests more tests for errors improve error handling rename id and pubkey to peer_id for consistency remove dead code fix http status codes make cache keys safer cache node public info comments for node settings rename node prop in frontend adjust tests to new status codes cln: use amount_msat instead of value for onchain balance turn transactions tab off by default enable transactions in tests only allow super user to create or delete fix prop name in admin navbar --------- Co-authored-by: jacksn <jkranawetter05@gmail.com>
224 lines
5.1 KiB
Python
224 lines
5.1 KiB
Python
from __future__ import annotations
|
|
|
|
from abc import ABC, abstractmethod
|
|
from enum import Enum
|
|
from typing import TYPE_CHECKING, List, Optional
|
|
|
|
from pydantic import BaseModel
|
|
|
|
from lnbits.db import FilterModel, Filters, Page
|
|
from lnbits.utils.cache import cache
|
|
|
|
if TYPE_CHECKING:
|
|
from lnbits.wallets.base import Wallet
|
|
|
|
|
|
class NodePeerInfo(BaseModel):
|
|
id: str
|
|
alias: Optional[str] = None
|
|
color: Optional[str] = None
|
|
last_timestamp: Optional[int] = None
|
|
addresses: Optional[list[str]] = None
|
|
|
|
|
|
class ChannelState(Enum):
|
|
ACTIVE = "active"
|
|
PENDING = "pending"
|
|
CLOSED = "closed"
|
|
INACTIVE = "inactive"
|
|
|
|
|
|
class ChannelBalance(BaseModel):
|
|
local_msat: int
|
|
remote_msat: int
|
|
total_msat: int
|
|
|
|
|
|
class ChannelPoint(BaseModel):
|
|
funding_txid: str
|
|
output_index: int
|
|
|
|
|
|
class NodeChannel(BaseModel):
|
|
short_id: Optional[str] = None
|
|
point: Optional[ChannelPoint] = None
|
|
peer_id: str
|
|
balance: ChannelBalance
|
|
state: ChannelState
|
|
name: Optional[str]
|
|
color: Optional[str]
|
|
|
|
|
|
class ChannelStats(BaseModel):
|
|
counts: dict[ChannelState, int]
|
|
avg_size: int
|
|
biggest_size: Optional[int]
|
|
smallest_size: Optional[int]
|
|
total_capacity: int
|
|
|
|
@classmethod
|
|
def from_list(cls, channels: list[NodeChannel]):
|
|
counts: dict[ChannelState, int] = {}
|
|
for channel in channels:
|
|
counts[channel.state] = counts.get(channel.state, 0) + 1
|
|
|
|
return cls(
|
|
counts=counts,
|
|
avg_size=int(
|
|
sum(channel.balance.total_msat for channel in channels) / len(channels)
|
|
),
|
|
biggest_size=max(channel.balance.total_msat for channel in channels),
|
|
smallest_size=min(channel.balance.total_msat for channel in channels),
|
|
total_capacity=sum(channel.balance.total_msat for channel in channels),
|
|
)
|
|
|
|
|
|
class NodeFees(BaseModel):
|
|
total_msat: int
|
|
daily_msat: Optional[int] = None
|
|
weekly_msat: Optional[int] = None
|
|
monthly_msat: Optional[int] = None
|
|
|
|
|
|
class PublicNodeInfo(BaseModel):
|
|
id: str
|
|
backend_name: str
|
|
alias: str
|
|
color: str
|
|
num_peers: int
|
|
blockheight: int
|
|
channel_stats: ChannelStats
|
|
addresses: list[str]
|
|
|
|
|
|
class NodeInfoResponse(PublicNodeInfo):
|
|
onchain_balance_sat: int
|
|
onchain_confirmed_sat: int
|
|
fees: NodeFees
|
|
balance_msat: int
|
|
|
|
|
|
class NodePayment(BaseModel):
|
|
pending: bool
|
|
amount: int
|
|
fee: Optional[int] = None
|
|
memo: Optional[str] = None
|
|
time: int
|
|
bolt11: str
|
|
preimage: Optional[str]
|
|
payment_hash: str
|
|
expiry: Optional[float] = None
|
|
destination: Optional[NodePeerInfo] = None
|
|
|
|
|
|
class NodeInvoice(BaseModel):
|
|
pending: bool
|
|
amount: int
|
|
memo: Optional[str]
|
|
bolt11: str
|
|
preimage: Optional[str]
|
|
payment_hash: str
|
|
paid_at: Optional[int] = None
|
|
expiry: Optional[int] = None
|
|
|
|
|
|
class NodeInvoiceFilters(FilterModel):
|
|
pass
|
|
|
|
|
|
class NodePaymentsFilters(FilterModel):
|
|
pass
|
|
|
|
|
|
class Node(ABC):
|
|
wallet: Wallet
|
|
|
|
def __init__(self, wallet: Wallet):
|
|
self.wallet = wallet
|
|
self.id: Optional[str] = None
|
|
|
|
@property
|
|
def name(self):
|
|
return self.__class__.__name__
|
|
|
|
async def get_id(self):
|
|
if not self.id:
|
|
self.id = await self._get_id()
|
|
return self.id
|
|
|
|
@abstractmethod
|
|
async def _get_id(self) -> str:
|
|
pass
|
|
|
|
async def get_peers(self) -> list[NodePeerInfo]:
|
|
peer_ids = await self.get_peer_ids()
|
|
return [await self.get_peer_info(peer_id) for peer_id in peer_ids]
|
|
|
|
@abstractmethod
|
|
async def get_peer_ids(self) -> list[str]:
|
|
pass
|
|
|
|
@abstractmethod
|
|
async def connect_peer(self, uri: str):
|
|
pass
|
|
|
|
@abstractmethod
|
|
async def disconnect_peer(self, peer_id: str):
|
|
pass
|
|
|
|
@abstractmethod
|
|
async def _get_peer_info(self, peer_id: str) -> NodePeerInfo:
|
|
pass
|
|
|
|
async def get_peer_info(self, peer_id: str) -> NodePeerInfo:
|
|
key = f"node:peers:{peer_id}"
|
|
info = cache.get(key)
|
|
if not info:
|
|
info = await self._get_peer_info(peer_id)
|
|
if info.last_timestamp:
|
|
cache.set(key, info)
|
|
return info
|
|
|
|
@abstractmethod
|
|
async def open_channel(
|
|
self,
|
|
peer_id: str,
|
|
local_amount: int,
|
|
push_amount: Optional[int] = None,
|
|
fee_rate: Optional[int] = None,
|
|
) -> ChannelPoint:
|
|
pass
|
|
|
|
@abstractmethod
|
|
async def close_channel(
|
|
self,
|
|
short_id: Optional[str] = None,
|
|
point: Optional[ChannelPoint] = None,
|
|
force: bool = False,
|
|
):
|
|
pass
|
|
|
|
@abstractmethod
|
|
async def get_channels(self) -> List[NodeChannel]:
|
|
pass
|
|
|
|
@abstractmethod
|
|
async def get_info(self) -> NodeInfoResponse:
|
|
pass
|
|
|
|
async def get_public_info(self) -> PublicNodeInfo:
|
|
info = await self.get_info()
|
|
return PublicNodeInfo(**info.__dict__)
|
|
|
|
@abstractmethod
|
|
async def get_payments(
|
|
self, filters: Filters[NodePaymentsFilters]
|
|
) -> Page[NodePayment]:
|
|
pass
|
|
|
|
@abstractmethod
|
|
async def get_invoices(
|
|
self, filters: Filters[NodeInvoiceFilters]
|
|
) -> Page[NodeInvoice]:
|
|
pass
|