Videos: Create some base classes for streams
This commit is contained in:
parent
cbbec00e1c
commit
365330f1d9
1 changed files with 267 additions and 0 deletions
267
src/invidious/videos/streams.cr
Normal file
267
src/invidious/videos/streams.cr
Normal file
|
@ -0,0 +1,267 @@
|
||||||
|
module Invidious::Videos
|
||||||
|
# ------------------
|
||||||
|
# Structs & Enums
|
||||||
|
# ------------------
|
||||||
|
|
||||||
|
# "AUDIO_QUALITY_"
|
||||||
|
enum AudioQuality
|
||||||
|
Low
|
||||||
|
Medium
|
||||||
|
end
|
||||||
|
|
||||||
|
enum ProjType
|
||||||
|
Rectangular
|
||||||
|
Equirectangular
|
||||||
|
Mesh
|
||||||
|
end
|
||||||
|
|
||||||
|
struct ByteRange
|
||||||
|
getter start : UInt32
|
||||||
|
getter end : UInt32
|
||||||
|
|
||||||
|
def initialize(@start, @end)
|
||||||
|
end
|
||||||
|
|
||||||
|
def to_s
|
||||||
|
return "#{@start}-#{@end}"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# ------------------
|
||||||
|
# Traits
|
||||||
|
# ------------------
|
||||||
|
|
||||||
|
# Properties common to all streams containing audio
|
||||||
|
module AudioProperties
|
||||||
|
macro included
|
||||||
|
property audio_quality : AudioQuality
|
||||||
|
property audio_sample_rate : UInt32
|
||||||
|
property audio_channels : UInt8
|
||||||
|
property audio_loudness_db : Float64 = 0.0
|
||||||
|
|
||||||
|
private macro init_audio_properties(format)
|
||||||
|
@audio_quality = AudioQuality.parse(format["audioQuality"].as_s.lstrip("AUDIO_QUALITY_"))
|
||||||
|
@audio_sample_rate = format["audioSampleRate"].as_s.to_u32
|
||||||
|
@audio_channels = format["audioChannels"].as_i.to_u8
|
||||||
|
@audio_loudness_db = format["loudnessDb"]?.try &.as_f || 0.0
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Properties common to all streams containing video
|
||||||
|
module VideoProperties
|
||||||
|
macro included
|
||||||
|
property video_width : UInt32
|
||||||
|
property video_height : UInt32
|
||||||
|
property video_fps : UInt16
|
||||||
|
|
||||||
|
private macro init_video_properties(format)
|
||||||
|
@video_width = format["width"].as_i.to_u32
|
||||||
|
@video_height = format["height"].as_i.to_u32
|
||||||
|
@video_fps = format["fps"].as_i.to_u16
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Properties common to all audio & video streams
|
||||||
|
module AVCommonProperties
|
||||||
|
macro included
|
||||||
|
property bitrate : UInt64
|
||||||
|
property bitrate_avg : UInt64?
|
||||||
|
property content_length : UInt64
|
||||||
|
|
||||||
|
private macro init_av_common_properties(format)
|
||||||
|
@bitrate = format["bitrate"].as_i.to_u64
|
||||||
|
@bitrate_avg = format["averageBitrate"]?.try &.as_i.to_u64
|
||||||
|
@content_length = format["contentLength"].as_s.to_u64
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Properties that only applies to mulit-lingual adaptative streams.
|
||||||
|
# They apply to audio and text streams (notably text/mp4).
|
||||||
|
#
|
||||||
|
# Sample JSON for an audio track:
|
||||||
|
# "audioTrack": {
|
||||||
|
# "displayName": "Arabic",
|
||||||
|
# "id": "ar.0",
|
||||||
|
# "audioIsDefault": false
|
||||||
|
# },
|
||||||
|
#
|
||||||
|
# Sample JSON for a caption track:
|
||||||
|
# "captionTrack": {
|
||||||
|
# "displayName": "English",
|
||||||
|
# "vssId": ".en.eEY6OEpapPo",
|
||||||
|
# "languageCode": "en"
|
||||||
|
# }
|
||||||
|
module TrackProperties
|
||||||
|
macro included
|
||||||
|
property track_id : String?
|
||||||
|
property track_name : String = "default"
|
||||||
|
property iso_code : String?
|
||||||
|
property default : Bool = false
|
||||||
|
|
||||||
|
private macro init_track_properties(format)
|
||||||
|
if audio_track = format["audioTrack"]?
|
||||||
|
id = audio_track["id"].as_s
|
||||||
|
|
||||||
|
@track_id = id
|
||||||
|
@track_name = audio_track["displayName"].as_s
|
||||||
|
|
||||||
|
@iso_code = id.gsub(".0", "")
|
||||||
|
@default = audio_track["audioIsDefault"].as_bool
|
||||||
|
#
|
||||||
|
elsif caption_track = format["captionTrack"]?
|
||||||
|
@track_name = caption_track["displayName"].as_s
|
||||||
|
@track_id = caption_track["vssId"].as_s
|
||||||
|
@iso_code = caption_track["languageCode"].as_s
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Properties that only apply to adaptative streams of regular videos
|
||||||
|
module AdaptativeProperties
|
||||||
|
macro included
|
||||||
|
property init_range : ByteRange?
|
||||||
|
property index_range : ByteRange?
|
||||||
|
|
||||||
|
private macro init_adaptative_properties(format)
|
||||||
|
if init_range = format["initRange"]?
|
||||||
|
@init_range = ByteRange.new(
|
||||||
|
init_range["start"].as_i.to_u32,
|
||||||
|
init_range["end"].as_i.to_u32
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
if index_range = format["indexRange"]?
|
||||||
|
@index_range = ByteRange.new(
|
||||||
|
index_range["start"].as_i.to_u32,
|
||||||
|
index_range["end"].as_i.to_u32
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Properties that only apply to adaptative streams from livestrams
|
||||||
|
# (either in progress, or recenlty ended)
|
||||||
|
module LiveProperties
|
||||||
|
macro included
|
||||||
|
property target_duration : UInt32?
|
||||||
|
property max_dvr_duration : UInt32?
|
||||||
|
|
||||||
|
private macro init_live_properties(format)
|
||||||
|
@target_duration = format["targetDurationSec"]?.try(&.as_i.to_u32)
|
||||||
|
@max_dvr_duration = format["maxDvrDurationSec"]?.try(&.as_i.to_u32)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# ------------------
|
||||||
|
# Base class
|
||||||
|
# ------------------
|
||||||
|
|
||||||
|
# Base stream class defining all the common properties for all streams
|
||||||
|
abstract class Stream
|
||||||
|
getter itag : UInt16
|
||||||
|
getter label : String
|
||||||
|
property url : String
|
||||||
|
|
||||||
|
getter raw_mime_type : String
|
||||||
|
getter mime_type : String
|
||||||
|
getter codecs : String
|
||||||
|
|
||||||
|
getter last_modified : Time?
|
||||||
|
|
||||||
|
getter projection_type : ProjType
|
||||||
|
|
||||||
|
def initialize(format : JSON::Any, @label)
|
||||||
|
@itag = format["itag"].as_i.to_u16
|
||||||
|
@url = format["url"].as_s
|
||||||
|
|
||||||
|
@raw_mime_type = format["mimeType"].as_s
|
||||||
|
|
||||||
|
# Extract MIME type and codecs from the raw mimeType string
|
||||||
|
@mime_type, raw_codecs = @raw_mime_type.split(';')
|
||||||
|
@codecs = raw_codecs.lstrip(" codecs=\"").rstrip('"')
|
||||||
|
|
||||||
|
# Last modified is not present on livestreams
|
||||||
|
if last_modified = format["lastModified"].as_s
|
||||||
|
# E.g "1670664306(.)849305"
|
||||||
|
# Note: (.) is not present in the input data, it's used here to show
|
||||||
|
# the demarcation between seconds and microseconds.
|
||||||
|
timestamp = last_modified[0...10]
|
||||||
|
microseconds = last_modified[10..]
|
||||||
|
|
||||||
|
@last_modified = Time.utc(
|
||||||
|
seconds: timestamp.to_i64,
|
||||||
|
nanoseconds: microseconds.to_i * 1000
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
@projection_type = ProjType.parse(format["projectionType"].as_s)
|
||||||
|
|
||||||
|
# Initialize extra properties as required
|
||||||
|
{% begin %}
|
||||||
|
{%
|
||||||
|
properties_types = [
|
||||||
|
AudioProperties,
|
||||||
|
VideoProperties,
|
||||||
|
TrackProperties,
|
||||||
|
AVCommonProperties,
|
||||||
|
AdaptativeProperties,
|
||||||
|
LiveProperties,
|
||||||
|
]
|
||||||
|
%}
|
||||||
|
|
||||||
|
{% for type in properties_types %}
|
||||||
|
# Call the appropriate initialization macro if self
|
||||||
|
# inherits from the given type
|
||||||
|
{% if @type < type %}
|
||||||
|
init_{{type.id.split("::").last.id.underscore}}(format)
|
||||||
|
{% end %}
|
||||||
|
{% end %}
|
||||||
|
{% end %}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# ------------------
|
||||||
|
# Children classes
|
||||||
|
# ------------------
|
||||||
|
|
||||||
|
# An HTTP progressive stream (audio + video)
|
||||||
|
class ProgressiveHttpStream < Stream
|
||||||
|
include AudioProperties
|
||||||
|
include VideoProperties
|
||||||
|
include AVCommonProperties
|
||||||
|
end
|
||||||
|
|
||||||
|
# Base class for adaptative (DASH) streams
|
||||||
|
abstract class AdaptativeStream < Stream
|
||||||
|
include AdaptativeProperties
|
||||||
|
include LiveProperties
|
||||||
|
end
|
||||||
|
|
||||||
|
# An audio-only adaptative (DASH) stream
|
||||||
|
class AdaptativeAudioStream < AdaptativeStream
|
||||||
|
include AudioProperties
|
||||||
|
include AVCommonProperties
|
||||||
|
end
|
||||||
|
|
||||||
|
# An audio-only adaptative (DASH) stream with track informations
|
||||||
|
class AdaptativeAudioTrackStream < AdaptativeAudioStream
|
||||||
|
include TrackProperties
|
||||||
|
end
|
||||||
|
|
||||||
|
# A video-only adaptative (DASH) stream
|
||||||
|
class AdaptativeVideoStream < AdaptativeStream
|
||||||
|
include VideoProperties
|
||||||
|
include AVCommonProperties
|
||||||
|
end
|
||||||
|
|
||||||
|
# A text-only adaptative (DASH) stream
|
||||||
|
class AdaptativeTextStream < AdaptativeStream
|
||||||
|
include TrackProperties
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in a new issue