2021-07-14 15:46:12 +00:00
|
|
|
def produce_channel_videos_continuation(ucid, page = 1, auto_generated = nil, sort_by = "newest", v2 = false)
|
2022-11-01 23:58:33 +00:00
|
|
|
object_inner_2 = {
|
|
|
|
"2:0:embedded" => {
|
|
|
|
"1:0:varint" => 0_i64,
|
2021-07-14 15:46:12 +00:00
|
|
|
},
|
2022-11-01 23:58:33 +00:00
|
|
|
"5:varint" => 50_i64,
|
|
|
|
"6:varint" => 1_i64,
|
|
|
|
"7:varint" => (page * 30).to_i64,
|
|
|
|
"9:varint" => 1_i64,
|
|
|
|
"10:varint" => 0_i64,
|
2021-07-14 15:46:12 +00:00
|
|
|
}
|
|
|
|
|
2022-11-01 23:58:33 +00:00
|
|
|
object_inner_2_encoded = object_inner_2
|
|
|
|
.try { |i| Protodec::Any.cast_json(i) }
|
|
|
|
.try { |i| Protodec::Any.from_json(i) }
|
|
|
|
.try { |i| Base64.urlsafe_encode(i) }
|
|
|
|
.try { |i| URI.encode_www_form(i) }
|
2021-07-14 15:46:12 +00:00
|
|
|
|
2022-11-11 19:26:34 +00:00
|
|
|
sort_by_numerical =
|
|
|
|
case sort_by
|
|
|
|
when "newest" then 1_i64
|
|
|
|
when "popular" then 2_i64
|
|
|
|
when "oldest" then 3_i64 # Broken as of 10/2022 :c
|
|
|
|
else 1_i64 # Fallback to "newest"
|
|
|
|
end
|
|
|
|
|
2022-11-01 23:58:33 +00:00
|
|
|
object_inner_1 = {
|
|
|
|
"110:embedded" => {
|
|
|
|
"3:embedded" => {
|
|
|
|
"15:embedded" => {
|
|
|
|
"1:embedded" => {
|
|
|
|
"1:string" => object_inner_2_encoded,
|
|
|
|
"2:string" => "00000000-0000-0000-0000-000000000000",
|
|
|
|
},
|
2022-11-11 19:26:34 +00:00
|
|
|
"3:varint" => sort_by_numerical,
|
2022-11-01 23:58:33 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2021-07-14 15:46:12 +00:00
|
|
|
|
2022-11-01 23:58:33 +00:00
|
|
|
object_inner_1_encoded = object_inner_1
|
|
|
|
.try { |i| Protodec::Any.cast_json(i) }
|
|
|
|
.try { |i| Protodec::Any.from_json(i) }
|
|
|
|
.try { |i| Base64.urlsafe_encode(i) }
|
|
|
|
.try { |i| URI.encode_www_form(i) }
|
2021-07-14 15:46:12 +00:00
|
|
|
|
2022-11-01 23:58:33 +00:00
|
|
|
object = {
|
|
|
|
"80226972:embedded" => {
|
|
|
|
"2:string" => ucid,
|
|
|
|
"3:string" => object_inner_1_encoded,
|
|
|
|
"35:string" => "browse-feed#{ucid}videos102",
|
|
|
|
},
|
|
|
|
}
|
2021-07-14 15:46:12 +00:00
|
|
|
|
2021-09-25 02:05:25 +00:00
|
|
|
continuation = object.try { |i| Protodec::Any.cast_json(i) }
|
2021-07-14 15:46:12 +00:00
|
|
|
.try { |i| Protodec::Any.from_json(i) }
|
|
|
|
.try { |i| Base64.urlsafe_encode(i) }
|
|
|
|
.try { |i| URI.encode_www_form(i) }
|
|
|
|
|
|
|
|
return continuation
|
|
|
|
end
|
|
|
|
|
2022-11-11 19:26:34 +00:00
|
|
|
# Used in bypass_captcha_job.cr
|
|
|
|
def produce_channel_videos_url(ucid, page = 1, auto_generated = nil, sort_by = "newest", v2 = false)
|
|
|
|
continuation = produce_channel_videos_continuation(ucid, page, auto_generated, sort_by, v2)
|
|
|
|
return "/browse_ajax?continuation=#{continuation}&gl=US&hl=en"
|
2021-07-14 15:46:12 +00:00
|
|
|
end
|
|
|
|
|
2022-11-11 19:26:34 +00:00
|
|
|
module Invidious::Channel::Tabs
|
|
|
|
extend self
|
2021-07-14 15:46:12 +00:00
|
|
|
|
2022-11-11 19:26:34 +00:00
|
|
|
# -------------------
|
|
|
|
# Regular videos
|
|
|
|
# -------------------
|
2021-07-14 15:46:12 +00:00
|
|
|
|
2022-11-11 19:26:34 +00:00
|
|
|
def make_initial_video_ctoken(ucid, sort_by) : String
|
|
|
|
return produce_channel_videos_continuation(ucid, sort_by: sort_by)
|
|
|
|
end
|
2021-07-14 15:46:12 +00:00
|
|
|
|
2022-11-11 19:26:34 +00:00
|
|
|
# Wrapper for AboutChannel, as we still need to call get_videos with
|
|
|
|
# an author name and ucid directly (e.g in RSS feeds).
|
|
|
|
# TODO: figure out how to get rid of that
|
|
|
|
def get_videos(channel : AboutChannel, *, continuation : String? = nil, sort_by = "newest")
|
|
|
|
return get_videos(
|
|
|
|
channel.author, channel.ucid,
|
|
|
|
continuation: continuation, sort_by: sort_by
|
|
|
|
)
|
|
|
|
end
|
2021-07-14 15:46:12 +00:00
|
|
|
|
2022-11-11 19:26:34 +00:00
|
|
|
# Wrapper for InvidiousChannel, as we still need to call get_videos with
|
|
|
|
# an author name and ucid directly (e.g in RSS feeds).
|
|
|
|
# TODO: figure out how to get rid of that
|
|
|
|
def get_videos(channel : InvidiousChannel, *, continuation : String? = nil, sort_by = "newest")
|
|
|
|
return get_videos(
|
|
|
|
channel.author, channel.id,
|
|
|
|
continuation: continuation, sort_by: sort_by
|
|
|
|
)
|
|
|
|
end
|
2021-07-14 15:46:12 +00:00
|
|
|
|
2022-11-11 19:26:34 +00:00
|
|
|
def get_videos(author : String, ucid : String, *, continuation : String? = nil, sort_by = "newest")
|
|
|
|
continuation ||= make_initial_video_ctoken(ucid, sort_by)
|
|
|
|
initial_data = YoutubeAPI.browse(continuation: continuation)
|
|
|
|
|
|
|
|
return extract_items(initial_data, author, ucid)
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_60_videos(channel : AboutChannel, *, continuation : String? = nil, sort_by = "newest")
|
|
|
|
if continuation.nil?
|
|
|
|
# Fetch the first "page" of video
|
|
|
|
items, next_continuation = get_videos(channel, sort_by: sort_by)
|
|
|
|
else
|
|
|
|
# Fetch a "page" of videos using the given continuation token
|
|
|
|
items, next_continuation = get_videos(channel, continuation: continuation)
|
|
|
|
end
|
|
|
|
|
|
|
|
# If there is more to load, then load a second "page"
|
|
|
|
# and replace the previous continuation token
|
|
|
|
if !next_continuation.nil?
|
|
|
|
items_2, next_continuation = get_videos(channel, continuation: next_continuation)
|
|
|
|
items.concat items_2
|
|
|
|
end
|
|
|
|
|
|
|
|
return items, next_continuation
|
|
|
|
end
|
2022-11-11 23:09:03 +00:00
|
|
|
|
|
|
|
# -------------------
|
|
|
|
# Shorts
|
|
|
|
# -------------------
|
|
|
|
|
2023-01-08 12:50:52 +00:00
|
|
|
private def fetch_shorts_data(ucid : String, continuation : String? = nil)
|
2022-11-11 23:09:03 +00:00
|
|
|
if continuation.nil?
|
|
|
|
# EgZzaG9ydHPyBgUKA5oBAA%3D%3D is the protobuf object to load "shorts"
|
|
|
|
# TODO: try to extract the continuation tokens that allows other sorting options
|
2023-01-08 12:50:52 +00:00
|
|
|
return YoutubeAPI.browse(ucid, params: "EgZzaG9ydHPyBgUKA5oBAA%3D%3D")
|
2022-11-11 23:09:03 +00:00
|
|
|
else
|
2023-01-08 12:50:52 +00:00
|
|
|
return YoutubeAPI.browse(continuation: continuation)
|
2022-11-11 23:09:03 +00:00
|
|
|
end
|
2023-01-08 12:50:52 +00:00
|
|
|
end
|
2022-11-11 23:09:03 +00:00
|
|
|
|
2023-01-08 12:50:52 +00:00
|
|
|
def get_shorts(channel : AboutChannel, continuation : String? = nil)
|
|
|
|
initial_data = self.fetch_shorts_data(channel.ucid, continuation)
|
|
|
|
|
|
|
|
begin
|
|
|
|
# Try to parse the initial data fetched above
|
|
|
|
return extract_items(initial_data, channel.author, channel.ucid)
|
|
|
|
rescue ex : RetryOnceException
|
|
|
|
# Sometimes, for a completely unknown reason, the "reelItemRenderer"
|
|
|
|
# object is missing some critical information (it happens once in about
|
|
|
|
# 20 subsequent requests). Refreshing the page is required to properly
|
|
|
|
# show the "shorts" tab.
|
|
|
|
#
|
|
|
|
# In order to make the experience smoother for the user, we simulate
|
|
|
|
# said page refresh by fetching again the JSON. If that still doesn't
|
|
|
|
# work, we raise a BrokenTubeException, as something is really broken.
|
|
|
|
begin
|
|
|
|
initial_data = self.fetch_shorts_data(channel.ucid, continuation)
|
|
|
|
return extract_items(initial_data, channel.author, channel.ucid)
|
|
|
|
rescue ex : RetryOnceException
|
|
|
|
raise BrokenTubeException.new "reelPlayerHeaderSupportedRenderers"
|
|
|
|
end
|
|
|
|
end
|
2022-11-11 23:09:03 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
# -------------------
|
|
|
|
# Livestreams
|
|
|
|
# -------------------
|
|
|
|
|
|
|
|
def get_livestreams(channel : AboutChannel, continuation : String? = nil)
|
|
|
|
if continuation.nil?
|
|
|
|
# EgdzdHJlYW1z8gYECgJ6AA%3D%3D is the protobuf object to load "streams"
|
|
|
|
initial_data = YoutubeAPI.browse(channel.ucid, params: "EgdzdHJlYW1z8gYECgJ6AA%3D%3D")
|
|
|
|
else
|
|
|
|
initial_data = YoutubeAPI.browse(continuation: continuation)
|
|
|
|
end
|
|
|
|
|
|
|
|
return extract_items(initial_data, channel.author, channel.ucid)
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_60_livestreams(channel : AboutChannel, continuation : String? = nil)
|
|
|
|
if continuation.nil?
|
|
|
|
# Fetch the first "page" of streams
|
|
|
|
items, next_continuation = get_livestreams(channel)
|
|
|
|
else
|
|
|
|
# Fetch a "page" of streams using the given continuation token
|
|
|
|
items, next_continuation = get_livestreams(channel, continuation: continuation)
|
|
|
|
end
|
|
|
|
|
|
|
|
# If there is more to load, then load a second "page"
|
|
|
|
# and replace the previous continuation token
|
|
|
|
if !next_continuation.nil?
|
|
|
|
items_2, next_continuation = get_livestreams(channel, continuation: next_continuation)
|
|
|
|
items.concat items_2
|
|
|
|
end
|
|
|
|
|
|
|
|
return items, next_continuation
|
|
|
|
end
|
2021-07-14 15:46:12 +00:00
|
|
|
end
|