Refactor: Add object to represent chapters

Prior to this commit we used an Array of Chapter structs to represent
a video's chapters. However, as we often needed to apply operations on
the entire sequence of chapters, multiple isolated functions had to be
created and in turn clogged up the code.

By grouping everything together under a chapters struct that stores a
sequence of chapters, these functions can be grouped together, and can
be simplifed due to instance variables containing the data that they need.

Co-authored-by: Samantaz Fox <coding@samantaz.fr>
This commit is contained in:
syeopite 2024-01-13 23:46:20 -08:00
parent 2744ea2244
commit 503ace90f5
No known key found for this signature in database
GPG key ID: A73C186DA3955A1A
6 changed files with 111 additions and 92 deletions

View file

@ -201,10 +201,10 @@ module Invidious::JSONify::APIv1
end end
end end
if !video.chapters.empty? if !video.chapters.nil?
json.field "chapters" do json.field "chapters" do
json.object do json.object do
Invidious::Videos::Chapters.to_json(json, video.chapters, video.automatically_generated_chapters?.as(Bool)) video.chapters.to_json(json)
end end
end end
end end

View file

@ -431,16 +431,16 @@ module Invidious::Routes::API::V1::Videos
haltf env, 500 haltf env, 500
end end
if chapters.nil?
return error_json(404, "No chapters are defined in video \"#{id}\"")
end
if format == "json" if format == "json"
env.response.content_type = "application/json" env.response.content_type = "application/json"
return chapters.to_json
response = Invidious::Videos::Chapters.to_json(chapters, video.automatically_generated_chapters?.as(Bool))
return response
else else
env.response.content_type = "text/vtt; charset=UTF-8" env.response.content_type = "text/vtt; charset=UTF-8"
return chapters.to_vtt
return Invidious::Videos::Chapters.chapters_to_vtt(chapters)
end end
end end
end end

View file

@ -27,7 +27,7 @@ struct Video
@captions = [] of Invidious::Videos::Captions::Metadata @captions = [] of Invidious::Videos::Captions::Metadata
@[DB::Field(ignore: true)] @[DB::Field(ignore: true)]
@chapters = [] of Invidious::Videos::Chapters::Chapter @chapters : Invidious::Videos::Chapters? = nil
@[DB::Field(ignore: true)] @[DB::Field(ignore: true)]
property adaptive_fmts : Array(Hash(String, JSON::Any))? property adaptive_fmts : Array(Hash(String, JSON::Any))?
@ -231,17 +231,23 @@ struct Video
end end
def chapters def chapters
if @chapters.empty? # As the chapters key is always present in @info we need to check that it is
@chapters = Invidious::Videos::Chapters.parse(@info["chapters"].as_a, self.length_seconds) # actually populated
if @chapters.nil?
chapters = @info["chapters"].as_a
return nil if chapters.empty?
@chapters = Invidious::Videos::Chapters.from_raw_chapters(
chapters,
self.length_seconds,
# Should never be nil but just in case
is_auto_generated: @info["autoGeneratedChapters"].as_bool? || false
)
end end
return @chapters return @chapters
end end
def automatically_generated_chapters? : Bool?
return @info["autoGeneratedChapters"]?.try &.as_bool
end
def hls_manifest_url : String? def hls_manifest_url : String?
info.dig?("streamingData", "hlsManifestUrl").try &.as_s info.dig?("streamingData", "hlsManifestUrl").try &.as_s
end end

View file

@ -1,16 +1,21 @@
# Namespace for methods and objects relating to chapters module Invidious::Videos
module Invidious::Videos::Chapters # A `Chapters` struct represents an sequence of chapters for a given video
struct Chapters
record Chapter, start_ms : Time::Span, end_ms : Time::Span, title : String, thumbnails : Array(Hash(String, Int32 | String)) record Chapter, start_ms : Time::Span, end_ms : Time::Span, title : String, thumbnails : Array(Hash(String, Int32 | String))
property? auto_generated : Bool
# Parse raw chapters data into an array of Chapter structs def initialize(@chapters : Array(Chapter), @auto_generated : Bool)
end
# Constructs a chapters object from InnerTube's JSON object for chapters
# #
# Requires the length of the video the chapters are associated to in order to construct correct ending time # Requires the length of the video the chapters are associated to in order to construct correct ending time
def self.parse(chapters : Array(JSON::Any), video_length_seconds : Int32) def Chapters.from_raw_chapters(raw_chapters : Array(JSON::Any), video_length_seconds : Int32, is_auto_generated : Bool = false)
video_length_milliseconds = video_length_seconds.seconds.total_milliseconds video_length_milliseconds = video_length_seconds.seconds.total_milliseconds
segments = [] of Chapter parsed_chapters = [] of Chapter
chapters.each_with_index do |chapter, index| raw_chapters.each_with_index do |chapter, index|
chapter = chapter["chapterRenderer"] chapter = chapter["chapterRenderer"]
title = chapter["title"]["simpleText"].as_s title = chapter["title"]["simpleText"].as_s
@ -30,13 +35,13 @@ module Invidious::Videos::Chapters
# To get the ending range we have to peek at the next chapter. # To get the ending range we have to peek at the next chapter.
# If we're the last chapter then we need to calculate the end time through the video length. # If we're the last chapter then we need to calculate the end time through the video length.
if next_chapter = chapters[index + 1]? if next_chapter = raw_chapters[index + 1]?
end_ms = next_chapter["chapterRenderer"]["timeRangeStartMillis"].as_i end_ms = next_chapter["chapterRenderer"]["timeRangeStartMillis"].as_i
else else
end_ms = video_length_milliseconds.to_i end_ms = video_length_milliseconds.to_i
end end
segments << Chapter.new( parsed_chapters << Chapter.new(
start_ms: start_ms.milliseconds, start_ms: start_ms.milliseconds,
end_ms: end_ms.milliseconds, end_ms: end_ms.milliseconds,
title: title, title: title,
@ -44,23 +49,29 @@ module Invidious::Videos::Chapters
) )
end end
return segments return Chapters.new(parsed_chapters, is_auto_generated)
end end
# Converts an array of Chapter objects to a webvtt file # Calls the given block for each chapter and passes it as a parameter
def self.chapters_to_vtt(chapters : Array(Chapter)) def each(&)
@chapters.each { |c| yield c }
end
# Converts the sequence of chapters to a WebVTT representation
def to_vtt
vtt = WebVTT.build do |build| vtt = WebVTT.build do |build|
chapters.each do |chapter| self.each do |chapter|
build.cue(chapter.start_ms, chapter.end_ms, chapter.title) build.cue(chapter.start_ms, chapter.end_ms, chapter.title)
end end
end end
end end
def self.to_json(json : JSON::Builder, chapters : Array(Chapter), auto_generated? : Bool) # Dumps a JSON representation of the sequence of chapters to the given JSON::Builder
json.field "autoGenerated", auto_generated?.to_s def to_json(json : JSON::Builder)
json.field "autoGenerated", @auto_generated.to_s
json.field "chapters" do json.field "chapters" do
json.array do json.array do
chapters.each do |chapter| @chapters.each do |chapter|
json.object do json.object do
json.field "title", chapter.title json.field "title", chapter.title
json.field "startMs", chapter.start_ms.total_milliseconds json.field "startMs", chapter.start_ms.total_milliseconds
@ -83,12 +94,14 @@ module Invidious::Videos::Chapters
end end
end end
def self.to_json(chapters : Array(Chapter), auto_generated? : Bool) # Create a JSON representation of the sequence of chapters
def to_json
JSON.build do |json| JSON.build do |json|
json.object do json.object do
json.field "chapters" do json.field "chapters" do
json.object do json.object do
to_json(json, chapters, auto_generated?) to_json(json)
end
end end
end end
end end

View file

@ -1,9 +1,9 @@
<% if !chapters.empty? %> <% if !chapters.nil? %>
<div class="description-chapters-section"> <div class="description-chapters-section">
<hr class="description-content-separator"/> <hr class="description-content-separator"/>
<h4><%=HTML.escape(translate(locale, "video_chapters_label"))%></h4> <h4><%=HTML.escape(translate(locale, "video_chapters_label"))%></h4>
<% if video.automatically_generated_chapters? %> <% if chapters.auto_generated? %>
<h5><%=HTML.escape(translate(locale, "video_chapters_auto_generated_label"))%> </h5> <h5><%=HTML.escape(translate(locale, "video_chapters_auto_generated_label"))%> </h5>
<% end %> <% end %>

View file

@ -64,7 +64,7 @@
<track kind="captions" src="/api/v1/captions/<%= video.id %>?label=<%= caption.name %>" label="<%= caption.name %>"> <track kind="captions" src="/api/v1/captions/<%= video.id %>?label=<%= caption.name %>" label="<%= caption.name %>">
<% end %> <% end %>
<% if !chapters.empty? %> <% if !chapters.nil? %>
<track kind="chapters" src="/api/v1/chapters/<%= video.id %>"> <track kind="chapters" src="/api/v1/chapters/<%= video.id %>">
<% end %> <% end %>
<% end %> <% end %>