From 7cbdb5077a3cf26da6f23ce33e2bb55b4045212d Mon Sep 17 00:00:00 2001 From: Fijxu Date: Fri, 27 Sep 2024 19:09:50 -0300 Subject: [PATCH] Feat: Experimental support for external videoplayback proxies --- src/invidious.cr | 4 ++++ src/invidious/config.cr | 2 ++ src/invidious/jobs/check_external_proxy.cr | 13 +++++++++++ src/invidious/routes/api/manifest.cr | 26 ++++++++++++++++++---- src/invidious/routes/before_all.cr | 8 ++++++- src/invidious/routes/watch.cr | 4 ++-- 6 files changed, 50 insertions(+), 7 deletions(-) create mode 100644 src/invidious/jobs/check_external_proxy.cr diff --git a/src/invidious.cr b/src/invidious.cr index 8f79bcf3..97ae8254 100644 --- a/src/invidious.cr +++ b/src/invidious.cr @@ -188,6 +188,10 @@ Invidious::Jobs.register Invidious::Jobs::ClearExpiredItemsJob.new Invidious::Jobs.register Invidious::Jobs::InstanceListRefreshJob.new +if CONFIG.external_videoplayback_proxy + Invidious::Jobs.register Invidious::Jobs::CheckExternalProxy.new +end + Invidious::Jobs.start_all def popular_videos diff --git a/src/invidious/config.cr b/src/invidious/config.cr index d9c41604..4c0b2686 100644 --- a/src/invidious/config.cr +++ b/src/invidious/config.cr @@ -172,6 +172,8 @@ class Config # List of names of the backends property backends : Array(String) = [] of String + property external_videoplayback_proxy : String? + # Materialious redirects property materialious_domain : String? diff --git a/src/invidious/jobs/check_external_proxy.cr b/src/invidious/jobs/check_external_proxy.cr new file mode 100644 index 00000000..ce0b296e --- /dev/null +++ b/src/invidious/jobs/check_external_proxy.cr @@ -0,0 +1,13 @@ +class Invidious::Jobs::CheckExternalProxy < Invidious::Jobs::BaseJob + def initialize + end + + def begin + loop do + Invidious::Routes::API::Manifest.check_external_proxy + LOGGER.info("CheckExternalProxy: Done, sleeping for 1 minute") + sleep 1.minutes + Fiber.yield + end + end +end diff --git a/src/invidious/routes/api/manifest.cr b/src/invidious/routes/api/manifest.cr index 8ba735bb..dcfec909 100644 --- a/src/invidious/routes/api/manifest.cr +++ b/src/invidious/routes/api/manifest.cr @@ -1,4 +1,15 @@ module Invidious::Routes::API::Manifest + @@proxy_alive : Bool = false + + def self.check_external_proxy + begin + response = HTTP::Client.get("#{CONFIG.external_videoplayback_proxy}") + @@proxy_alive = response.status_code == 200 + rescue + @@proxy_alive = false + end + end + # /api/manifest/dash/id/:id def self.get_dash_video_id(env) env.response.headers.add("Access-Control-Allow-Origin", "*") @@ -35,7 +46,11 @@ module Invidious::Routes::API::Manifest if local uri = URI.parse(url) - url = "#{HOST_URL}#{uri.request_target}host/#{uri.host}/" + if @@proxy_alive + url = "#{CONFIG.external_videoplayback_proxy}#{uri.request_target}host/#{uri.host}/" + else + url = "#{HOST_URL}#{uri.request_target}host/#{uri.host}/" + end end "#{url}" @@ -48,21 +63,24 @@ module Invidious::Routes::API::Manifest if local adaptive_fmts.each do |fmt| - fmt["url"] = JSON::Any.new("#{HOST_URL}#{URI.parse(fmt["url"].as_s).request_target}") + if @@proxy_alive + fmt["url"] = JSON::Any.new("#{CONFIG.external_videoplayback_proxy}#{URI.parse(fmt["url"].as_s).request_target}") + else + fmt["url"] = JSON::Any.new("#{HOST_URL}#{URI.parse(fmt["url"].as_s).request_target}") + end end end audio_streams = video.audio_streams.sort_by { |stream| {stream["bitrate"].as_i} }.reverse! video_streams = video.video_streams.sort_by { |stream| {stream["width"].as_i, stream["fps"].as_i} }.reverse! - # Removes all the resolutions with a height higher than CONFIG.max_dash_resolution + # Removes all the resolutions with a height higher than CONFIG.max_dash_resolution if CONFIG.max_dash_resolution video_streams.reject! do |z| (z["height"].as_i > CONFIG.max_dash_resolution.not_nil!) if z["height"]? end end - manifest = XML.build(indent: " ", encoding: "UTF-8") do |xml| xml.element("MPD", "xmlns": "urn:mpeg:dash:schema:mpd:2011", "profiles": "urn:mpeg:dash:profile:full:2011", minBufferTime: "PT1.5S", type: "static", diff --git a/src/invidious/routes/before_all.cr b/src/invidious/routes/before_all.cr index 5695dee9..54c30fc5 100644 --- a/src/invidious/routes/before_all.cr +++ b/src/invidious/routes/before_all.cr @@ -28,6 +28,12 @@ module Invidious::Routes::BeforeAll extra_media_csp = "" end + if CONFIG.external_videoplayback_proxy + external_videoplayback_proxy = " #{CONFIG.external_videoplayback_proxy}" + else + external_videoplayback_proxy = "" + end + # Only allow the pages at /embed/* to be embedded if env.request.resource.starts_with?("/embed") frame_ancestors = "'self' file: http: https:" @@ -43,7 +49,7 @@ module Invidious::Routes::BeforeAll "style-src 'self' 'unsafe-inline'", "img-src 'self' data:", "font-src 'self' data:", - "connect-src 'self'", + "connect-src 'self'" + external_videoplayback_proxy, "manifest-src 'self'", "media-src 'self' blob:" + extra_media_csp, "child-src 'self' blob:", diff --git a/src/invidious/routes/watch.cr b/src/invidious/routes/watch.cr index a5ff1497..612a689f 100644 --- a/src/invidious/routes/watch.cr +++ b/src/invidious/routes/watch.cr @@ -120,7 +120,7 @@ module Invidious::Routes::Watch fmt_stream = video.fmt_stream adaptive_fmts = video.adaptive_fmts - # Removes all the resolutions with a height higher than CONFIG.max_dash_resolution + # Removes all the resolutions with a height higher than CONFIG.max_dash_resolution if CONFIG.max_dash_resolution adaptive_fmts.reject! do |z| (z["height"].as_i > CONFIG.max_dash_resolution.not_nil!) if z["height"]? @@ -135,7 +135,7 @@ module Invidious::Routes::Watch video_streams = video.video_streams audio_streams = video.audio_streams - # Removes all the resolutions with a height higher than CONFIG.max_dash_resolution + # Removes all the resolutions with a height higher than CONFIG.max_dash_resolution if CONFIG.max_dash_resolution video_streams.reject! do |z| (z["height"].as_i > CONFIG.max_dash_resolution.not_nil!) if z["height"]?