From 7ba95e32b5daa43c11e3a69cec05b8f795413899 Mon Sep 17 00:00:00 2001 From: Fijxu Date: Thu, 10 Oct 2024 18:35:09 -0300 Subject: [PATCH] Try to use iOS client. Signed-off-by: Fijxu --- src/invidious/videos.cr | 2 +- src/invidious/videos/parser.cr | 49 +++++++++++++---------- src/invidious/views/components/player.ecr | 9 +++-- 3 files changed, 33 insertions(+), 27 deletions(-) diff --git a/src/invidious/videos.cr b/src/invidious/videos.cr index c6e69ee5..60016d9c 100644 --- a/src/invidious/videos.cr +++ b/src/invidious/videos.cr @@ -324,7 +324,7 @@ rescue DB::Error end def fetch_video(id, region) - info = extract_video_info(video_id: id) + info = extract_video_info(video_id: id, level: 0) if reason = info["reason"]? if reason == "Video unavailable" diff --git a/src/invidious/videos/parser.cr b/src/invidious/videos/parser.cr index 4683058b..f8d4625f 100644 --- a/src/invidious/videos/parser.cr +++ b/src/invidious/videos/parser.cr @@ -50,13 +50,22 @@ def parse_related_video(related : JSON::Any) : Hash(String, JSON::Any)? } end -def extract_video_info(video_id : String) +def extract_video_info(video_id : String, *, level = 0, client_type = YoutubeAPI::ClientType::WebMobile) + # Infinite recursion prevention + level += 1 + if level >= 3 + return { + "version" => JSON::Any.new(Video::SCHEMA_VERSION.to_i64), + "reason" => JSON::Any.new("All counter-measures exhausted"), + } + end + # Init client config for the API client_config = YoutubeAPI::ClientConfig.new + client_config.client_type = client_type # Fetch data from the player endpoint player_response = YoutubeAPI.player(video_id: video_id, params: "2AMB", client_config: client_config) - playability_status = player_response.dig?("playabilityStatus", "status").try &.as_s if playability_status != "OK" @@ -65,10 +74,12 @@ def extract_video_info(video_id : String) reason ||= subreason.try &.[]("runs").as_a.map(&.[]("text")).join("") reason ||= player_response.dig("playabilityStatus", "reason").as_s - # Stop here if video is not a scheduled livestream or - # for LOGIN_REQUIRED when videoDetails element is not found because retrying won't help - if !{"LIVE_STREAM_OFFLINE", "LOGIN_REQUIRED"}.any?(playability_status) || + if playability_status == "UNPLAYABLE" && reason.includes?("Get the YouTube app") + return extract_video_info(video_id: video_id, level: level, client_type: YoutubeAPI::ClientType::IOS) + elsif !{"LIVE_STREAM_OFFLINE", "LOGIN_REQUIRED"}.any?(playability_status) || playability_status == "LOGIN_REQUIRED" && !player_response.dig?("videoDetails") + # Stop here if video is not a scheduled livestream or + # for LOGIN_REQUIRED when videoDetails element is not found because retrying won't help return { "version" => JSON::Any.new(Video::SCHEMA_VERSION.to_i64), "reason" => JSON::Any.new(reason), @@ -92,7 +103,7 @@ def extract_video_info(video_id : String) end # Don't fetch the next endpoint if the video is unavailable. - if {"OK", "LIVE_STREAM_OFFLINE", "LOGIN_REQUIRED"}.any?(playability_status) + if {"OK", "LIVE_STREAM_OFFLINE", "LOGIN_REQUIRED", "UNPLAYABLE"}.any?(playability_status) next_response = YoutubeAPI.next({"videoId": video_id, "params": ""}) player_response = player_response.merge(next_response) end @@ -102,22 +113,12 @@ def extract_video_info(video_id : String) new_player_response = nil - # Don't use Android client if po_token is passed because po_token doesn't - # work for Android client. - if reason.nil? && CONFIG.po_token.nil? - # Fetch the video streams using an Android client in order to get the - # decrypted URLs and maybe fix throttling issues (#2194). See the - # following issue for an explanation about decrypted URLs: - # https://github.com/TeamNewPipe/NewPipeExtractor/issues/562 - client_config.client_type = YoutubeAPI::ClientType::AndroidTestSuite + if reason || DECRYPT_FUNCTION.nil? || CONFIG.po_token.nil? + client_config.client_type = YoutubeAPI::ClientType::IOS new_player_response = try_fetch_streaming_data(video_id, client_config) end - # Last hope - # Only trigger if reason found and po_token or didn't work wth Android client. - # TvHtml5ScreenEmbed now requires sig helper for it to work but po_token is not required - # if the IP address is not blocked. - if CONFIG.po_token && reason || CONFIG.po_token.nil? && new_player_response.nil? + if reason && !DECRYPT_FUNCTION.nil? && CONFIG.po_token.nil? client_config.client_type = YoutubeAPI::ClientType::TvHtml5ScreenEmbed new_player_response = try_fetch_streaming_data(video_id, client_config) end @@ -470,11 +471,15 @@ private def convert_url(fmt) params = url.query_params end - n = DECRYPT_FUNCTION.try &.decrypt_nsig(params["n"]) - params["n"] = n if n + if old_n = params["n"]? + n = DECRYPT_FUNCTION.try &.decrypt_nsig(old_n) + params["n"] = n if n + end if token = CONFIG.po_token - params["pot"] = token + if {"WEB", "TVHTML5"}.any? { |x| params["c"].starts_with?(x) } + params["pot"] = token + end end url.query_params = params diff --git a/src/invidious/views/components/player.ecr b/src/invidious/views/components/player.ecr index 5c28358b..08e8ccfd 100644 --- a/src/invidious/views/components/player.ecr +++ b/src/invidious/views/components/player.ecr @@ -4,9 +4,6 @@ <% if params.autoplay %>autoplay<% end %> <% if params.video_loop %>loop<% end %> <% if params.controls %>controls<% end %>> - <% if (hlsvp = video.hls_manifest_url) && !CONFIG.disabled?("livestreams") %> - - <% else %> <% if params.listen %> <% # default to 128k m4a stream best_m4a_stream_index = 0 @@ -57,6 +54,11 @@ <% end %> <% end %> + <% if (hlsvp = video.hls_manifest_url) && !CONFIG.disabled?("livestreams") %> + + <% end %> + + <% preferred_captions.each do |caption| %> <% end %> @@ -64,7 +66,6 @@ <% captions.each do |caption| %> <% end %> - <% end %>