mirror of
https://github.com/iv-org/invidious.git
synced 2024-12-11 16:59:16 +05:30
Prevent autoplay breakage by vetting each change from top to compilation_video.ecr
This commit is contained in:
parent
0211d69f11
commit
f05d38aa8e
@ -22,7 +22,8 @@ CREATE TABLE IF NOT EXISTS public.compilations
|
|||||||
created timestamptz,
|
created timestamptz,
|
||||||
updated timestamptz,
|
updated timestamptz,
|
||||||
privacy compilation_privacy,
|
privacy compilation_privacy,
|
||||||
index int8[]
|
index int8[],
|
||||||
|
first_video_id text
|
||||||
);
|
);
|
||||||
|
|
||||||
GRANT ALL ON public.compilations TO current_user;
|
GRANT ALL ON public.compilations TO current_user;
|
||||||
|
@ -244,6 +244,7 @@
|
|||||||
"Not a playlist.": "Not a playlist.",
|
"Not a playlist.": "Not a playlist.",
|
||||||
"Playlist does not exist.": "Playlist does not exist.",
|
"Playlist does not exist.": "Playlist does not exist.",
|
||||||
"Could not pull trending pages.": "Could not pull trending pages.",
|
"Could not pull trending pages.": "Could not pull trending pages.",
|
||||||
|
"Compilation does not exist.": "Compilation does not exist.",
|
||||||
"Hidden field \"challenge\" is a required field": "Hidden field \"challenge\" is a required field",
|
"Hidden field \"challenge\" is a required field": "Hidden field \"challenge\" is a required field",
|
||||||
"Hidden field \"token\" is a required field": "Hidden field \"token\" is a required field",
|
"Hidden field \"token\" is a required field": "Hidden field \"token\" is a required field",
|
||||||
"Erroneous challenge": "Erroneous challenge",
|
"Erroneous challenge": "Erroneous challenge",
|
||||||
|
@ -95,6 +95,7 @@ struct Compilation
|
|||||||
property views : Int64
|
property views : Int64
|
||||||
property updated : Time
|
property updated : Time
|
||||||
property thumbnail : String?
|
property thumbnail : String?
|
||||||
|
property first_video_id : String
|
||||||
|
|
||||||
def to_json(offset, json : JSON::Builder, video_id : String? = nil)
|
def to_json(offset, json : JSON::Builder, video_id : String? = nil)
|
||||||
json.object do
|
json.object do
|
||||||
@ -169,6 +170,7 @@ struct InvidiousCompilation
|
|||||||
@[DB::Field(converter: InvidiousCompilation::CompilationPrivacyConverter)]
|
@[DB::Field(converter: InvidiousCompilation::CompilationPrivacyConverter)]
|
||||||
property privacy : CompilationPrivacy = CompilationPrivacy::Private
|
property privacy : CompilationPrivacy = CompilationPrivacy::Private
|
||||||
property index : Array(Int64)
|
property index : Array(Int64)
|
||||||
|
property first_video_id : String
|
||||||
|
|
||||||
@[DB::Field(ignore: true)]
|
@[DB::Field(ignore: true)]
|
||||||
property thumbnail_id : String?
|
property thumbnail_id : String?
|
||||||
@ -248,15 +250,16 @@ def create_compilation(title, privacy, user)
|
|||||||
LOGGER.info("generated compilation id")
|
LOGGER.info("generated compilation id")
|
||||||
|
|
||||||
compilation = InvidiousCompilation.new({
|
compilation = InvidiousCompilation.new({
|
||||||
title: title.byte_slice(0, 150),
|
title: title.byte_slice(0, 150),
|
||||||
id: compid,
|
id: compid,
|
||||||
author: user.email,
|
author: user.email,
|
||||||
description: "", # Max 5000 characters
|
description: "", # Max 5000 characters
|
||||||
video_count: 0,
|
video_count: 0,
|
||||||
created: Time.utc,
|
created: Time.utc,
|
||||||
updated: Time.utc,
|
updated: Time.utc,
|
||||||
privacy: privacy,
|
privacy: privacy,
|
||||||
index: [] of Int64,
|
index: [] of Int64,
|
||||||
|
first_video_id: ""
|
||||||
})
|
})
|
||||||
LOGGER.info("Creating compilation db")
|
LOGGER.info("Creating compilation db")
|
||||||
|
|
||||||
@ -268,15 +271,16 @@ end
|
|||||||
|
|
||||||
def subscribe_compilation(user, compilation)
|
def subscribe_compilation(user, compilation)
|
||||||
compilation = InvidiousCompilation.new({
|
compilation = InvidiousCompilation.new({
|
||||||
title: compilation.title.byte_slice(0, 150),
|
title: compilation.title.byte_slice(0, 150),
|
||||||
id: compilation.id,
|
id: compilation.id,
|
||||||
author: user.email,
|
author: user.email,
|
||||||
description: "", # Max 5000 characters
|
description: "", # Max 5000 characters
|
||||||
video_count: compilation.video_count,
|
video_count: compilation.video_count,
|
||||||
created: Time.utc,
|
created: Time.utc,
|
||||||
updated: compilation.updated,
|
updated: compilation.updated,
|
||||||
privacy: CompilationPrivacy::Private,
|
privacy: CompilationPrivacy::Private,
|
||||||
index: [] of Int64,
|
index: [] of Int64,
|
||||||
|
first_video_id: ""
|
||||||
})
|
})
|
||||||
|
|
||||||
Invidious::Database::Compilations.insert(compilation)
|
Invidious::Database::Compilations.insert(compilation)
|
||||||
@ -329,6 +333,19 @@ def get_compilation(compid : String)
|
|||||||
#end
|
#end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def update_first_video_id(compid : String)
|
||||||
|
if compilation = Invidious::Database::Compilations.select(id: compid)
|
||||||
|
compilation_index_array = compilation.index
|
||||||
|
first_index = compilation_index_array[0]
|
||||||
|
first_id = Invidious::Database::CompilationVideos.select_id_from_index(first_index)
|
||||||
|
if !first_id.nil?
|
||||||
|
Invidious::Database::Compilations.update_first_video_id(compid, first_id)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
raise NotFoundException.new("Compilation does not exist.")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def get_compilation_videos(compilation : InvidiousCompilation | Compilation, offset : Int32, video_id = nil)
|
def get_compilation_videos(compilation : InvidiousCompilation | Compilation, offset : Int32, video_id = nil)
|
||||||
LOGGER.info("1. get_compilation")
|
LOGGER.info("1. get_compilation")
|
||||||
LOGGER.info("Getting compilation")
|
LOGGER.info("Getting compilation")
|
||||||
|
@ -90,6 +90,15 @@ module Invidious::Database::Compilations
|
|||||||
PG_DB.exec(request, id, index)
|
PG_DB.exec(request, id, index)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def update_first_video_id(id : String, first_video_id : String)
|
||||||
|
request = <<-SQL
|
||||||
|
UPDATE compilations
|
||||||
|
SET first_video_id = $2
|
||||||
|
WHERE id = $1
|
||||||
|
SQL
|
||||||
|
|
||||||
|
PG_DB.exec(request, id, first_video_id)
|
||||||
|
end
|
||||||
# -------------------
|
# -------------------
|
||||||
# Select
|
# Select
|
||||||
# -------------------
|
# -------------------
|
||||||
@ -112,6 +121,16 @@ module Invidious::Database::Compilations
|
|||||||
return PG_DB.query_all(request, author, as: InvidiousCompilation)
|
return PG_DB.query_all(request, author, as: InvidiousCompilation)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def select_index_array(id : String)
|
||||||
|
request = <<-SQL
|
||||||
|
SELECT index FROM compilations
|
||||||
|
WHERE id = $1
|
||||||
|
LIMIT 1
|
||||||
|
SQL
|
||||||
|
|
||||||
|
PG_DB.query_one?(request, id, as: String)
|
||||||
|
end
|
||||||
|
|
||||||
# -------------------
|
# -------------------
|
||||||
# Select (filtered)
|
# Select (filtered)
|
||||||
# -------------------
|
# -------------------
|
||||||
@ -249,6 +268,16 @@ module Invidious::Database::CompilationVideos
|
|||||||
return PG_DB.query_one?(request, order_index, as: String)
|
return PG_DB.query_one?(request, order_index, as: String)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def select_id_from_index(index : Int64)
|
||||||
|
request = <<-SQL
|
||||||
|
SELECT id FROM compilation_videos
|
||||||
|
WHERE index = $1
|
||||||
|
LIMIT 1
|
||||||
|
SQL
|
||||||
|
|
||||||
|
return PG_DB.query_one?(request, index, as: String)
|
||||||
|
end
|
||||||
|
|
||||||
def select_index_from_order_index(order_index : Int32)
|
def select_index_from_order_index(order_index : Int32)
|
||||||
request = <<-SQL
|
request = <<-SQL
|
||||||
SELECT index FROM compilation_videos
|
SELECT index FROM compilation_videos
|
||||||
|
@ -24,7 +24,8 @@ module Invidious::Database::Migrations
|
|||||||
created timestamptz,
|
created timestamptz,
|
||||||
updated timestamptz,
|
updated timestamptz,
|
||||||
privacy compilation_privacy,
|
privacy compilation_privacy,
|
||||||
index int8[]
|
index int8[],
|
||||||
|
first_video_id text
|
||||||
);
|
);
|
||||||
SQL
|
SQL
|
||||||
|
|
||||||
|
@ -129,7 +129,7 @@ module Invidious::Routes::Compilations
|
|||||||
sid = sid.as(String)
|
sid = sid.as(String)
|
||||||
|
|
||||||
compid = env.params.query["list"]?
|
compid = env.params.query["list"]?
|
||||||
if !compid || !compid.starts_with?("IV")
|
if !compid || !compid.starts_with?("IVCMP")
|
||||||
return env.redirect referer
|
return env.redirect referer
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -238,50 +238,30 @@ module Invidious::Routes::Compilations
|
|||||||
title = env.params.body["title"]?.try &.delete("<>") || ""
|
title = env.params.body["title"]?.try &.delete("<>") || ""
|
||||||
privacy = CompilationPrivacy.parse(env.params.body["privacy"]? || "Private")
|
privacy = CompilationPrivacy.parse(env.params.body["privacy"]? || "Private")
|
||||||
|
|
||||||
#title = env.params.json["title"].try &.as(String).delete("<>").byte_slice(0, 150) || compilation.title
|
|
||||||
#privacy = env.params.json["privacy"]?.try { |p| CompilationPrivacy.parse(p.as(String).downcase) } || compilation.privacy
|
|
||||||
|
|
||||||
#if title != compilation.title ||
|
|
||||||
# privacy != compilation.privacy
|
|
||||||
# updated = Time.utc
|
|
||||||
#else
|
|
||||||
# updated = compilation.updated
|
|
||||||
#end
|
|
||||||
|
|
||||||
Invidious::Database::Compilations.update(compid, title, privacy, "", compilation.updated)
|
Invidious::Database::Compilations.update(compid, title, privacy, "", compilation.updated)
|
||||||
|
|
||||||
#{1...Invidious::Database::Compilations.count_owned_by(user.email)}.each do |index|
|
(0..compilation.index.size - 1).each do |index|
|
||||||
# start_timestamp = env.params.json["_start_timestamp"]?.try &.as(String).byte_slice(0, 150) || compilation.title
|
compilation_video_index = compilation.index[index]
|
||||||
compilation_video_cardinality = Invidious::Database::CompilationVideos.select_ids(compid, compilation.index).size
|
compilation_video = Invidious::Database::CompilationVideos.select_video(compid, compilation.index, compilation_video_index, 0, 1)
|
||||||
|
json_timestamp_query_start = compilation_video_index.to_s + "_start_timestamp"
|
||||||
|
|
||||||
(0..compilation_video_cardinality-1).each do |index|
|
start_timestamp = env.params.body[json_timestamp_query_start]?.try &.as(String).byte_slice(0, 8)
|
||||||
LOGGER.info("for loop cycle #{index} of #{Invidious::Database::Compilations.count_owned_by(user.email)}")
|
if !start_timestamp.nil? && !compilation_video[0].id.nil?
|
||||||
compilation_video_id = Invidious::Database::CompilationVideos.select_id_from_order_index(order_index: index)
|
|
||||||
#compilation_video_index = Invidious::Database::CompilationVideos.select_index_from_order_index(order_index: index)
|
|
||||||
compilation_video = Invidious::Database::CompilationVideos.select(compid, compilation.index, 0, 1)
|
|
||||||
#numerical_string = index.to
|
|
||||||
json_timestamp_query = index.to_s + "_start_timestamp"
|
|
||||||
LOGGER.info("adjust #{json_timestamp_query} ")
|
|
||||||
start_timestamp = env.params.body[json_timestamp_query]?.try &.as(String).byte_slice(0, 8)
|
|
||||||
LOGGER.info("render #{env.params.body[json_timestamp_query]?} ")
|
|
||||||
if !start_timestamp.nil? && !compilation_video_id.nil?
|
|
||||||
LOGGER.info("adjust #{json_timestamp_query} which renders as #{start_timestamp}")
|
|
||||||
start_timestamp_seconds = decode_length_seconds(start_timestamp)
|
start_timestamp_seconds = decode_length_seconds(start_timestamp)
|
||||||
if !start_timestamp_seconds.nil?
|
if !start_timestamp_seconds.nil?
|
||||||
if start_timestamp_seconds >= 0 && start_timestamp_seconds <= compilation_video[0].length_seconds
|
if start_timestamp_seconds >= 0 && start_timestamp_seconds <= compilation_video[0].length_seconds
|
||||||
LOGGER.info("adjusting timestamps to #{start_timestamp_seconds} which is #{start_timestamp_seconds.to_i}")
|
Invidious::Database::CompilationVideos.update_start_timestamp(compilation_video[0].id, start_timestamp_seconds.to_i)
|
||||||
Invidious::Database::CompilationVideos.update_start_timestamp(compilation_video_id, start_timestamp_seconds.to_i)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
json_timestamp_query = index.to_s + "_end_timestamp"
|
json_timestamp_query_end = compilation_video_index.to_s + "_end_timestamp"
|
||||||
end_timestamp = env.params.json[json_timestamp_query]?.try &.as(String).byte_slice(0, 8)
|
end_timestamp = env.params.json[json_timestamp_query_end]?.try &.as(String).byte_slice(0, 8)
|
||||||
if !end_timestamp.nil? && !compilation_video_id.nil?
|
if !end_timestamp.nil? && !compilation_video[0].id.nil?
|
||||||
end_timestamp_seconds = decode_length_seconds(end_timestamp)
|
end_timestamp_seconds = decode_length_seconds(end_timestamp)
|
||||||
if !end_timestamp_seconds.nil?
|
if !end_timestamp_seconds.nil?
|
||||||
if end_timestamp_seconds >= 0 && end_timestamp_seconds <= compilation_video[0].ending_timestamp_seconds
|
if end_timestamp_seconds >= 0 && end_timestamp_seconds <= compilation_video[0].ending_timestamp_seconds && end_timestamp_seconds > compilation_video[0].starting_timestamp_seconds
|
||||||
Invidious::Database::CompilationVideos.update_end_timestamp(compilation_video_id, end_timestamp_seconds)
|
Invidious::Database::CompilationVideos.update_end_timestamp(compilation_video[0].id, end_timestamp_seconds)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -309,7 +289,7 @@ module Invidious::Routes::Compilations
|
|||||||
sid = sid.as(String)
|
sid = sid.as(String)
|
||||||
|
|
||||||
compid = env.params.query["list"]?
|
compid = env.params.query["list"]?
|
||||||
if !compid || !compid.starts_with?("IV")
|
if !compid || !compid.starts_with?("IVCMP")
|
||||||
return env.redirect referer
|
return env.redirect referer
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -445,10 +425,12 @@ module Invidious::Routes::Compilations
|
|||||||
|
|
||||||
Invidious::Database::CompilationVideos.insert(compilation_video)
|
Invidious::Database::CompilationVideos.insert(compilation_video)
|
||||||
Invidious::Database::Compilations.update_video_added(compilation_id, compilation_video.index)
|
Invidious::Database::Compilations.update_video_added(compilation_id, compilation_video.index)
|
||||||
|
update_first_video_id(compilation_id)
|
||||||
when "action_remove_video"
|
when "action_remove_video"
|
||||||
index = env.params.query["set_video_id"]
|
index = env.params.query["set_video_id"]
|
||||||
Invidious::Database::CompilationVideos.delete(index)
|
Invidious::Database::CompilationVideos.delete(index)
|
||||||
Invidious::Database::Compilations.update_video_removed(compilation_id, index)
|
Invidious::Database::Compilations.update_video_removed(compilation_id, index)
|
||||||
|
update_first_video_id(compilation_id)
|
||||||
when "action_move_video_before"
|
when "action_move_video_before"
|
||||||
# TODO: Compilation stub
|
# TODO: Compilation stub
|
||||||
#video_index = compilation.index
|
#video_index = compilation.index
|
||||||
@ -473,6 +455,7 @@ module Invidious::Routes::Compilations
|
|||||||
compilation_index_array.insert(compilation_index_array_position-1,compilation_video[0].index)
|
compilation_index_array.insert(compilation_index_array_position-1,compilation_video[0].index)
|
||||||
Invidious::Database::Compilations.move_video_before(compilation_id, compilation_index_array)
|
Invidious::Database::Compilations.move_video_before(compilation_id, compilation_index_array)
|
||||||
end
|
end
|
||||||
|
update_first_video_id(compilation_id)
|
||||||
else
|
else
|
||||||
return error_json(400, "Unsupported action #{action}")
|
return error_json(400, "Unsupported action #{action}")
|
||||||
end
|
end
|
||||||
|
@ -39,8 +39,13 @@ module Invidious::Routes::Watch
|
|||||||
embed_link += embed_params.to_s
|
embed_link += embed_params.to_s
|
||||||
end
|
end
|
||||||
|
|
||||||
plid = env.params.query["list"]?.try &.gsub(/[^a-zA-Z0-9_-]/, "")
|
if env.params.query["list"]?.try &.starts_with? "IVPL"
|
||||||
continuation = process_continuation(env.params.query, plid, id)
|
plid = env.params.query["list"]?.try &.gsub(/[^a-zA-Z0-9_-]/, "")
|
||||||
|
continuation = process_continuation(env.params.query, plid, id)
|
||||||
|
elsif env.params.query["list"]?.try &.starts_with? "IVCMP"
|
||||||
|
compid = env.params.query["list"]?.try &.gsub(/[^a-zA-Z0-9_-]/, "")
|
||||||
|
continuation = process_continuation(env.params.query, compid, id)
|
||||||
|
end
|
||||||
|
|
||||||
nojs = env.params.query["nojs"]?
|
nojs = env.params.query["nojs"]?
|
||||||
|
|
||||||
|
@ -410,9 +410,9 @@ def fetch_video(id, region)
|
|||||||
return video
|
return video
|
||||||
end
|
end
|
||||||
|
|
||||||
def process_continuation(query, plid, id)
|
def process_continuation(query, list_id, id)
|
||||||
continuation = nil
|
continuation = nil
|
||||||
if plid
|
if list_id
|
||||||
if index = query["index"]?.try &.to_i?
|
if index = query["index"]?.try &.to_i?
|
||||||
continuation = index
|
continuation = index
|
||||||
else
|
else
|
||||||
|
@ -11,11 +11,13 @@
|
|||||||
|
|
||||||
<div class="flex-right button-container">
|
<div class="flex-right button-container">
|
||||||
<%- if compilation.is_a?(InvidiousCompilation) && compilation.author == user.try &.email -%>
|
<%- if compilation.is_a?(InvidiousCompilation) && compilation.author == user.try &.email -%>
|
||||||
<div class="pure-u">
|
<%- if compilation.index.size > 0 -%>
|
||||||
<a class="pure-button pure-button-secondary low-profile" dir="auto" href="/play_compilation?list=<%= compid %>">
|
<div class="pure-u">
|
||||||
<i class="icon ion-md-play"></i> <%= translate(locale, "compilation_button_play") %>
|
<a class="pure-button pure-button-secondary low-profile" dir="auto" href="/watch?v=<%= compilation.first_video_id %>&list=<%= compid %>&index=<%= compilation.index[0] %>">
|
||||||
</a>
|
<i class="icon ion-md-play"></i> <%= translate(locale, "compilation_button_play") %>
|
||||||
</div>
|
</a>
|
||||||
|
</div>
|
||||||
|
<%- end -%>
|
||||||
<div class="pure-u">
|
<div class="pure-u">
|
||||||
<a class="pure-button pure-button-secondary low-profile" dir="auto" href="/add_compilation_items?list=<%= compid %>">
|
<a class="pure-button pure-button-secondary low-profile" dir="auto" href="/add_compilation_items?list=<%= compid %>">
|
||||||
<i class="icon ion-md-add"></i> <%= translate(locale, "compilation_button_add_items") %>
|
<i class="icon ion-md-add"></i> <%= translate(locale, "compilation_button_add_items") %>
|
||||||
|
@ -38,16 +38,16 @@
|
|||||||
<% if compid_form = env.get?("remove_compilation_items") %>
|
<% if compid_form = env.get?("remove_compilation_items") %>
|
||||||
<div class="compilation-video-timestamp-set">
|
<div class="compilation-video-timestamp-set">
|
||||||
<p style="margin-right:10px;">from</p>
|
<p style="margin-right:10px;">from</p>
|
||||||
<input class="compilation-video-timestamp" placeholder="0:00" disabled type="text">
|
<input class="compilation-video-timestamp" value="<%= recode_length_seconds(compilation_video.starting_timestamp_seconds) %>" disabled type="text">
|
||||||
<p style="margin-right:10px; margin-left:10px">to</p>
|
<p style="margin-right:10px; margin-left:10px">to</p>
|
||||||
<input class="compilation-video-timestamp" placeholder="<%= recode_length_seconds(compilation_video.length_seconds) %>" disabled type="text">
|
<input class="compilation-video-timestamp" value="<%= recode_length_seconds(compilation_video.ending_timestamp_seconds) %>" disabled type="text">
|
||||||
</div>
|
</div>
|
||||||
<% else %>
|
<% else %>
|
||||||
<div class="compilation-video-timestamp-set">
|
<div class="compilation-video-timestamp-set">
|
||||||
<p style="margin-right:10px;">from</p>
|
<p style="margin-right:10px;">from</p>
|
||||||
<input maxlength="8" name="<%= compilation_video.order_index %>_start_timestamp" class="compilation-video-timestamp" placeholder="0:00" type="text">
|
<input maxlength="8" name="<%= compilation_video.index %>_start_timestamp" class="compilation-video-timestamp" value="<%= recode_length_seconds(compilation_video.starting_timestamp_seconds) %>" type="text">
|
||||||
<p style="margin-right:10px; margin-left:10px">to</p>
|
<p style="margin-right:10px; margin-left:10px">to</p>
|
||||||
<input maxlength="8" name="<%= compilation_video.order_index %>_end_timestamp" class="compilation-video-timestamp" placeholder="<%= recode_length_seconds(compilation_video.length_seconds) %>" type="text">
|
<input maxlength="8" name="<%= compilation_video.index %>_end_timestamp" class="compilation-video-timestamp" value="<%= recode_length_seconds(compilation_video.ending_timestamp_seconds) %>" type="text">
|
||||||
</div>
|
</div>
|
||||||
<% end %>
|
<% end %>
|
||||||
</div>
|
</div>
|
||||||
|
Loading…
Reference in New Issue
Block a user