mirror of
https://codeberg.org/SimpleWeb/SimpleerTube.git
synced 2024-11-14 21:23:16 +00:00
325 lines
9.2 KiB
Python
325 lines
9.2 KiB
Python
from quart import Quart, request, render_template, redirect
|
|
from datetime import datetime
|
|
import peertube
|
|
|
|
commit = "not found"
|
|
with open(".git/refs/heads/main") as file:
|
|
for line in file:
|
|
commit = line
|
|
# we only expect one line
|
|
break
|
|
|
|
# Wrapper, only containing information that's important for us, and in some cases provides simplified ways to get information
|
|
class VideoWrapper:
|
|
def __init__(self, a, quality):
|
|
self.name = a["name"]
|
|
self.channel = a["channel"]
|
|
self.description = a["description"]
|
|
self.thumbnailPath = a["thumbnailPath"]
|
|
|
|
self.category = a["category"]
|
|
self.licence = a["licence"]
|
|
self.language = a["language"]
|
|
self.privacy = a["privacy"]
|
|
self.tags = a["tags"]
|
|
|
|
self.views = a["views"]
|
|
self.likes = a["likes"]
|
|
self.dislikes = a["dislikes"]
|
|
|
|
self.embedPath = a["embedPath"]
|
|
self.commentsEnabled = a["commentsEnabled"]
|
|
|
|
self.resolutions = []
|
|
self.video = None
|
|
|
|
self.files = a["files"]
|
|
if len(self.files) == 0:
|
|
self.files = ((a["streamingPlaylists"])[0])["files"]
|
|
|
|
for entry in self.files:
|
|
resolution = (entry["resolution"])["id"]
|
|
self.resolutions.append(entry["resolution"])
|
|
|
|
if str(resolution) == str(quality):
|
|
self.video = entry["fileUrl"]
|
|
|
|
self.no_quality_selected = not self.video
|
|
|
|
|
|
# Helper Class for using caches
|
|
class Cache:
|
|
def __init__(self):
|
|
self.dict = {}
|
|
|
|
def get(self, arg, func):
|
|
if arg in self.dict:
|
|
last_time_updated = (self.dict[arg])[1]
|
|
time_diff = datetime.now() - last_time_updated
|
|
|
|
if time_diff.days > 0:
|
|
self.dict[arg] = [
|
|
func(arg),
|
|
datetime.now()
|
|
]
|
|
else:
|
|
self.dict[arg] = [
|
|
func(arg),
|
|
datetime.now()
|
|
]
|
|
|
|
return (self.dict[arg])[0]
|
|
|
|
|
|
cached_instance_names = Cache()
|
|
cached_account_infos = Cache()
|
|
cached_video_channel_infos = Cache()
|
|
|
|
# cache the instance names so we don't have to send a request to the domain every time someone
|
|
# loads any site
|
|
def get_instance_name(domain):
|
|
return cached_instance_names.get(domain, peertube.get_instance_name)
|
|
|
|
|
|
# simple wrapper that is used inside the cached_account_infos
|
|
def get_account(info):
|
|
info = info.split("@")
|
|
return peertube.account(info[1], info[0])
|
|
|
|
def get_account_info(name):
|
|
return cached_account_infos.get(name, get_account)
|
|
|
|
|
|
# simple wrapper that is used inside the cached_video_channel_infos
|
|
def get_video_channel(info):
|
|
info = info.split("@")
|
|
return peertube.video_channel(info[1], info[0])
|
|
|
|
def get_video_channel_info(name):
|
|
return cached_video_channel_infos.get(name, get_video_channel)
|
|
|
|
|
|
|
|
|
|
app = Quart(__name__)
|
|
|
|
@app.route("/")
|
|
async def main():
|
|
return await render_template(
|
|
"index.html",
|
|
commit=commit,
|
|
)
|
|
|
|
@app.route("/search", methods = ["POST"])
|
|
async def simpleer_search_redirect():
|
|
query = (await request.form)["query"]
|
|
return redirect("/search/" + query)
|
|
|
|
@app.route("/search/<string:query>")
|
|
async def simpleer_search(query):
|
|
return await render_template(
|
|
"simpleer_search_results.html",
|
|
commit=commit,
|
|
|
|
results=peertube.sepia_search(query)
|
|
)
|
|
|
|
|
|
|
|
@app.route("/<string:domain>/")
|
|
async def instance(domain):
|
|
return redirect("/" + domain + "/videos/trending")
|
|
|
|
@app.route("/<string:domain>/videos/local")
|
|
async def instance_videos_local(domain):
|
|
return await render_template(
|
|
"instance/local.html",
|
|
domain=domain,
|
|
instance_name=get_instance_name(domain),
|
|
commit=commit,
|
|
|
|
videos = peertube.get_videos_local(domain),
|
|
)
|
|
|
|
@app.route("/<string:domain>/videos/trending")
|
|
async def instance_videos_trending(domain):
|
|
return await render_template(
|
|
"instance/trending.html",
|
|
domain=domain,
|
|
instance_name=get_instance_name(domain),
|
|
commit=commit,
|
|
|
|
videos = peertube.get_videos_trending(domain),
|
|
)
|
|
|
|
|
|
@app.route("/<string:domain>/videos/most-liked")
|
|
async def instance_videos_most_liked(domain):
|
|
return await render_template(
|
|
"instance/most-liked.html",
|
|
domain=domain,
|
|
instance_name=get_instance_name(domain),
|
|
commit=commit,
|
|
|
|
videos = peertube.get_videos_most_liked(domain),
|
|
)
|
|
|
|
|
|
@app.route("/<string:domain>/videos/recently-added")
|
|
async def instance_videos_recently_added(domain):
|
|
return await render_template(
|
|
"instance/recently-added.html",
|
|
domain=domain,
|
|
instance_name=get_instance_name(domain),
|
|
commit=commit,
|
|
|
|
videos = peertube.get_videos_recently_added(domain),
|
|
)
|
|
|
|
|
|
@app.route("/<string:domain>/search", methods=["POST"])
|
|
async def search_redirect(domain):
|
|
query = (await request.form)["query"]
|
|
return redirect("/" + domain + "/search/" + query)
|
|
|
|
|
|
@app.route("/<string:domain>/search/<string:term>")
|
|
async def search(domain, term):
|
|
amount, results = peertube.search(domain, term)
|
|
return await render_template(
|
|
"search_results.html",
|
|
domain=domain,
|
|
instance_name=get_instance_name(domain),
|
|
commit=commit,
|
|
|
|
amount=amount,
|
|
results=results,
|
|
search_term=term,
|
|
)
|
|
|
|
|
|
@app.route("/<string:domain>/videos/watch/<string:id>/")
|
|
async def video(domain, id):
|
|
data = peertube.video(domain, id)
|
|
quality = request.args.get("quality")
|
|
embed = request.args.get("embed")
|
|
if quality == None:
|
|
quality = "best"
|
|
vid = VideoWrapper(data, quality)
|
|
|
|
# only make a request for the comments if commentsEnabled
|
|
comments = ""
|
|
if data["commentsEnabled"]:
|
|
comments = peertube.get_comments(domain, id)
|
|
|
|
return await render_template(
|
|
"video.html",
|
|
domain=domain,
|
|
commit=commit,
|
|
instance_name=get_instance_name(domain),
|
|
|
|
video=vid,
|
|
comments=comments,
|
|
quality=quality,
|
|
embed=embed,
|
|
)
|
|
|
|
|
|
def build_channel_or_account_name(domain, name):
|
|
if '@' in name:
|
|
return name
|
|
return name + "@" + domain
|
|
|
|
# --- Accounts ---
|
|
|
|
@app.route("/<string:domain>/accounts/<string:name>")
|
|
async def accounts(domain, name):
|
|
return redirect("/" + domain + "/accounts/" + name + "/video-channels")
|
|
|
|
@app.route("/<string:domain>/accounts/<string:name>/video-channels")
|
|
async def account__video_channels(domain, name):
|
|
return await render_template(
|
|
"accounts/video_channels.html",
|
|
domain=domain,
|
|
commit=commit,
|
|
instance_name=get_instance_name(domain),
|
|
|
|
name = name,
|
|
account = get_account_info(build_channel_or_account_name(domain, name)),
|
|
video_channels = peertube.account_video_channels(domain, name)
|
|
)
|
|
|
|
@app.route("/<string:domain>/accounts/<string:name>/videos")
|
|
async def account__videos(domain, name):
|
|
return await render_template(
|
|
"accounts/videos.html",
|
|
domain=domain,
|
|
commit=commit,
|
|
instance_name=get_instance_name(domain),
|
|
|
|
name = name,
|
|
account = get_account_info(build_channel_or_account_name(domain, name)),
|
|
videos = peertube.account_videos(domain, name)
|
|
)
|
|
|
|
@app.route("/<string:domain>/accounts/<string:name>/about")
|
|
async def account__about(domain, name):
|
|
return await render_template(
|
|
"accounts/about.html",
|
|
domain=domain,
|
|
commit=commit,
|
|
instance_name=get_instance_name(domain),
|
|
|
|
name = name,
|
|
account = get_account_info(build_channel_or_account_name(domain, name)),
|
|
about = peertube.account(domain, name)
|
|
)
|
|
|
|
# --- Video-Channels ---
|
|
|
|
@app.route("/<string:domain>/video-channels/<string:name>")
|
|
async def video_channels(domain, name):
|
|
return redirect("/" + domain + "/video-channels/" + name + "/videos")
|
|
|
|
@app.route("/<string:domain>/video-channels/<string:name>/videos")
|
|
async def video_channels__videos(domain, name):
|
|
return await render_template(
|
|
"video_channels/videos.html",
|
|
domain=domain,
|
|
commit=commit,
|
|
instance_name=get_instance_name(domain),
|
|
|
|
name = name,
|
|
video_channel = get_video_channel_info(build_channel_or_account_name(domain, name)),
|
|
videos = peertube.video_channel_videos(domain, name)
|
|
)
|
|
|
|
@app.route("/<string:domain>/video-channels/<string:name>/video-playlists")
|
|
async def video_channels__video_playlists(domain, name):
|
|
return await render_template(
|
|
"video_channels/video_playlists.html",
|
|
domain=domain,
|
|
commit=commit,
|
|
instance_name=get_instance_name(domain),
|
|
|
|
name = name,
|
|
video_channel = get_video_channel_info(build_channel_or_account_name(domain, name)),
|
|
video_playlists = peertube.video_channel_video_playlists(domain, name)
|
|
)
|
|
|
|
@app.route("/<string:domain>/video-channels/<string:name>/about")
|
|
async def video_channels__about(domain, name):
|
|
return await render_template(
|
|
"video_channels/about.html",
|
|
domain=domain,
|
|
commit=commit,
|
|
instance_name=get_instance_name(domain),
|
|
|
|
name = name,
|
|
video_channel = get_video_channel_info(build_channel_or_account_name(domain, name)),
|
|
about = peertube.video_channel(domain, name)
|
|
)
|
|
|
|
if __name__ == "__main__":
|
|
app.run()
|