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

109 lines
3.5 KiB
Python
Raw Normal View History

2021-10-28 17:02:07 +01:00
import hashlib
import math
from http import HTTPStatus
from os import name
from fastapi.exceptions import HTTPException
from fastapi.params import Query
from lnurl import LnurlErrorResponse, LnurlPayActionResponse, LnurlPayResponse
from starlette.requests import Request # type: ignore
from lnbits.core.services import create_invoice
from . import livestream_ext
from .crud import get_livestream, get_livestream_by_track, get_track
@livestream_ext.get("/lnurl/{ls_id}", name="livestream.lnurl_livestream")
async def lnurl_livestream(ls_id, request: Request):
ls = await get_livestream(ls_id)
if not ls:
raise HTTPException(
2021-11-12 04:14:55 +00:00
status_code=HTTPStatus.NOT_FOUND, detail="Livestream not found."
2021-10-28 17:02:07 +01:00
)
track = await get_track(ls.current_track)
if not track:
raise HTTPException(
2021-11-12 04:14:55 +00:00
status_code=HTTPStatus.NOT_FOUND, detail="This livestream is offline."
2021-10-28 17:02:07 +01:00
)
resp = LnurlPayResponse(
2021-11-12 04:14:55 +00:00
callback=request.url_for("livestream.lnurl_callback", track_id=track.id),
2021-10-28 17:02:07 +01:00
min_sendable=track.min_sendable,
max_sendable=track.max_sendable,
metadata=await track.lnurlpay_metadata(),
)
params = resp.dict()
params["commentAllowed"] = 300
return params
@livestream_ext.get("/lnurl/t/{track_id}", name="livestream.lnurl_track")
async def lnurl_track(track_id, request: Request):
track = await get_track(track_id)
if not track:
2021-11-12 04:14:55 +00:00
raise HTTPException(status_code=HTTPStatus.NOT_FOUND, detail="Track not found.")
2021-10-28 17:02:07 +01:00
resp = LnurlPayResponse(
2021-11-12 04:14:55 +00:00
callback=request.url_for("livestream.lnurl_callback", track_id=track.id),
2021-10-28 17:02:07 +01:00
min_sendable=track.min_sendable,
max_sendable=track.max_sendable,
metadata=await track.lnurlpay_metadata(),
)
params = resp.dict()
params["commentAllowed"] = 300
return params
@livestream_ext.get("/lnurl/cb/{track_id}", name="livestream.lnurl_callback")
2021-11-12 04:14:55 +00:00
async def lnurl_callback(
track_id, request: Request, amount: int = Query(...), comment: str = Query("")
):
2021-10-28 17:02:07 +01:00
track = await get_track(track_id)
if not track:
2021-11-12 04:14:55 +00:00
raise HTTPException(status_code=HTTPStatus.NOT_FOUND, detail="Track not found.")
2021-10-28 17:02:07 +01:00
amount_received = int(amount or 0)
if amount_received < track.min_sendable:
return LnurlErrorResponse(
2021-11-12 04:14:55 +00:00
reason=f"Amount {round(amount_received / 1000)} is smaller than minimum {math.floor(track.min_sendable)}."
).dict()
2021-10-28 17:02:07 +01:00
elif track.max_sendable < amount_received:
return LnurlErrorResponse(
2021-11-12 04:14:55 +00:00
reason=f"Amount {round(amount_received / 1000)} is greater than maximum {math.floor(track.max_sendable)}."
).dict()
2021-10-28 17:02:07 +01:00
if len(comment or "") > 300:
return LnurlErrorResponse(
2021-11-12 04:14:55 +00:00
reason=f"Got a comment with {len(comment)} characters, but can only accept 300"
).dict()
2021-10-28 17:02:07 +01:00
ls = await get_livestream_by_track(track_id)
payment_hash, payment_request = await create_invoice(
wallet_id=ls.wallet,
amount=int(amount_received / 1000),
memo=await track.fullname(),
description_hash=hashlib.sha256(
(await track.lnurlpay_metadata()).encode("utf-8")
).digest(),
extra={"tag": "livestream", "track": track.id, "comment": comment},
)
if amount_received < track.price_msat:
success_action = None
else:
success_action = track.success_action(payment_hash, request=request)
resp = LnurlPayActionResponse(
2021-11-12 04:14:55 +00:00
pr=payment_request, success_action=success_action, routes=[]
2021-10-28 17:02:07 +01:00
)
return resp.dict()