lnbits-legend/lnbits/extensions/withdraw/lnurl.py

161 lines
5 KiB
Python
Raw Normal View History

2021-11-17 21:20:48 +00:00
import json
from datetime import datetime
2021-10-18 10:58:09 +01:00
from http import HTTPStatus
2021-10-18 10:58:09 +01:00
import shortuuid # type: ignore
from fastapi import HTTPException
from fastapi.param_functions import Query
2021-10-08 17:10:25 +01:00
from starlette.requests import Request
2021-11-10 21:12:47 +00:00
from starlette.responses import HTMLResponse # type: ignore
2021-10-18 10:58:09 +01:00
from lnbits.core.services import pay_invoice
from . import withdraw_ext
from .crud import get_withdraw_link_by_hash, update_withdraw_link
# FOR LNURLs WHICH ARE NOT UNIQUE
2021-10-17 18:33:29 +01:00
@withdraw_ext.get(
"/api/v1/lnurl/{unique_hash}",
2021-11-10 21:12:47 +00:00
response_class=HTMLResponse,
2021-10-17 18:33:29 +01:00
name="withdraw.api_lnurl_response",
)
2021-10-08 17:10:25 +01:00
async def api_lnurl_response(request: Request, unique_hash):
link = await get_withdraw_link_by_hash(unique_hash)
if not link:
2021-09-30 14:42:04 +01:00
raise HTTPException(
2021-10-17 18:33:29 +01:00
status_code=HTTPStatus.NOT_FOUND, detail="Withdraw link does not exist."
)
if link.is_spent:
2022-05-18 11:08:52 +01:00
raise HTTPException(status_code=HTTPStatus.NOT_FOUND, detail="Withdraw is spent.")
2021-11-10 21:12:47 +00:00
url = request.url_for("withdraw.api_lnurl_callback", unique_hash=link.unique_hash)
withdrawResponse = {
"tag": "withdrawRequest",
"callback": url,
"k1": link.k1,
"minWithdrawable": link.min_withdrawable * 1000,
"maxWithdrawable": link.max_withdrawable * 1000,
"defaultDescription": link.title,
}
return json.dumps(withdrawResponse)
2021-11-10 21:12:47 +00:00
# CALLBACK
2021-11-26 05:58:20 +00:00
@withdraw_ext.get("/api/v1/lnurl/cb/{unique_hash}", name="withdraw.api_lnurl_callback")
2021-10-17 18:33:29 +01:00
async def api_lnurl_callback(
2022-05-18 11:08:52 +01:00
unique_hash, request: Request, k1: str = Query(...), pr: str = Query(...), id_unique_hash=None
2021-10-17 18:33:29 +01:00
):
link = await get_withdraw_link_by_hash(unique_hash)
now = int(datetime.now().timestamp())
if not link:
2021-09-30 14:42:04 +01:00
raise HTTPException(
2021-11-17 21:20:48 +00:00
status_code=HTTPStatus.NOT_FOUND, detail="LNURL-withdraw not found"
)
if link.is_spent:
2022-05-18 11:08:52 +01:00
raise HTTPException(status_code=HTTPStatus.NOT_FOUND, detail="Withdraw is spent.")
2021-11-04 12:46:22 +00:00
if link.k1 != k1:
2022-05-18 11:08:52 +01:00
raise HTTPException(status_code=HTTPStatus.NOT_FOUND, detail="Bad request.")
2021-11-04 12:46:22 +00:00
if now < link.open_time:
2021-09-30 14:42:04 +01:00
return {"status": "ERROR", "reason": f"Wait {link.open_time - now} seconds."}
2021-11-04 12:46:22 +00:00
2022-05-18 11:08:52 +01:00
usescsv = ""
try:
for x in range(1, link.uses - link.used):
usecv = link.usescsv.split(",")
usescsv += "," + str(usecv[x])
usecsvback = usescsv
2022-05-18 11:08:52 +01:00
found = False
if id_unique_hash is not None:
useslist = link.usescsv.split(",")
for ind, x in enumerate(useslist):
tohash = link.id + link.unique_hash + str(x)
if id_unique_hash == shortuuid.uuid(name=tohash):
found = True
useslist.pop(ind)
usescsv = ','.join(useslist)
if not found:
raise HTTPException(
status_code=HTTPStatus.NOT_FOUND, detail="LNURL-withdraw not found."
)
else:
usescsv = usescsv[1:]
changesback = {
"open_time": link.wait_time,
"used": link.used,
"usescsv": usecsvback,
}
changes = {
"open_time": link.wait_time + now,
"used": link.used + 1,
"usescsv": usescsv,
}
await update_withdraw_link(link.id, **changes)
2021-11-26 05:58:20 +00:00
payment_request = pr
await pay_invoice(
wallet_id=link.wallet,
payment_request=payment_request,
max_sat=link.max_withdrawable,
extra={"tag": "withdraw"},
)
2021-11-04 12:41:31 +00:00
return {"status": "OK"}
except Exception as e:
2021-11-04 12:46:22 +00:00
await update_withdraw_link(link.id, **changesback)
return {"status": "ERROR", "reason": "Link not working"}
2021-11-17 21:20:48 +00:00
# FOR LNURLs WHICH ARE UNIQUE
@withdraw_ext.get(
"/api/v1/lnurl/{unique_hash}/{id_unique_hash}",
response_class=HTMLResponse,
name="withdraw.api_lnurl_multi_response",
)
async def api_lnurl_multi_response(request: Request, unique_hash, id_unique_hash):
link = await get_withdraw_link_by_hash(unique_hash)
if not link:
raise HTTPException(
2022-05-18 11:08:52 +01:00
status_code=HTTPStatus.NOT_FOUND, detail="LNURL-withdraw not found."
2021-11-17 21:20:48 +00:00
)
if link.is_spent:
2022-05-18 11:08:52 +01:00
raise HTTPException(status_code=HTTPStatus.NOT_FOUND, detail="Withdraw is spent.")
2021-11-17 21:20:48 +00:00
useslist = link.usescsv.split(",")
found = False
for x in useslist:
tohash = link.id + link.unique_hash + str(x)
if id_unique_hash == shortuuid.uuid(name=tohash):
found = True
2022-05-18 11:08:52 +01:00
2021-11-17 21:20:48 +00:00
if not found:
raise HTTPException(
2022-05-18 11:08:52 +01:00
status_code=HTTPStatus.NOT_FOUND, detail="LNURL-withdraw not found."
2021-11-17 21:20:48 +00:00
)
url = request.url_for("withdraw.api_lnurl_callback", unique_hash=link.unique_hash)
withdrawResponse = {
"tag": "withdrawRequest",
2022-05-18 11:08:52 +01:00
"callback": url + "?id_unique_hash=" + id_unique_hash,
2021-11-17 21:20:48 +00:00
"k1": link.k1,
"minWithdrawable": link.min_withdrawable * 1000,
"maxWithdrawable": link.max_withdrawable * 1000,
"defaultDescription": link.title,
}
return json.dumps(withdrawResponse)