From 6df8494ed65030f647cc49552633e62681cf336a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?dni=20=E2=9A=A1?= Date: Wed, 15 Feb 2023 10:58:36 +0100 Subject: [PATCH] remove livestream --- lnbits/extensions/livestream/README.md | 45 --- lnbits/extensions/livestream/__init__.py | 35 -- lnbits/extensions/livestream/config.json | 10 - lnbits/extensions/livestream/crud.py | 203 ----------- lnbits/extensions/livestream/lnurl.py | 118 ------- lnbits/extensions/livestream/migrations.py | 39 --- lnbits/extensions/livestream/models.py | 103 ------ .../livestream/static/image/livestream.png | Bin 68286 -> 0 bytes .../extensions/livestream/static/js/index.js | 216 ------------ lnbits/extensions/livestream/tasks.py | 70 ---- .../templates/livestream/_api_docs.html | 150 -------- .../templates/livestream/index.html | 323 ------------------ lnbits/extensions/livestream/views.py | 48 --- lnbits/extensions/livestream/views_api.py | 101 ------ 14 files changed, 1461 deletions(-) delete mode 100644 lnbits/extensions/livestream/README.md delete mode 100644 lnbits/extensions/livestream/__init__.py delete mode 100644 lnbits/extensions/livestream/config.json delete mode 100644 lnbits/extensions/livestream/crud.py delete mode 100644 lnbits/extensions/livestream/lnurl.py delete mode 100644 lnbits/extensions/livestream/migrations.py delete mode 100644 lnbits/extensions/livestream/models.py delete mode 100644 lnbits/extensions/livestream/static/image/livestream.png delete mode 100644 lnbits/extensions/livestream/static/js/index.js delete mode 100644 lnbits/extensions/livestream/tasks.py delete mode 100644 lnbits/extensions/livestream/templates/livestream/_api_docs.html delete mode 100644 lnbits/extensions/livestream/templates/livestream/index.html delete mode 100644 lnbits/extensions/livestream/views.py delete mode 100644 lnbits/extensions/livestream/views_api.py diff --git a/lnbits/extensions/livestream/README.md b/lnbits/extensions/livestream/README.md deleted file mode 100644 index b2cf41d6..00000000 --- a/lnbits/extensions/livestream/README.md +++ /dev/null @@ -1,45 +0,0 @@ -# DJ Livestream - -## Help DJ's and music producers conduct music livestreams - -LNbits Livestream extension produces a static QR code that can be shown on screen while livestreaming a DJ set for example. If someone listening to the livestream likes a song and want to support the DJ and/or the producer he can scan the QR code with a LNURL-pay capable wallet. - -When scanned, the QR code sends up information about the song playing at the moment (name and the producer of that song). Also, if the user likes the song and would like to support the producer, he can send a tip and a message for that specific track. If the user sends an amount over a specific threshold they will be given a link to download it (optional). - -The revenue will be sent to a wallet created specifically for that producer, with optional revenue splitting between the DJ and the producer. - -[**Wallets supporting LNURL**](https://github.com/fiatjaf/awesome-lnurl#wallets) - -[![video tutorial livestream](http://img.youtube.com/vi/zDrSWShKz7k/0.jpg)](https://youtu.be/zDrSWShKz7k 'video tutorial offline shop') - -## Usage - -1. Start by adding a track\ - ![add new track](https://i.imgur.com/Cu0eGrW.jpg) - - set the producer, or choose an existing one - - set the track name - - define a minimum price where a user can download the track - - set the download URL, where user will be redirected if he tips the livestream and the tip is equal or above the set price\ - ![track settings](https://i.imgur.com/HTJYwcW.jpg) -2. Adjust the percentage of the pay you want to take from the user's tips. 10%, the default, means that the DJ will keep 10% of all the tips sent by users. The other 90% will go to an auto generated producer wallet\ - ![adjust percentage](https://i.imgur.com/9weHKAB.jpg) -3. For every different producer added, when adding tracks, a wallet is generated for them\ - ![producer wallet](https://i.imgur.com/YFIZ7Tm.jpg) -4. On the bottom of the LNbits DJ Livestream extension you'll find the static QR code ([LNURL-pay](https://github.com/lnbits/lnbits/blob/master/lnbits/extensions/lnurlp/README.md)) you can add to the livestream or if you're a street performer you can print it and have it displayed -5. After all tracks and producers are added, you can start "playing" songs\ - ![play tracks](https://i.imgur.com/7ytiBkq.jpg) -6. You'll see the current track playing and a green icon indicating active track also\ - ![active track](https://i.imgur.com/W1vBz54.jpg) -7. When a user scans the QR code, and sends a tip, you'll receive 10%, in the example case, in your wallet and the producer's wallet will get the rest. For example someone tips 100 sats, you'll get 10 sats and the producer will get 90 sats - - producer's wallet receiving 18 sats from 20 sats tips\ - ![producer wallet](https://i.imgur.com/OM9LawA.jpg) - -## Use cases - -You can print the QR code and display it on a live gig, a street performance, etc... OR you can use the QR as an overlay in an online stream of you playing music, doing a DJ set, making a podcast. - -You can use the extension's API to trigger updates for the current track, update fees, add tracks... - -## Sponsored by - -[![](https://cdn.shopify.com/s/files/1/0826/9235/files/cryptograffiti_logo_clear_background.png?v=1504730421)](https://cryptograffiti.com/) diff --git a/lnbits/extensions/livestream/__init__.py b/lnbits/extensions/livestream/__init__.py deleted file mode 100644 index accd61de..00000000 --- a/lnbits/extensions/livestream/__init__.py +++ /dev/null @@ -1,35 +0,0 @@ -import asyncio - -from fastapi import APIRouter -from fastapi.staticfiles import StaticFiles - -from lnbits.db import Database -from lnbits.helpers import template_renderer -from lnbits.tasks import catch_everything_and_restart - -db = Database("ext_livestream") - -livestream_static_files = [ - { - "path": "/livestream/static", - "app": StaticFiles(packages=[("lnbits", "extensions/livestream/static")]), - "name": "livestream_static", - } -] - -livestream_ext: APIRouter = APIRouter(prefix="/livestream", tags=["livestream"]) - - -def livestream_renderer(): - return template_renderer(["lnbits/extensions/livestream/templates"]) - - -from .lnurl import * # noqa: F401,F403 -from .tasks import wait_for_paid_invoices -from .views import * # noqa: F401,F403 -from .views_api import * # noqa: F401,F403 - - -def livestream_start(): - loop = asyncio.get_event_loop() - loop.create_task(catch_everything_and_restart(wait_for_paid_invoices)) diff --git a/lnbits/extensions/livestream/config.json b/lnbits/extensions/livestream/config.json deleted file mode 100644 index d2674e70..00000000 --- a/lnbits/extensions/livestream/config.json +++ /dev/null @@ -1,10 +0,0 @@ -{ - "name": "DJ Livestream", - "short_description": "Sell tracks and split revenue (lnurl-pay)", - "tile": "/livestream/static/image/livestream.png", - "contributors": [ - "fiatjaf", - "cryptograffiti" - ], - "hidden": false -} diff --git a/lnbits/extensions/livestream/crud.py b/lnbits/extensions/livestream/crud.py deleted file mode 100644 index 037851aa..00000000 --- a/lnbits/extensions/livestream/crud.py +++ /dev/null @@ -1,203 +0,0 @@ -from typing import List, Optional - -from lnbits.core.crud import create_account, create_wallet -from lnbits.db import SQLITE - -from . import db -from .models import Livestream, Producer, Track - - -async def create_livestream(*, wallet_id: str) -> int: - returning = "" if db.type == SQLITE else "RETURNING ID" - method = db.execute if db.type == SQLITE else db.fetchone - - result = await (method)( - f""" - INSERT INTO livestream.livestreams (wallet) - VALUES (?) - {returning} - """, - (wallet_id,), - ) - - if db.type == SQLITE: - return result._result_proxy.lastrowid - else: - return result[0] # type: ignore - - -async def get_livestream(ls_id: int) -> Optional[Livestream]: - row = await db.fetchone( - "SELECT * FROM livestream.livestreams WHERE id = ?", (ls_id,) - ) - return Livestream(**row) if row else None - - -async def get_livestream_by_track(track_id: int) -> Optional[Livestream]: - row = await db.fetchone( - """ - SELECT * FROM livestream.tracks WHERE tracks.id = ? - """, - (track_id,), - ) - row2 = await db.fetchone( - """ - SELECT * FROM livestream.livestreams WHERE livestreams.id = ? - """, - (row.livestream,), - ) - return Livestream(**row2) if row2 else None - - -async def get_or_create_livestream_by_wallet(wallet: str) -> Optional[Livestream]: - row = await db.fetchone( - "SELECT * FROM livestream.livestreams WHERE wallet = ?", (wallet,) - ) - - if not row: - # create on the fly - ls_id = await create_livestream(wallet_id=wallet) - return await get_livestream(ls_id) - - return Livestream(**row) if row else None - - -async def update_current_track(ls_id: int, track_id: Optional[int]): - await db.execute( - "UPDATE livestream.livestreams SET current_track = ? WHERE id = ?", - (track_id, ls_id), - ) - - -async def update_livestream_fee(ls_id: int, fee_pct: int): - await db.execute( - "UPDATE livestream.livestreams SET fee_pct = ? WHERE id = ?", (fee_pct, ls_id) - ) - - -async def add_track( - livestream: int, - name: str, - download_url: Optional[str], - price_msat: int, - producer: Optional[int], -) -> int: - result = await db.execute( - """ - INSERT INTO livestream.tracks (livestream, name, download_url, price_msat, producer) - VALUES (?, ?, ?, ?, ?) - """, - (livestream, name, download_url, price_msat, producer), - ) - return result._result_proxy.lastrowid - - -async def update_track( - livestream: int, - track_id: int, - name: str, - download_url: Optional[str], - price_msat: int, - producer: int, -) -> int: - result = await db.execute( - """ - UPDATE livestream.tracks SET - name = ?, - download_url = ?, - price_msat = ?, - producer = ? - WHERE livestream = ? AND id = ? - """, - (name, download_url, price_msat, producer, livestream, track_id), - ) - return result._result_proxy.lastrowid - - -async def get_track(track_id: Optional[int]) -> Optional[Track]: - if not track_id: - return None - - row = await db.fetchone( - """ - SELECT id, download_url, price_msat, name, producer - FROM livestream.tracks WHERE id = ? - """, - (track_id,), - ) - return Track(**row) if row else None - - -async def get_tracks(livestream: int) -> List[Track]: - rows = await db.fetchall( - """ - SELECT id, download_url, price_msat, name, producer - FROM livestream.tracks WHERE livestream = ? - """, - (livestream,), - ) - return [Track(**row) for row in rows] - - -async def delete_track_from_livestream(livestream: int, track_id: int): - await db.execute( - """ - DELETE FROM livestream.tracks WHERE livestream = ? AND id = ? - """, - (livestream, track_id), - ) - - -async def add_producer(livestream: int, name: str) -> int: - name = name.strip() - - existing = await db.fetchall( - """ - SELECT id FROM livestream.producers - WHERE livestream = ? AND lower(name) = ? - """, - (livestream, name.lower()), - ) - if existing: - return existing[0].id - - user = await create_account() - wallet = await create_wallet(user_id=user.id, wallet_name="livestream: " + name) - - returning = "" if db.type == SQLITE else "RETURNING ID" - method = db.execute if db.type == SQLITE else db.fetchone - - result = await method( - f""" - INSERT INTO livestream.producers (livestream, name, "user", wallet) - VALUES (?, ?, ?, ?) - {returning} - """, - (livestream, name, user.id, wallet.id), - ) - if db.type == SQLITE: - return result._result_proxy.lastrowid - else: - return result[0] # type: ignore - - -async def get_producer(producer_id: int) -> Optional[Producer]: - row = await db.fetchone( - """ - SELECT id, "user", wallet, name - FROM livestream.producers WHERE id = ? - """, - (producer_id,), - ) - return Producer(**row) if row else None - - -async def get_producers(livestream: int) -> List[Producer]: - rows = await db.fetchall( - """ - SELECT id, "user", wallet, name - FROM livestream.producers WHERE livestream = ? - """, - (livestream,), - ) - return [Producer(**row) for row in rows] diff --git a/lnbits/extensions/livestream/lnurl.py b/lnbits/extensions/livestream/lnurl.py deleted file mode 100644 index e3e1b1be..00000000 --- a/lnbits/extensions/livestream/lnurl.py +++ /dev/null @@ -1,118 +0,0 @@ -import math -from http import HTTPStatus - -from fastapi import HTTPException, Query, Request -from lnurl import LnurlErrorResponse, LnurlPayActionResponse, LnurlPayResponse -from lnurl.models import ClearnetUrl, LightningInvoice, MilliSatoshi - -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( - status_code=HTTPStatus.NOT_FOUND, detail="Livestream not found." - ) - - track = await get_track(ls.current_track) - if not track: - raise HTTPException( - status_code=HTTPStatus.NOT_FOUND, detail="This livestream is offline." - ) - - resp = LnurlPayResponse( - callback=ClearnetUrl( - request.url_for("livestream.lnurl_callback", track_id=track.id), - scheme="https", - ), - minSendable=MilliSatoshi(track.min_sendable), - maxSendable=MilliSatoshi(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: - raise HTTPException(status_code=HTTPStatus.NOT_FOUND, detail="Track not found.") - - resp = LnurlPayResponse( - callback=ClearnetUrl( - request.url_for("livestream.lnurl_callback", track_id=track.id), - scheme="https", - ), - minSendable=MilliSatoshi(track.min_sendable), - maxSendable=MilliSatoshi(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") -async def lnurl_callback( - track_id, request: Request, amount: int = Query(...), comment: str = Query("") -): - track = await get_track(track_id) - if not track: - raise HTTPException(status_code=HTTPStatus.NOT_FOUND, detail="Track not found.") - - amount_received = int(amount or 0) - - if amount_received < track.min_sendable: - return LnurlErrorResponse( - reason=f"Amount {round(amount_received / 1000)} is smaller than minimum {math.floor(track.min_sendable)}." - ).dict() - elif track.max_sendable < amount_received: - return LnurlErrorResponse( - reason=f"Amount {round(amount_received / 1000)} is greater than maximum {math.floor(track.max_sendable)}." - ).dict() - - if len(comment or "") > 300: - return LnurlErrorResponse( - reason=f"Got a comment with {len(comment)} characters, but can only accept 300" - ).dict() - - ls = await get_livestream_by_track(track_id) - assert ls - - extra_amount = amount_received - int(amount_received * (100 - ls.fee_pct) / 100) - - payment_hash, payment_request = await create_invoice( - wallet_id=ls.wallet, - amount=int(amount_received / 1000), - memo=await track.fullname(), - unhashed_description=(await track.lnurlpay_metadata()).encode(), - extra={ - "tag": "livestream", - "track": track.id, - "comment": comment, - "amount": int(extra_amount / 1000), - }, - ) - - assert track.price_msat - if amount_received < track.price_msat: - success_action = None - else: - success_action = track.success_action(payment_hash, request=request) - - resp = LnurlPayActionResponse( - pr=LightningInvoice(payment_request), successAction=success_action, routes=[] - ) - - return resp.dict() diff --git a/lnbits/extensions/livestream/migrations.py b/lnbits/extensions/livestream/migrations.py deleted file mode 100644 index fb664ab1..00000000 --- a/lnbits/extensions/livestream/migrations.py +++ /dev/null @@ -1,39 +0,0 @@ -async def m001_initial(db): - """ - Initial livestream tables. - """ - await db.execute( - f""" - CREATE TABLE livestream.livestreams ( - id {db.serial_primary_key}, - wallet TEXT NOT NULL, - fee_pct INTEGER NOT NULL DEFAULT 10, - current_track INTEGER - ); - """ - ) - - await db.execute( - f""" - CREATE TABLE livestream.producers ( - livestream INTEGER NOT NULL REFERENCES {db.references_schema}livestreams (id), - id {db.serial_primary_key}, - "user" TEXT NOT NULL, - wallet TEXT NOT NULL, - name TEXT NOT NULL - ); - """ - ) - - await db.execute( - f""" - CREATE TABLE livestream.tracks ( - livestream INTEGER NOT NULL REFERENCES {db.references_schema}livestreams (id), - id {db.serial_primary_key}, - download_url TEXT, - price_msat INTEGER NOT NULL DEFAULT 0, - name TEXT, - producer INTEGER REFERENCES {db.references_schema}producers (id) NOT NULL - ); - """ - ) diff --git a/lnbits/extensions/livestream/models.py b/lnbits/extensions/livestream/models.py deleted file mode 100644 index 31d3f6eb..00000000 --- a/lnbits/extensions/livestream/models.py +++ /dev/null @@ -1,103 +0,0 @@ -import json -from sqlite3 import Row -from typing import Optional - -from fastapi import Query, Request -from lnurl import Lnurl -from lnurl import encode as lnurl_encode -from lnurl.models import ClearnetUrl, Max144Str, UrlAction -from lnurl.types import LnurlPayMetadata -from pydantic import BaseModel - - -class CreateTrack(BaseModel): - name: str = Query(...) - download_url: str = Query(None) - price_msat: int = Query(None, ge=0) - producer_id: str = Query(None) - producer_name: str = Query(None) - - -class Livestream(BaseModel): - id: int - wallet: str - fee_pct: int - current_track: Optional[int] - - def lnurl(self, request: Request) -> Lnurl: - url = request.url_for("livestream.lnurl_livestream", ls_id=self.id) - return lnurl_encode(url) - - @classmethod - def from_row(cls, row: Row): - return cls(**dict(row)) - - -class Track(BaseModel): - id: int - download_url: Optional[str] - price_msat: int = 0 - name: str - producer: int - - @classmethod - def from_row(cls, row: Row): - return cls(**dict(row)) - - @property - def min_sendable(self) -> int: - return min(100_000, self.price_msat or 100_000) - - @property - def max_sendable(self) -> int: - return max(50_000_000, self.price_msat * 5) - - def lnurl(self, request: Request) -> Lnurl: - url = request.url_for("livestream.lnurl_track", track_id=self.id) - return lnurl_encode(url) - - async def fullname(self) -> str: - from .crud import get_producer - - producer = await get_producer(self.producer) - if producer: - producer_name = producer.name - else: - producer_name = "unknown author" - - return f"'{self.name}', from {producer_name}." - - async def lnurlpay_metadata(self) -> LnurlPayMetadata: - description = ( - await self.fullname() - ) + " Like this track? Send some sats in appreciation." - - if self.download_url: - description += f" Send {round(self.price_msat/1000)} sats or more and you can download it." - - return LnurlPayMetadata(json.dumps([["text/plain", description]])) - - def success_action( - self, payment_hash: str, request: Request - ) -> Optional[UrlAction]: - if not self.download_url: - return None - - url = request.url_for("livestream.track_redirect_download", track_id=self.id) - url_with_query = f"{url}?p={payment_hash}" - - return UrlAction( - url=ClearnetUrl(url_with_query, scheme="https"), - description=Max144Str(f"Download the track {self.name}!"), - ) - - -class Producer(BaseModel): - id: int - user: str - wallet: str - name: str - - @classmethod - def from_row(cls, row: Row): - return cls(**dict(row)) diff --git a/lnbits/extensions/livestream/static/image/livestream.png b/lnbits/extensions/livestream/static/image/livestream.png deleted file mode 100644 index 7d53d5661eefda1c4ed0a187c29eda68a383b686..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 68286 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 O0000 {}, - selectedWallet: null, - nextCurrentTrack: null, - livestream: { - tracks: [], - producers: [] - }, - trackDialog: { - show: false, - data: {} - } - } - }, - computed: { - sortedTracks() { - return this.livestream.tracks.sort((a, b) => a.name - b.name) - }, - tracksMap() { - return Object.fromEntries( - this.livestream.tracks.map(track => [track.id, track]) - ) - }, - producersMap() { - return Object.fromEntries( - this.livestream.producers.map(prod => [prod.id, prod]) - ) - } - }, - methods: { - getTrackLabel(trackId) { - if (!trackId) return - let track = this.tracksMap[trackId] - return `${track.name}, ${this.producersMap[track.producer].name}` - }, - disabledAddTrackButton() { - return ( - !this.trackDialog.data.name || - this.trackDialog.data.name.length === 0 || - !this.trackDialog.data.producer || - this.trackDialog.data.producer.length === 0 - ) - }, - changedWallet(wallet) { - this.selectedWallet = wallet - this.loadLivestream() - this.startPaymentNotifier() - }, - loadLivestream() { - LNbits.api - .request( - 'GET', - '/livestream/api/v1/livestream', - this.selectedWallet.inkey - ) - .then(response => { - this.livestream = response.data - this.nextCurrentTrack = this.livestream.current_track - }) - .catch(err => { - LNbits.utils.notifyApiError(err) - }) - }, - startPaymentNotifier() { - this.cancelListener() - - this.cancelListener = LNbits.events.onInvoicePaid( - this.selectedWallet, - payment => { - let satoshiAmount = Math.round(payment.amount / 1000) - let trackName = ( - this.tracksMap[payment.extra.track] || {name: '[unknown]'} - ).name - - this.$q.notify({ - message: `Someone paid ${satoshiAmount} sat for the track ${trackName}.`, - caption: payment.extra.comment - ? `"${payment.extra.comment}"` - : undefined, - color: 'secondary', - html: true, - timeout: 0, - actions: [{label: 'Dismiss', color: 'white', handler: () => {}}] - }) - } - ) - }, - addTrack() { - let {id, name, producer, price_sat, download_url} = this.trackDialog.data - - const [method, path] = id - ? ['PUT', `/livestream/api/v1/livestream/tracks/${id}`] - : ['POST', '/livestream/api/v1/livestream/tracks'] - - LNbits.api - .request(method, path, this.selectedWallet.inkey, { - download_url: - download_url && download_url.length > 0 ? download_url : undefined, - name, - price_msat: price_sat * 1000 || 0, - producer_name: typeof producer === 'string' ? producer : undefined, - producer_id: typeof producer === 'object' ? producer.id : undefined - }) - .then(response => { - this.$q.notify({ - message: `Track '${this.trackDialog.data.name}' added.`, - timeout: 700 - }) - this.loadLivestream() - this.trackDialog.show = false - this.trackDialog.data = {} - }) - .catch(err => { - LNbits.utils.notifyApiError(err) - }) - }, - openAddTrackDialog() { - this.trackDialog.show = true - this.trackDialog.data = {} - }, - openUpdateDialog(itemId) { - this.trackDialog.show = true - let item = this.livestream.tracks.find(item => item.id === itemId) - this.trackDialog.data = { - ...item, - producer: this.livestream.producers.find( - prod => prod.id === item.producer - ), - price_sat: Math.round(item.price_msat / 1000) - } - }, - deleteTrack(trackId) { - LNbits.utils - .confirmDialog('Are you sure you want to delete this track?') - .onOk(() => { - LNbits.api - .request( - 'DELETE', - '/livestream/api/v1/livestream/tracks/' + trackId, - this.selectedWallet.inkey - ) - .then(response => { - this.$q.notify({ - message: `Track deleted`, - timeout: 700 - }) - this.livestream.tracks.splice( - this.livestream.tracks.findIndex(track => track.id === trackId), - 1 - ) - }) - .catch(err => { - LNbits.utils.notifyApiError(err) - }) - }) - }, - updateCurrentTrack(track) { - console.log(this.nextCurrentTrack, this.livestream) - if (this.livestream.current_track === track) { - // if clicking the same, stop it - track = 0 - } - - LNbits.api - .request( - 'PUT', - '/livestream/api/v1/livestream/track/' + track, - this.selectedWallet.inkey - ) - .then(() => { - this.livestream.current_track = track - this.nextCurrentTrack = track - this.$q.notify({ - message: `Current track updated.`, - timeout: 700 - }) - }) - .catch(err => { - LNbits.utils.notifyApiError(err) - }) - }, - updateFeePct() { - LNbits.api - .request( - 'PUT', - '/livestream/api/v1/livestream/fee/' + this.livestream.fee_pct, - this.selectedWallet.inkey - ) - .then(() => { - this.$q.notify({ - message: `Percentage updated.`, - timeout: 700 - }) - }) - .catch(err => { - LNbits.utils.notifyApiError(err) - }) - }, - producerAdded(added, cb) { - cb(added) - } - }, - created() { - this.selectedWallet = this.g.user.wallets[0] - this.loadLivestream() - this.startPaymentNotifier() - } -}) diff --git a/lnbits/extensions/livestream/tasks.py b/lnbits/extensions/livestream/tasks.py deleted file mode 100644 index 4489dad9..00000000 --- a/lnbits/extensions/livestream/tasks.py +++ /dev/null @@ -1,70 +0,0 @@ -import asyncio - -from loguru import logger - -from lnbits.core.models import Payment -from lnbits.core.services import create_invoice, pay_invoice -from lnbits.helpers import get_current_extension_name -from lnbits.tasks import register_invoice_listener - -from .crud import get_livestream_by_track, get_producer, get_track - - -async def wait_for_paid_invoices(): - invoice_queue = asyncio.Queue() - register_invoice_listener(invoice_queue, get_current_extension_name()) - - while True: - payment = await invoice_queue.get() - await on_invoice_paid(payment) - - -async def on_invoice_paid(payment: Payment) -> None: - - if payment.extra.get("tag") != "livestream": - # not a livestream invoice - return - - track = await get_track(payment.extra.get("track", -1)) - if not track: - logger.error("this should never happen", payment) - return - - if payment.extra.get("shared_with"): - logger.error("payment was shared already", payment) - return - - producer = await get_producer(track.producer) - assert producer, f"track {track.id} is not associated with a producer" - - ls = await get_livestream_by_track(track.id) - assert ls, f"track {track.id} is not associated with a livestream" - - amount = int(payment.amount * (100 - ls.fee_pct) / 100) - - payment_hash, payment_request = await create_invoice( - wallet_id=producer.wallet, - amount=int(amount / 1000), - internal=True, - memo=f"Revenue from '{track.name}'.", - ) - logger.debug( - f"livestream: producer invoice created: {payment_hash}, {amount} msats" - ) - - checking_id = await pay_invoice( - payment_request=payment_request, - wallet_id=payment.wallet_id, - extra={ - **payment.extra, - "shared_with": f"Producer ID: {producer.id}", - "received": payment.amount, - }, - ) - logger.debug(f"livestream: producer invoice paid: {checking_id}") - - # so the flow is the following: - # - we receive, say, 1000 satoshis - # - if the fee_pct is, say, 30%, the amount we will send is 700 - # - we change the amount of receiving payment on the database from 1000 to 300 - # - we create a new payment on the producer's wallet with amount 700 diff --git a/lnbits/extensions/livestream/templates/livestream/_api_docs.html b/lnbits/extensions/livestream/templates/livestream/_api_docs.html deleted file mode 100644 index 34b6518d..00000000 --- a/lnbits/extensions/livestream/templates/livestream/_api_docs.html +++ /dev/null @@ -1,150 +0,0 @@ - - - -

Add tracks, profit.

-
-
-
- - - - - - - - GET - /livestream/api/v1/livestream -
Headers
- {"X-Api-Key": <invoice_key>}
-
Body (application/json)
-
- Returns 200 OK (application/json) -
- [<livestream_object>, ...] -
Curl example
- curl -X GET {{ request.base_url }}livestream/api/v1/livestream -H - "X-Api-Key: {{ user.wallets[0].inkey }}" - -
-
-
- - - - PUT - /livestream/api/v1/livestream/track/<track_id> -
Headers
- {"X-Api-Key": <invoice_key>}
-
Body (application/json)
-
- Returns 201 CREATED (application/json) -
-
Curl example
- curl -X PUT {{ request.base_url }} - livestream/api/v1/livestream/track/<track_id> -H "X-Api-Key: {{ - user.wallets[0].inkey }}" - -
-
-
- - - - PUT - /livestream/api/v1/livestream/fee/<fee_pct> -
Headers
- {"X-Api-Key": <invoice_key>}
-
Body (application/json)
-
- Returns 201 CREATED (application/json) -
-
Curl example
- curl -X PUT {{ request.base_url }} - livestream/api/v1/livestream/fee/<fee_pct> -H "X-Api-Key: {{ - user.wallets[0].inkey }}" - -
-
-
- - - - - POST - /livestream/api/v1/livestream/tracks -
Headers
- {"X-Api-Key": <invoice_key>}
-
Body (application/json)
- {"name": <string>, "download_url": <string>, - "price_msat": <integer>, "producer_id": <integer>, - "producer_name": <string>} -
- Returns 201 CREATED (application/json) -
-
Curl example
- curl -X POST {{ request.base_url }} - livestream/api/v1/livestream/tracks -d '{"name": <string>, - "download_url": <string>, "price_msat": <integer>, - "producer_id": <integer>, "producer_name": <string>}' -H - "Content-type: application/json" -H "X-Api-Key: {{ - user.wallets[0].adminkey }}" - -
-
-
- - - - DELETE - /livestream/api/v1/livestream/tracks/<track_id> -
Headers
- {"X-Api-Key": <invoice_key>}
-
Returns 204 NO CONTENT
- -
Curl example
- curl -X DELETE {{ request.base_url }} - livestream/api/v1/livestream/tracks/<track_id> -H "X-Api-Key: {{ - user.wallets[0].inkey }}" - -
-
-
-
diff --git a/lnbits/extensions/livestream/templates/livestream/index.html b/lnbits/extensions/livestream/templates/livestream/index.html deleted file mode 100644 index 8749a9cf..00000000 --- a/lnbits/extensions/livestream/templates/livestream/index.html +++ /dev/null @@ -1,323 +0,0 @@ -{% extends "base.html" %} {% from "macros.jinja" import window_vars with context -%} {% block page %} -
-{% endblock %} {% block scripts %} {{ window_vars(user) }} - -{% endblock %} diff --git a/lnbits/extensions/livestream/views.py b/lnbits/extensions/livestream/views.py deleted file mode 100644 index ca12f16b..00000000 --- a/lnbits/extensions/livestream/views.py +++ /dev/null @@ -1,48 +0,0 @@ -from http import HTTPStatus - -from fastapi import Depends, HTTPException, Query, Request -from starlette.datastructures import URL -from starlette.responses import HTMLResponse, RedirectResponse - -from lnbits.core.crud import get_wallet_payment -from lnbits.core.models import User -from lnbits.decorators import check_user_exists - -from . import livestream_ext, livestream_renderer -from .crud import get_livestream_by_track, get_track - - -@livestream_ext.get("/", response_class=HTMLResponse) -async def index(request: Request, user: User = Depends(check_user_exists)): - return livestream_renderer().TemplateResponse( - "livestream/index.html", {"request": request, "user": user.dict()} - ) - - -@livestream_ext.get("/track/{track_id}", name="livestream.track_redirect_download") -async def track_redirect_download(track_id, p: str = Query(...)): - payment_hash = p - track = await get_track(track_id) - ls = await get_livestream_by_track(track_id) - assert ls - payment = await get_wallet_payment(ls.wallet, payment_hash) - - if not payment: - raise HTTPException( - status_code=HTTPStatus.NOT_FOUND, - detail=f"Couldn't find the payment {payment_hash}.", - ) - if not track: - raise HTTPException( - status_code=HTTPStatus.NOT_FOUND, - detail=f"Couldn't find the track {track_id}.", - ) - - if payment.pending: - raise HTTPException( - status_code=HTTPStatus.PAYMENT_REQUIRED, - detail=f"Payment {payment_hash} wasn't received yet. Please try again in a minute.", - ) - - assert track.download_url - return RedirectResponse(url=URL(track.download_url)) diff --git a/lnbits/extensions/livestream/views_api.py b/lnbits/extensions/livestream/views_api.py deleted file mode 100644 index b346f353..00000000 --- a/lnbits/extensions/livestream/views_api.py +++ /dev/null @@ -1,101 +0,0 @@ -from http import HTTPStatus - -from fastapi import Depends, HTTPException, Request -from lnurl.exceptions import InvalidUrl as LnurlInvalidUrl - -from lnbits.decorators import WalletTypeInfo, get_key_type - -from . import livestream_ext -from .crud import ( - add_producer, - add_track, - delete_track_from_livestream, - get_or_create_livestream_by_wallet, - get_producers, - get_tracks, - update_current_track, - update_livestream_fee, - update_track, -) -from .models import CreateTrack - - -@livestream_ext.get("/api/v1/livestream") -async def api_livestream_from_wallet( - req: Request, g: WalletTypeInfo = Depends(get_key_type) -): - ls = await get_or_create_livestream_by_wallet(g.wallet.id) - assert ls - tracks = await get_tracks(ls.id) - producers = await get_producers(ls.id) - - try: - return { - **ls.dict(), - **{ - "lnurl": ls.lnurl(request=req), - "tracks": [ - dict(lnurl=track.lnurl(request=req), **track.dict()) - for track in tracks - ], - "producers": [producer.dict() for producer in producers], - }, - } - except LnurlInvalidUrl: - raise HTTPException( - status_code=HTTPStatus.UPGRADE_REQUIRED, - detail="LNURLs need to be delivered over a publically accessible `https` domain or Tor.", - ) - - -@livestream_ext.put("/api/v1/livestream/track/{track_id}") -async def api_update_track(track_id, g: WalletTypeInfo = Depends(get_key_type)): - try: - id = int(track_id) - except ValueError: - id = 0 - - ls = await get_or_create_livestream_by_wallet(g.wallet.id) - assert ls - await update_current_track(ls.id, None if id <= 0 else id) - return "", HTTPStatus.NO_CONTENT - - -@livestream_ext.put("/api/v1/livestream/fee/{fee_pct}") -async def api_update_fee(fee_pct, g: WalletTypeInfo = Depends(get_key_type)): - ls = await get_or_create_livestream_by_wallet(g.wallet.id) - assert ls - await update_livestream_fee(ls.id, int(fee_pct)) - return "", HTTPStatus.NO_CONTENT - - -@livestream_ext.post("/api/v1/livestream/tracks") -@livestream_ext.put("/api/v1/livestream/tracks/{id}") -async def api_add_track( - data: CreateTrack, id=None, g: WalletTypeInfo = Depends(get_key_type) -): - ls = await get_or_create_livestream_by_wallet(g.wallet.id) - assert ls - - if data.producer_id: - p_id = int(data.producer_id) - elif data.producer_name: - p_id = await add_producer(ls.id, data.producer_name) - else: - raise TypeError("need either producer_id or producer_name arguments") - - if id: - await update_track( - ls.id, id, data.name, data.download_url, data.price_msat or 0, p_id - ) - else: - await add_track(ls.id, data.name, data.download_url, data.price_msat or 0, p_id) - return - - -@livestream_ext.delete("/api/v1/livestream/tracks/{track_id}") -async def api_delete_track(track_id, g: WalletTypeInfo = Depends(get_key_type)): - ls = await get_or_create_livestream_by_wallet(g.wallet.id) - assert ls - await delete_track_from_livestream(ls.id, track_id) - return "", HTTPStatus.NO_CONTENT
- - -
- - -
- {{SITE_TITLE}} Livestream extension -
-
- - - {% include "livestream/_api_docs.html" %} - -
-
- - - - -

- Standalone QR Code for this track -

- - - - - - - Copy LNURL-pay code -
- - - - - - - -
-
- - Update track - Add track - -
-
- Cancel -
-
-
-
-
-
-