mirror of
https://github.com/lnbits/lnbits-legend.git
synced 2025-02-26 07:31:22 +01:00
* feat: add shortcuts for insert_query and update_query into `Database` example: await db.insert("table_name", base_model) * remove where from argument * chore: code clean-up * extension manager * lnbits-qrcode components * parse date from dict * refactor: make `settings` a fixture * chore: remove verbose key names * fix: time column * fix: cast balance to `int` * extension toggle vue3 * vue3 @input migration * fix: payment extra and payment hash * fix dynamic fields and ext db migration * remove shadow on cards in dark theme * screwed up and made more css pushes to this branch * attempt to make chip component in settings dynamic fields * dynamic chips * qrscanner * clean init admin settings * make get_user better * add dbversion model * remove update_payment_status/extra/details * traces for value and assertion errors * refactor services * add PaymentFiatAmount * return Payment on api endpoints * rename to get_user_from_account * refactor: just refactor (#2740) * rc5 * Fix db cache (#2741) * [refactor] split services.py (#2742) * refactor: spit `core.py` (#2743) * refactor: make QR more customizable * fix: print.html * fix: qrcode options * fix: white shadow on dark theme * fix: datetime wasnt parsed in dict_to_model * add timezone for conversion * only parse timestamp for sqlite, postgres does it * log internal payment success * fix: export wallet to phone QR * Adding a customisable border theme, like gradient (#2746) * fixed mobile scan btn * fix test websocket * fix get_payments tests * dict_to_model skip none values * preimage none instead of defaulting to 0000... * fixup test real invoice tests * fixed pheonixd for wss * fix nodemanager test settings * fix lnbits funding * only insert extension when they dont exist --------- Co-authored-by: Vlad Stan <stan.v.vlad@gmail.com> Co-authored-by: Tiago Vasconcelos <talvasconcelos@gmail.com> Co-authored-by: Arc <ben@arc.wales> Co-authored-by: Arc <33088785+arcbtc@users.noreply.github.com>
153 lines
4.8 KiB
Python
153 lines
4.8 KiB
Python
import base64
|
|
import json
|
|
import time
|
|
from http import HTTPStatus
|
|
from typing import List
|
|
|
|
from fastapi import APIRouter, Depends
|
|
from fastapi.exceptions import HTTPException
|
|
|
|
from lnbits.core.crud import (
|
|
delete_account,
|
|
delete_wallet,
|
|
force_delete_wallet,
|
|
get_accounts,
|
|
get_wallet,
|
|
get_wallets,
|
|
update_admin_settings,
|
|
)
|
|
from lnbits.core.models import (
|
|
AccountFilters,
|
|
AccountOverview,
|
|
CreateTopup,
|
|
User,
|
|
Wallet,
|
|
)
|
|
from lnbits.core.services import update_wallet_balance
|
|
from lnbits.db import Filters, Page
|
|
from lnbits.decorators import check_admin, check_super_user, parse_filters
|
|
from lnbits.helpers import encrypt_internal_message, generate_filter_params_openapi
|
|
from lnbits.settings import EditableSettings, settings
|
|
|
|
users_router = APIRouter(prefix="/users/api/v1", dependencies=[Depends(check_admin)])
|
|
|
|
|
|
@users_router.get(
|
|
"/user",
|
|
name="get accounts",
|
|
summary="Get paginated list of accounts",
|
|
openapi_extra=generate_filter_params_openapi(AccountFilters),
|
|
)
|
|
async def api_get_users(
|
|
filters: Filters = Depends(parse_filters(AccountFilters)),
|
|
) -> Page[AccountOverview]:
|
|
return await get_accounts(filters=filters)
|
|
|
|
|
|
@users_router.delete("/user/{user_id}", status_code=HTTPStatus.OK)
|
|
async def api_users_delete_user(
|
|
user_id: str, user: User = Depends(check_admin)
|
|
) -> None:
|
|
wallets = await get_wallets(user_id)
|
|
if len(wallets) > 0:
|
|
raise HTTPException(
|
|
status_code=HTTPStatus.BAD_REQUEST,
|
|
detail="Cannot delete user with wallets.",
|
|
)
|
|
|
|
if user_id == settings.super_user:
|
|
raise HTTPException(
|
|
status_code=HTTPStatus.BAD_REQUEST,
|
|
detail="Cannot delete super user.",
|
|
)
|
|
|
|
if user_id in settings.lnbits_admin_users and not user.super_user:
|
|
raise HTTPException(
|
|
status_code=HTTPStatus.BAD_REQUEST,
|
|
detail="Only super_user can delete admin user.",
|
|
)
|
|
await delete_account(user_id)
|
|
|
|
|
|
@users_router.put(
|
|
"/user/{user_id}/reset_password", dependencies=[Depends(check_super_user)]
|
|
)
|
|
async def api_users_reset_password(user_id: str) -> str:
|
|
if user_id == settings.super_user:
|
|
raise HTTPException(
|
|
status_code=HTTPStatus.FORBIDDEN,
|
|
detail="Cannot change superuser password.",
|
|
)
|
|
|
|
reset_data = ["reset", user_id, int(time.time())]
|
|
reset_data_json = json.dumps(reset_data, separators=(",", ":"), ensure_ascii=False)
|
|
reset_key = encrypt_internal_message(reset_data_json)
|
|
assert reset_key, "Cannot generate reset key."
|
|
reset_key_b64 = base64.b64encode(reset_key.encode()).decode()
|
|
return f"reset_key_{reset_key_b64}"
|
|
|
|
|
|
@users_router.get("/user/{user_id}/admin", dependencies=[Depends(check_super_user)])
|
|
async def api_users_toggle_admin(user_id: str) -> None:
|
|
if user_id == settings.super_user:
|
|
raise HTTPException(
|
|
status_code=HTTPStatus.BAD_REQUEST,
|
|
detail="Cannot change super user.",
|
|
)
|
|
if user_id in settings.lnbits_admin_users:
|
|
settings.lnbits_admin_users.remove(user_id)
|
|
else:
|
|
settings.lnbits_admin_users.append(user_id)
|
|
update_settings = EditableSettings(lnbits_admin_users=settings.lnbits_admin_users)
|
|
await update_admin_settings(update_settings)
|
|
|
|
|
|
@users_router.get("/user/{user_id}/wallet")
|
|
async def api_users_get_user_wallet(user_id: str) -> List[Wallet]:
|
|
return await get_wallets(user_id)
|
|
|
|
|
|
@users_router.get("/user/{user_id}/wallet/{wallet}/undelete")
|
|
async def api_users_undelete_user_wallet(user_id: str, wallet: str) -> None:
|
|
wal = await get_wallet(wallet)
|
|
if not wal:
|
|
raise HTTPException(
|
|
status_code=HTTPStatus.NOT_FOUND,
|
|
detail="Wallet does not exist.",
|
|
)
|
|
|
|
if user_id != wal.user:
|
|
raise HTTPException(
|
|
status_code=HTTPStatus.FORBIDDEN,
|
|
detail="Wallet does not belong to user.",
|
|
)
|
|
if wal.deleted:
|
|
await delete_wallet(user_id=user_id, wallet_id=wallet, deleted=False)
|
|
|
|
|
|
@users_router.delete("/user/{user_id}/wallet/{wallet}")
|
|
async def api_users_delete_user_wallet(user_id: str, wallet: str) -> None:
|
|
wal = await get_wallet(wallet)
|
|
if not wal:
|
|
raise HTTPException(
|
|
status_code=HTTPStatus.NOT_FOUND,
|
|
detail="Wallet does not exist.",
|
|
)
|
|
if wal.deleted:
|
|
await force_delete_wallet(wallet)
|
|
await delete_wallet(user_id=user_id, wallet_id=wallet)
|
|
|
|
|
|
@users_router.put(
|
|
"/topup",
|
|
name="Topup",
|
|
status_code=HTTPStatus.OK,
|
|
dependencies=[Depends(check_super_user)],
|
|
)
|
|
async def api_topup_balance(data: CreateTopup) -> dict[str, str]:
|
|
await get_wallet(data.id)
|
|
if settings.lnbits_backend_wallet_class == "VoidWallet":
|
|
raise Exception("VoidWallet active")
|
|
|
|
await update_wallet_balance(wallet_id=data.id, amount=int(data.amount))
|
|
return {"status": "Success"}
|