lnbits-legend/lnbits/core/views/generic.py

442 lines
15 KiB
Python
Raw Normal View History

import asyncio
from http import HTTPStatus
from typing import List, Optional
from fastapi import Depends, Query, Request, status
2021-08-29 19:38:42 +02:00
from fastapi.exceptions import HTTPException
from fastapi.responses import FileResponse, RedirectResponse
from fastapi.routing import APIRouter
2022-07-16 14:23:03 +02:00
from loguru import logger
from pydantic.types import UUID4
from starlette.responses import HTMLResponse, JSONResponse
from lnbits.core import db
from lnbits.core.helpers import to_valid_user_id
2021-09-11 20:44:22 +02:00
from lnbits.core.models import User
from lnbits.decorators import check_admin, check_user_exists
2021-10-25 19:26:21 +01:00
from lnbits.helpers import template_renderer, url_for
from lnbits.settings import get_wallet_class, settings
2021-10-17 18:33:29 +01:00
2023-01-20 10:06:32 +02:00
from ...extension_manager import InstallableExtension, get_valid_extensions
from ...utils.exchange_rates import currencies
2021-10-17 18:33:29 +01:00
from ..crud import (
create_account,
create_wallet,
delete_wallet,
get_balance_check,
get_dbversions,
get_inactive_extensions,
2023-01-17 16:28:24 +02:00
get_installed_extensions,
2021-10-17 18:33:29 +01:00
get_user,
save_balance_notify,
update_installed_extension_state,
2021-10-17 18:33:29 +01:00
update_user_extension,
)
from ..services import pay_invoice, redeem_lnurl_withdraw
core_html_routes: APIRouter = APIRouter(tags=["Core NON-API Website Routes"])
2022-09-22 10:46:11 +02:00
@core_html_routes.get("/favicon.ico", response_class=FileResponse)
async def favicon():
return FileResponse("lnbits/core/static/favicon.ico")
2021-10-17 18:33:29 +01:00
@core_html_routes.get("/", response_class=HTMLResponse)
2022-12-23 11:53:45 +01:00
async def home(request: Request, lightning: str = ""):
2021-10-17 18:33:29 +01:00
return template_renderer().TemplateResponse(
"core/index.html", {"request": request, "lnurl": lightning}
)
2023-01-20 09:25:46 +01:00
@core_html_routes.get("/robots.txt", response_class=HTMLResponse)
async def robots():
data = """
User-agent: *
Disallow: /
"""
return HTMLResponse(content=data, media_type="text/plain")
@core_html_routes.get(
"/extensions", name="install.extensions", response_class=HTMLResponse
)
async def extensions_install(
2021-10-17 18:33:29 +01:00
request: Request,
user: User = Depends(check_user_exists),
activate: str = Query(None),
deactivate: str = Query(None),
enable: str = Query(None),
disable: str = Query(None),
2021-10-17 18:33:29 +01:00
):
await toggle_extension(enable, disable, user.id)
2021-09-11 20:44:22 +02:00
# Update user as his extensions have been updated
if enable or disable:
user = await get_user(user.id) # type: ignore
try:
installed_exts: List["InstallableExtension"] = await get_installed_extensions()
installed_exts_ids = [e.id for e in installed_exts]
2023-01-17 17:07:52 +02:00
installable_exts: List[
InstallableExtension
2023-01-17 17:07:52 +02:00
] = await InstallableExtension.get_installable_extensions()
installable_exts += [
e for e in installed_exts if e.id not in installed_exts_ids
2023-01-17 17:07:52 +02:00
]
for e in installable_exts:
2023-01-20 16:11:21 +02:00
installed_ext = next((ie for ie in installed_exts if e.id == ie.id), None)
if installed_ext:
e.installed_release = installed_ext.installed_release
# use the installed extension values
2023-01-20 16:11:21 +02:00
e.name = installed_ext.name
e.short_description = installed_ext.short_description
e.icon = installed_ext.icon
2023-01-17 17:07:52 +02:00
except Exception as ex:
logger.warning(ex)
installable_exts = []
try:
ext_id = activate or deactivate
if ext_id and user.admin:
if deactivate and deactivate not in settings.lnbits_deactivated_extensions:
settings.lnbits_deactivated_extensions += [deactivate]
elif activate:
settings.lnbits_deactivated_extensions = list(
filter(
lambda e: e != activate, settings.lnbits_deactivated_extensions
)
)
await update_installed_extension_state(
2023-01-21 15:08:59 +00:00
ext_id=ext_id, active=activate is not None
)
all_extensions = list(map(lambda e: e.code, get_valid_extensions()))
inactive_extensions = await get_inactive_extensions()
db_version = await get_dbversions()
extensions = list(
map(
lambda ext: {
"id": ext.id,
"name": ext.name,
"icon": ext.icon,
"shortDescription": ext.short_description,
"stars": ext.stars,
"isFeatured": ext.featured,
"dependencies": ext.dependencies,
"isInstalled": ext.id in installed_exts_ids,
"hasDatabaseTables": ext.id in db_version,
2023-01-17 17:34:50 +02:00
"isAvailable": ext.id in all_extensions,
"isAdminOnly": ext.id in settings.lnbits_admin_extensions,
2023-01-21 15:07:40 +00:00
"isActive": ext.id not in inactive_extensions,
"latestRelease": dict(ext.latest_release)
if ext.latest_release
else None,
2023-01-17 14:51:09 +02:00
"installedRelease": dict(ext.installed_release)
if ext.installed_release
else None,
},
installable_exts,
)
)
return template_renderer().TemplateResponse(
"core/extensions.html",
{
"request": request,
"user": user.dict(),
"extensions": extensions,
},
)
except Exception as e:
logger.warning(e)
raise HTTPException(status_code=HTTPStatus.INTERNAL_SERVER_ERROR, detail=str(e))
@core_html_routes.get(
"/wallet",
response_class=HTMLResponse,
description="""
Args:
just **wallet_name**: create a new user, then create a new wallet for user with wallet_name<br>
just **user_id**: return the first user wallet or create one if none found (with default wallet_name)<br>
**user_id** and **wallet_name**: create a new wallet for user with wallet_name<br>
**user_id** and **wallet_id**: return that wallet if user is the owner<br>
nothing: create everything<br>
""",
)
2021-10-17 18:33:29 +01:00
async def wallet(
request: Request = Query(None),
nme: Optional[str] = Query(None),
usr: Optional[UUID4] = Query(None),
wal: Optional[UUID4] = Query(None),
2021-10-17 18:33:29 +01:00
):
user_id = usr.hex if usr else None
wallet_id = wal.hex if wal else None
2021-08-21 19:04:10 +01:00
wallet_name = nme
2022-04-18 14:25:06 +01:00
if not user_id:
new_user = await create_account()
user = await get_user(new_user.id)
assert user, "Newly created user has to exist."
logger.info(f"Create user {user.id}")
else:
user = await get_user(user_id)
if not user:
2021-10-17 18:33:29 +01:00
return template_renderer().TemplateResponse(
"error.html", {"request": request, "err": "User does not exist."}
)
if (
len(settings.lnbits_allowed_users) > 0
and user_id not in settings.lnbits_allowed_users
2022-12-23 11:53:45 +01:00
and user_id not in settings.lnbits_admin_users
and user_id != settings.super_user
):
2022-09-21 15:37:07 +01:00
return template_renderer().TemplateResponse(
"error.html", {"request": request, "err": "User not authorized."}
)
if user_id == settings.super_user or user_id in settings.lnbits_admin_users:
2022-09-21 15:37:07 +01:00
user.admin = True
2023-01-04 10:41:56 +01:00
if user_id == settings.super_user:
user.super_user = True
2022-12-23 11:53:45 +01:00
if not wallet_id:
if user.wallets and not wallet_name:
wallet = user.wallets[0]
else:
wallet = await create_wallet(user_id=user.id, wallet_name=wallet_name)
logger.info(
f"Created new wallet {wallet_name if wallet_name else '(no name)'} for user {user.id}"
)
2021-10-17 18:33:29 +01:00
return RedirectResponse(
f"/wallet?usr={user.id}&wal={wallet.id}",
2021-10-17 18:33:29 +01:00
status_code=status.HTTP_307_TEMPORARY_REDIRECT,
)
logger.debug(
f"Access {'user '+ user.id + ' ' if user else ''} {'wallet ' + wallet_name if wallet_name else ''}"
)
userwallet = user.get_wallet(wallet_id)
2022-07-25 13:27:31 +02:00
if not userwallet:
2021-10-17 18:33:29 +01:00
return template_renderer().TemplateResponse(
"error.html", {"request": request, "err": "Wallet not found"}
)
return template_renderer().TemplateResponse(
2021-10-17 18:33:29 +01:00
"core/wallet.html",
{
"request": request,
"user": user.dict(),
2022-07-25 13:27:31 +02:00
"wallet": userwallet.dict(),
"service_fee": settings.lnbits_service_fee,
"web_manifest": f"/manifest/{user.id}.webmanifest",
2021-10-17 18:33:29 +01:00
},
)
@core_html_routes.get("/withdraw", response_class=JSONResponse)
async def lnurl_full_withdraw(request: Request):
user = await get_user(request.query_params.get("usr"))
2021-04-17 18:27:15 -03:00
if not user:
2021-08-20 21:31:01 +01:00
return {"status": "ERROR", "reason": "User does not exist."}
2021-04-17 18:27:15 -03:00
wallet = user.get_wallet(request.query_params.get("wal"))
2021-04-17 18:27:15 -03:00
if not wallet:
2021-10-17 18:33:29 +01:00
return {"status": "ERROR", "reason": "Wallet does not exist."}
2021-04-17 18:27:15 -03:00
2021-08-20 21:31:01 +01:00
return {
2021-10-17 18:33:29 +01:00
"tag": "withdrawRequest",
"callback": url_for("/withdraw/cb", external=True, usr=user.id, wal=wallet.id),
"k1": "0",
"minWithdrawable": 1000 if wallet.withdrawable_balance else 0,
"maxWithdrawable": wallet.withdrawable_balance,
"defaultDescription": f"{settings.lnbits_site_title} balance withdraw from {wallet.id[0:5]}",
2021-10-17 18:33:29 +01:00
"balanceCheck": url_for("/withdraw", external=True, usr=user.id, wal=wallet.id),
}
2021-04-17 18:27:15 -03:00
@core_html_routes.get("/withdraw/cb", response_class=JSONResponse)
async def lnurl_full_withdraw_callback(request: Request):
user = await get_user(request.query_params.get("usr"))
2021-04-17 18:27:15 -03:00
if not user:
2021-08-20 21:31:01 +01:00
return {"status": "ERROR", "reason": "User does not exist."}
2021-04-17 18:27:15 -03:00
wallet = user.get_wallet(request.query_params.get("wal"))
2021-04-17 18:27:15 -03:00
if not wallet:
2021-08-20 21:31:01 +01:00
return {"status": "ERROR", "reason": "Wallet does not exist."}
2021-04-17 18:27:15 -03:00
pr = request.query_params.get("pr")
2021-04-17 18:27:15 -03:00
async def pay():
try:
await pay_invoice(wallet_id=wallet.id, payment_request=pr)
except:
pass
2021-04-17 18:27:15 -03:00
asyncio.create_task(pay())
2021-04-17 18:27:15 -03:00
balance_notify = request.query_params.get("balanceNotify")
2021-04-17 18:27:15 -03:00
if balance_notify:
await save_balance_notify(wallet.id, balance_notify)
2021-08-20 21:31:01 +01:00
return {"status": "OK"}
2021-04-17 18:27:15 -03:00
@core_html_routes.get("/deletewallet", response_class=RedirectResponse)
async def deletewallet(wal: str = Query(...), usr: str = Query(...)):
2021-10-25 19:26:21 +01:00
user = await get_user(usr)
if not user:
raise HTTPException(HTTPStatus.FORBIDDEN, "User not found.")
user_wallet_ids = [u.id for u in user.wallets]
2021-10-25 19:26:21 +01:00
if wal not in user_wallet_ids:
2021-08-29 19:38:42 +02:00
raise HTTPException(HTTPStatus.FORBIDDEN, "Not your wallet.")
else:
await delete_wallet(user_id=user.id, wallet_id=wal)
2021-10-25 19:26:21 +01:00
user_wallet_ids.remove(wal)
logger.debug("Deleted wallet {wal} of user {user.id}")
if user_wallet_ids:
2021-10-17 18:33:29 +01:00
return RedirectResponse(
url_for("/wallet", usr=user.id, wal=user_wallet_ids[0]),
2021-10-17 18:33:29 +01:00
status_code=status.HTTP_307_TEMPORARY_REDIRECT,
)
2021-10-17 18:33:29 +01:00
return RedirectResponse(
url_for("/"), status_code=status.HTTP_307_TEMPORARY_REDIRECT
)
@core_html_routes.get("/withdraw/notify/{service}")
async def lnurl_balance_notify(request: Request, service: str):
bc = await get_balance_check(request.query_params.get("wal"), service)
2021-04-17 18:27:15 -03:00
if bc:
2022-07-19 18:51:35 +02:00
await redeem_lnurl_withdraw(bc.wallet, bc.url)
2021-04-17 18:27:15 -03:00
2022-06-01 14:53:05 +02:00
@core_html_routes.get(
"/lnurlwallet", response_class=RedirectResponse, name="core.lnurlwallet"
)
async def lnurlwallet(request: Request):
async with db.connect() as conn:
account = await create_account(conn=conn)
user = await get_user(account.id, conn=conn)
assert user, "Newly created user not found."
wallet = await create_wallet(user_id=user.id, conn=conn)
asyncio.create_task(
redeem_lnurl_withdraw(
2021-08-29 19:38:42 +02:00
wallet.id,
request.query_params.get("lightning"),
2021-08-29 19:38:42 +02:00
"LNbits initial funding: voucher redeem.",
{"tag": "lnurlwallet"},
2021-10-17 18:33:29 +01:00
5, # wait 5 seconds before sending the invoice to the service
)
)
2021-10-17 18:33:29 +01:00
return RedirectResponse(
f"/wallet?usr={user.id}&wal={wallet.id}",
2021-10-17 18:33:29 +01:00
status_code=status.HTTP_307_TEMPORARY_REDIRECT,
)
@core_html_routes.get("/service-worker.js", response_class=FileResponse)
async def service_worker():
return FileResponse("lnbits/core/static/js/service-worker.js")
2022-07-05 09:14:24 -06:00
@core_html_routes.get("/manifest/{usr}.webmanifest")
async def manifest(usr: str):
user = await get_user(usr)
if not user:
2021-09-11 11:02:48 +02:00
raise HTTPException(status_code=HTTPStatus.NOT_FOUND)
2021-08-20 21:31:01 +01:00
return {
"short_name": settings.lnbits_site_title,
"name": settings.lnbits_site_title + " Wallet",
2021-10-17 18:33:29 +01:00
"icons": [
{
"src": settings.lnbits_custom_logo
if settings.lnbits_custom_logo
2022-07-05 09:14:24 -06:00
else "https://cdn.jsdelivr.net/gh/lnbits/lnbits@0.3.0/docs/logos/lnbits.png",
2021-10-17 18:33:29 +01:00
"type": "image/png",
"sizes": "900x900",
}
],
"start_url": f"/wallet?usr={usr}&wal={user.wallets[0].id}",
"background_color": "#1F2234",
"description": "Bitcoin Lightning Wallet",
2021-10-17 18:33:29 +01:00
"display": "standalone",
"scope": "/",
"theme_color": "#1F2234",
2021-10-17 18:33:29 +01:00
"shortcuts": [
{
"name": wallet.name,
"short_name": wallet.name,
"description": wallet.name,
"url": f"/wallet?usr={usr}&wal={wallet.id}",
2021-10-17 18:33:29 +01:00
}
for wallet in user.wallets
],
}
@core_html_routes.get("/admin", response_class=HTMLResponse)
async def index(request: Request, user: User = Depends(check_admin)):
if not settings.lnbits_admin_ui:
raise HTTPException(status_code=HTTPStatus.NOT_FOUND)
WALLET = get_wallet_class()
_, balance = await WALLET.status()
return template_renderer().TemplateResponse(
"admin/index.html",
{
"request": request,
"user": user.dict(),
"settings": settings.dict(),
"balance": balance,
"currencies": list(currencies.keys()),
},
)
@core_html_routes.get("/uuidv4/{hex_value}")
async def hex_to_uuid4(hex_value: str):
try:
user_id = to_valid_user_id(hex_value).hex
return RedirectResponse(url=f"/wallet?usr={user_id}")
except Exception as e:
raise HTTPException(status_code=HTTPStatus.BAD_REQUEST, detail=str(e))
async def toggle_extension(extension_to_enable, extension_to_disable, user_id):
if extension_to_enable and extension_to_disable:
raise HTTPException(
HTTPStatus.BAD_REQUEST, "You can either `enable` or `disable` an extension."
)
# check if extension exists
if extension_to_enable or extension_to_disable:
ext = extension_to_enable or extension_to_disable
if ext not in [e.code for e in get_valid_extensions()]:
raise HTTPException(
HTTPStatus.BAD_REQUEST, f"Extension '{ext}' doesn't exist."
)
if extension_to_enable:
logger.info(f"Enabling extension: {extension_to_enable} for user {user_id}")
await update_user_extension(
user_id=user_id, extension=extension_to_enable, active=True
)
elif extension_to_disable:
logger.info(f"Disabling extension: {extension_to_disable} for user {user_id}")
await update_user_extension(
user_id=user_id, extension=extension_to_disable, active=False
)