From 7c990e1627384a3af6c3bc3a1d4331ed0866bcdf Mon Sep 17 00:00:00 2001 From: Fijxu Date: Wed, 30 Oct 2024 01:59:08 -0300 Subject: [PATCH] External Proxies: Rotate between proxies with `balance` enabled Closes #17 --- src/invidious/helpers/utils.cr | 2 +- src/invidious/http_server/utils.cr | 44 +++++++++++++++++----- src/invidious/jobs/check_external_proxy.cr | 5 ++- 3 files changed, 39 insertions(+), 12 deletions(-) diff --git a/src/invidious/helpers/utils.cr b/src/invidious/helpers/utils.cr index c95532a1..a067b142 100644 --- a/src/invidious/helpers/utils.cr +++ b/src/invidious/helpers/utils.cr @@ -390,7 +390,7 @@ def gen_videoplayback_proxy_list if !CONFIG.external_videoplayback_proxy.empty? external_videoplayback_proxy = "" CONFIG.external_videoplayback_proxy.each do |proxy| - external_videoplayback_proxy += " #{proxy}" + external_videoplayback_proxy += " #{proxy[:url]}" end else external_videoplayback_proxy = "" diff --git a/src/invidious/http_server/utils.cr b/src/invidious/http_server/utils.cr index eecfb1cc..947d5303 100644 --- a/src/invidious/http_server/utils.cr +++ b/src/invidious/http_server/utils.cr @@ -4,25 +4,51 @@ module Invidious::HttpServer module Utils extend self - @@proxy_alive : String = "" + @@proxy_list : Array(String) = [] of String + @@current_proxy : String = "" + @@count : Int64 = Time.utc.to_unix def check_external_proxy CONFIG.external_videoplayback_proxy.each do |proxy| begin - response = HTTP::Client.get(proxy) + response = HTTP::Client.get("#{proxy[:url]}/health") if response.status_code == 200 - @@proxy_alive = proxy - LOGGER.debug("CheckExternalProxy: Proxy set to: '#{proxy}'") - break + if @@proxy_list.includes?(proxy[:url]) + next + end + if proxy[:balance] + @@proxy_list << proxy[:url] + LOGGER.debug("CheckExternalProxy: Adding proxy '#{proxy[:url]}' to the list of proxies") + end + break if proxy[:balance] == false && !@@proxy_list.empty? + @@proxy_list << proxy[:url] end rescue - LOGGER.debug("CheckExternalProxy: Proxy '#{proxy}' is not available") + if @@proxy_list.includes?(proxy[:url]) + LOGGER.debug("CheckExternalProxy: Proxy '#{proxy[:url]}' is not available, removing it from the list of proxies") + @@proxy_list.delete(proxy[:url]) + end + LOGGER.debug("CheckExternalProxy: Proxy '#{proxy[:url]}' is not available") end end + LOGGER.trace("CheckExternalProxy: List of proxies:") + LOGGER.trace("#{@@proxy_list.inspect}") + end + + # TODO: If the function is called many times, it will return a random + # proxy from the list. That is not how it should be. + # It should return the same proxy, in multiple function calls + def select_proxy + if (@@count - (Time.utc.to_unix - 30)) <= 0 + return if @@proxy_list.size <= 0 + @@current_proxy = @@proxy_list[Random.rand(@@proxy_list.size)] + LOGGER.debug("Current proxy is: '#{@@current_proxy}'") + @@count = Time.utc.to_unix + end end def get_external_proxy - return @@proxy_alive + return @@current_proxy end def proxy_video_url(raw_url : String, *, region : String? = nil, absolute : Bool = false) @@ -35,8 +61,8 @@ module Invidious::HttpServer url.query_params = params if absolute - if !@@proxy_alive.empty? - return "#{@@proxy_alive}#{url.request_target}" + if !(proxy = get_external_proxy()).empty? + return "#{proxy}#{url.request_target}" else return "#{HOST_URL}#{url.request_target}" end diff --git a/src/invidious/jobs/check_external_proxy.cr b/src/invidious/jobs/check_external_proxy.cr index 7a4a2080..28b57795 100644 --- a/src/invidious/jobs/check_external_proxy.cr +++ b/src/invidious/jobs/check_external_proxy.cr @@ -5,8 +5,9 @@ class Invidious::Jobs::CheckExternalProxy < Invidious::Jobs::BaseJob def begin loop do HttpServer::Utils.check_external_proxy - LOGGER.info("CheckExternalProxy: Done, sleeping for 1 minute") - sleep 1.minutes + HttpServer::Utils.select_proxy + LOGGER.info("CheckExternalProxy: Done, sleeping for 15 seconds") + sleep 15.seconds Fiber.yield end end