1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
|
// yt - A fully featured command line YouTube client
//
// Copyright (C) 2024 Benedikt Peetz <benedikt.peetz@b-peetz.de>
// SPDX-License-Identifier: GPL-3.0-or-later
//
// This file is part of Yt.
//
// You should have received a copy of the License along with this program.
// If not, see <https://www.gnu.org/licenses/gpl-3.0.txt>.
use std::{collections::HashMap, path::PathBuf};
use pyo3::{types::PyDict, Bound, PyResult, Python};
use serde::{Deserialize, Deserializer, Serialize};
use serde_json::Value;
use url::Url;
use crate::json_loads_str;
type Todo = String;
// TODO: Change this to map `_type` to a structure of values, instead of the options <2024-05-27>
// And replace all the strings with better types (enums or urls)
#[derive(Debug, Deserialize, Serialize, PartialEq)]
#[serde(deny_unknown_fields)]
pub struct InfoJson {
pub __files_to_move: Option<FilesToMove>,
pub __last_playlist_index: Option<u32>,
pub __post_extractor: Option<String>,
pub __x_forwarded_for_ip: Option<String>,
pub _filename: Option<PathBuf>,
pub _format_sort_fields: Option<Vec<String>>,
pub _has_drm: Option<Todo>,
pub _type: Option<InfoType>,
pub _version: Option<Version>,
pub abr: Option<f64>,
pub acodec: Option<String>,
pub age_limit: Option<u32>,
pub aspect_ratio: Option<f64>,
pub asr: Option<u32>,
pub audio_channels: Option<u32>,
pub audio_ext: Option<String>,
pub automatic_captions: Option<HashMap<String, Vec<Caption>>>,
pub availability: Option<String>,
pub average_rating: Option<String>,
pub categories: Option<Vec<String>>,
pub channel: Option<String>,
pub channel_follower_count: Option<u32>,
pub channel_id: Option<String>,
pub channel_is_verified: Option<bool>,
pub channel_url: Option<String>,
pub chapters: Option<Vec<Chapter>>,
pub comment_count: Option<u32>,
pub comments: Option<Vec<Comment>>,
pub concurrent_view_count: Option<u32>,
pub description: Option<String>,
pub display_id: Option<String>,
pub downloader_options: Option<DownloaderOptions>,
pub duration: Option<f64>,
pub duration_string: Option<String>,
pub dynamic_range: Option<String>,
pub entries: Option<Vec<InfoJson>>,
pub episode: Option<String>,
pub episode_number: Option<u32>,
pub epoch: Option<u32>,
pub ext: Option<String>,
pub extractor: Option<Extractor>,
pub extractor_key: Option<ExtractorKey>,
pub filename: Option<PathBuf>,
pub filesize: Option<u64>,
pub filesize_approx: Option<u64>,
pub format: Option<String>,
pub format_id: Option<String>,
pub format_index: Option<u32>,
pub format_note: Option<String>,
pub formats: Option<Vec<Format>>,
pub fps: Option<f64>,
pub fulltitle: Option<String>,
pub has_drm: Option<bool>,
pub heatmap: Option<Vec<HeatMapEntry>>,
pub height: Option<u32>,
pub http_headers: Option<HttpHeader>,
pub id: Option<String>,
pub ie_key: Option<ExtractorKey>,
pub is_live: Option<bool>,
pub language: Option<String>,
pub language_preference: Option<i32>,
pub license: Option<Todo>,
pub like_count: Option<u32>,
pub live_status: Option<String>,
pub location: Option<Todo>,
pub manifest_url: Option<Url>,
pub modified_date: Option<String>,
pub n_entries: Option<u32>,
pub original_url: Option<String>,
pub playable_in_embed: Option<bool>,
pub playlist: Option<Todo>,
pub playlist_autonumber: Option<u32>,
pub playlist_channel: Option<Todo>,
pub playlist_channel_id: Option<Todo>,
pub playlist_count: Option<u32>,
pub playlist_id: Option<Todo>,
pub playlist_index: Option<u64>,
pub playlist_title: Option<Todo>,
pub playlist_uploader: Option<Todo>,
pub playlist_uploader_id: Option<Todo>,
pub preference: Option<Todo>,
pub protocol: Option<String>,
pub quality: Option<f64>,
pub release_date: Option<String>,
pub release_timestamp: Option<u64>,
pub release_year: Option<u32>,
pub requested_downloads: Option<Vec<RequestedDownloads>>,
pub requested_entries: Option<Vec<u32>>,
pub requested_formats: Option<Vec<Format>>,
pub requested_subtitles: Option<HashMap<String, Subtitle>>,
pub resolution: Option<String>,
pub season: Option<String>,
pub season_number: Option<u32>,
pub series: Option<String>,
pub source_preference: Option<i32>,
pub sponsorblock_chapters: Option<Vec<SponsorblockChapter>>,
pub stretched_ratio: Option<Todo>,
pub subtitles: Option<HashMap<String, Vec<Caption>>>,
pub tags: Option<Vec<String>>,
pub tbr: Option<f64>,
pub thumbnail: Option<Url>,
pub thumbnails: Option<Vec<ThumbNail>>,
pub timestamp: Option<u64>,
pub title: Option<String>,
pub upload_date: Option<String>,
pub uploader: Option<String>,
pub uploader_id: Option<String>,
pub uploader_url: Option<String>,
pub url: Option<Url>,
pub vbr: Option<f64>,
pub vcodec: Option<String>,
pub video_ext: Option<String>,
pub view_count: Option<u32>,
pub was_live: Option<bool>,
pub webpage_url: Option<Url>,
pub webpage_url_basename: Option<String>,
pub webpage_url_domain: Option<String>,
pub width: Option<u32>,
}
#[derive(Debug, Deserialize, Serialize, PartialEq)]
#[serde(deny_unknown_fields)]
pub struct FilesToMove {}
#[derive(Debug, Deserialize, Serialize, PartialEq)]
#[serde(deny_unknown_fields)]
pub struct RequestedDownloads {
pub __files_to_merge: Option<Vec<Todo>>,
pub __finaldir: PathBuf,
pub __infojson_filename: PathBuf,
pub __postprocessors: Vec<Todo>,
pub __real_download: bool,
pub __write_download_archive: bool,
pub _filename: PathBuf,
pub _type: InfoType,
pub _version: Version,
pub abr: f64,
pub acodec: String,
pub aspect_ratio: f64,
pub asr: u32,
pub audio_channels: Option<u32>,
pub chapters: Option<Vec<SponsorblockChapter>>,
pub duration: Option<f64>,
pub dynamic_range: String,
pub ext: String,
pub filename: PathBuf,
pub filepath: PathBuf,
pub filesize_approx: Option<u64>,
pub format: String,
pub format_id: String,
pub format_note: String,
pub fps: f64,
pub height: u32,
pub infojson_filename: PathBuf,
pub language: Option<String>,
pub protocol: String,
pub requested_formats: Vec<Format>,
pub resolution: String,
pub tbr: f64,
pub vbr: f64,
pub vcodec: String,
pub width: u32,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd, Ord, Eq)]
#[serde(deny_unknown_fields)]
pub struct Subtitle {
pub ext: SubtitleExt,
pub filepath: PathBuf,
pub name: Option<String>,
pub url: Url,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd, Ord, Eq)]
#[serde(deny_unknown_fields)]
pub enum SubtitleExt {
#[serde(alias = "vtt")]
Vtt,
#[serde(alias = "mp4")]
Mp4,
#[serde(alias = "json")]
Json,
#[serde(alias = "json3")]
Json3,
#[serde(alias = "ttml")]
Ttml,
#[serde(alias = "srv1")]
Srv1,
#[serde(alias = "srv2")]
Srv2,
#[serde(alias = "srv3")]
Srv3,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, Eq, PartialOrd, Ord)]
#[serde(deny_unknown_fields)]
pub struct Caption {
pub ext: SubtitleExt,
pub name: Option<String>,
pub protocol: Option<String>,
pub url: String,
pub filepath: Option<PathBuf>,
pub video_id: Option<String>,
pub manifest_url: Option<Url>,
pub filesize: Option<u64>,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd)]
#[serde(deny_unknown_fields)]
pub struct Chapter {
pub end_time: f64,
pub start_time: f64,
pub title: String,
}
#[derive(Debug, Deserialize, Serialize, PartialEq)]
#[serde(deny_unknown_fields)]
pub struct SponsorblockChapter {
/// This is an utterly useless field, and should thus be ignored
pub _categories: Option<Vec<Vec<Value>>>,
pub categories: Option<Vec<SponsorblockChapterCategory>>,
pub category: Option<SponsorblockChapterCategory>,
pub category_names: Option<Vec<String>>,
pub end_time: f64,
pub name: Option<String>,
pub r#type: Option<SponsorblockChapterType>,
pub start_time: f64,
pub title: String,
}
pub fn get_none<'de, D, T>(_: D) -> Result<Option<T>, D::Error>
where
D: Deserializer<'de>,
{
Ok(None)
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd, Ord, Eq)]
#[serde(deny_unknown_fields)]
pub enum SponsorblockChapterType {
#[serde(alias = "skip")]
Skip,
#[serde(alias = "chapter")]
Chapter,
#[serde(alias = "poi")]
Poi,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd, Ord, Eq)]
#[serde(deny_unknown_fields)]
pub enum SponsorblockChapterCategory {
#[serde(alias = "filler")]
Filler,
#[serde(alias = "interaction")]
Interaction,
#[serde(alias = "music_offtopic")]
MusicOfftopic,
#[serde(alias = "poi_highlight")]
PoiHighlight,
#[serde(alias = "preview")]
Preview,
#[serde(alias = "sponsor")]
Sponsor,
#[serde(alias = "selfpromo")]
SelfPromo,
#[serde(alias = "chapter")]
Chapter,
#[serde(alias = "intro")]
Intro,
#[serde(alias = "outro")]
Outro,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd)]
#[serde(deny_unknown_fields)]
pub struct HeatMapEntry {
pub start_time: f64,
pub end_time: f64,
pub value: f64,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd, Ord, Eq)]
#[serde(deny_unknown_fields)]
pub enum Extractor {
#[serde(alias = "generic")]
Generic,
#[serde(alias = "SVTSeries")]
SVTSeries,
#[serde(alias = "SVTPlay")]
SVTPlay,
#[serde(alias = "youtube")]
YouTube,
#[serde(alias = "youtube:tab")]
YouTubeTab,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd, Ord, Eq)]
#[serde(deny_unknown_fields)]
pub enum ExtractorKey {
#[serde(alias = "Generic")]
Generic,
#[serde(alias = "SVTSeries")]
SVTSeries,
#[serde(alias = "SVTPlay")]
SVTPlay,
#[serde(alias = "Youtube")]
YouTube,
#[serde(alias = "YoutubeTab")]
YouTubeTab,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd, Ord, Eq)]
#[serde(deny_unknown_fields)]
pub enum InfoType {
#[serde(alias = "playlist")]
Playlist,
#[serde(alias = "url")]
Url,
#[serde(alias = "video")]
Video,
}
#[derive(Debug, Deserialize, Serialize, Eq, PartialEq, PartialOrd, Ord)]
#[serde(deny_unknown_fields)]
pub struct Version {
pub current_git_head: Option<String>,
pub release_git_head: String,
pub repository: String,
pub version: String,
}
#[derive(Debug, Deserialize, Serialize, Clone, Eq, PartialEq, PartialOrd, Ord)]
#[serde(from = "String")]
#[serde(deny_unknown_fields)]
pub enum Parent {
Root,
Id(String),
}
impl Parent {
pub fn id(&self) -> Option<&str> {
if let Self::Id(id) = self {
Some(id)
} else {
None
}
}
}
impl From<String> for Parent {
fn from(value: String) -> Self {
if value == "root" {
Self::Root
} else {
Self::Id(value)
}
}
}
#[derive(Debug, Deserialize, Serialize, Clone, Eq, PartialEq, PartialOrd, Ord)]
#[serde(from = "String")]
#[serde(deny_unknown_fields)]
pub struct Id {
pub id: String,
}
impl From<String> for Id {
fn from(value: String) -> Self {
Self {
// Take the last element if the string is split with dots, otherwise take the full id
id: value.split('.').last().unwrap_or(&value).to_owned(),
}
}
}
#[derive(Debug, Deserialize, Serialize, Clone, Eq, PartialEq, PartialOrd, Ord)]
#[serde(deny_unknown_fields)]
pub struct Comment {
pub id: Id,
pub text: String,
#[serde(default = "zero")]
pub like_count: u32,
pub is_pinned: bool,
pub author_id: String,
#[serde(default = "unknown")]
pub author: String,
pub author_is_verified: bool,
pub author_thumbnail: Url,
pub parent: Parent,
#[serde(deserialize_with = "edited_from_time_text", alias = "_time_text")]
pub edited: bool,
// Can't also be deserialized, as it's already used in 'edited'
// _time_text: String,
pub timestamp: i64,
pub author_url: Url,
pub author_is_uploader: bool,
pub is_favorited: bool,
}
fn unknown() -> String {
"<Unknown>".to_string()
}
fn zero() -> u32 {
0
}
fn edited_from_time_text<'de, D>(d: D) -> Result<bool, D::Error>
where
D: Deserializer<'de>,
{
let s = String::deserialize(d)?;
if s.contains(" (edited)") {
Ok(true)
} else {
Ok(false)
}
}
#[derive(Debug, Deserialize, Serialize, PartialEq, Eq, PartialOrd, Ord)]
#[serde(deny_unknown_fields)]
pub struct ThumbNail {
pub id: Option<String>,
pub preference: Option<i32>,
/// in the form of "[`height`]x[`width`]"
pub resolution: Option<String>,
pub url: Url,
pub width: Option<u32>,
pub height: Option<u32>,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd)]
#[serde(deny_unknown_fields)]
pub struct Format {
pub __needs_testing: Option<bool>,
pub __working: Option<bool>,
pub abr: Option<f64>,
pub acodec: Option<String>,
pub aspect_ratio: Option<f64>,
pub asr: Option<f64>,
pub audio_channels: Option<u32>,
pub audio_ext: Option<String>,
pub columns: Option<u32>,
pub container: Option<String>,
pub downloader_options: Option<DownloaderOptions>,
pub dynamic_range: Option<String>,
pub ext: String,
pub filepath: Option<PathBuf>,
pub filesize: Option<u64>,
pub filesize_approx: Option<u64>,
pub format: Option<String>,
pub format_id: String,
pub format_index: Option<String>,
pub format_note: Option<String>,
pub fps: Option<f64>,
pub fragment_base_url: Option<Todo>,
pub fragments: Option<Vec<Fragment>>,
pub has_drm: Option<bool>,
pub height: Option<u32>,
pub http_headers: Option<HttpHeader>,
pub is_dash_periods: Option<bool>,
pub language: Option<String>,
pub language_preference: Option<i32>,
pub manifest_stream_number: Option<u32>,
pub manifest_url: Option<Url>,
pub preference: Option<i32>,
pub protocol: Option<String>,
pub quality: Option<f64>,
pub resolution: Option<String>,
pub rows: Option<u32>,
pub source_preference: Option<i32>,
pub tbr: Option<f64>,
pub url: Url,
pub vbr: Option<f64>,
pub vcodec: String,
pub video_ext: Option<String>,
pub width: Option<u32>,
}
#[derive(Debug, Deserialize, Serialize, Eq, PartialEq, PartialOrd, Ord)]
#[serde(deny_unknown_fields)]
pub struct DownloaderOptions {
http_chunk_size: u64,
}
#[derive(Debug, Deserialize, Serialize, Eq, PartialEq, PartialOrd, Ord)]
#[serde(deny_unknown_fields)]
pub struct HttpHeader {
#[serde(alias = "User-Agent")]
pub user_agent: Option<String>,
#[serde(alias = "Accept")]
pub accept: Option<String>,
#[serde(alias = "X-Forwarded-For")]
pub x_forwarded_for: Option<String>,
#[serde(alias = "Accept-Language")]
pub accept_language: Option<String>,
#[serde(alias = "Sec-Fetch-Mode")]
pub sec_fetch_mode: Option<String>,
}
#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd)]
#[serde(deny_unknown_fields)]
pub struct Fragment {
pub url: Option<Url>,
pub duration: Option<f64>,
pub path: Option<PathBuf>,
}
impl InfoJson {
pub fn to_py_dict(self, py: Python) -> PyResult<Bound<PyDict>> {
let output: Bound<PyDict> = json_loads_str(py, self)?;
Ok(output)
}
}
|