aboutsummaryrefslogtreecommitdiffstats
path: root/crates/yt_dlp
diff options
context:
space:
mode:
Diffstat (limited to 'crates/yt_dlp')
-rw-r--r--crates/yt_dlp/.cargo/config.toml12
-rw-r--r--crates/yt_dlp/Cargo.toml13
-rw-r--r--crates/yt_dlp/src/duration.rs78
-rw-r--r--crates/yt_dlp/src/error.rs68
-rw-r--r--crates/yt_dlp/src/lib.rs904
-rw-r--r--crates/yt_dlp/src/logging.rs144
-rw-r--r--crates/yt_dlp/src/progress_hook.rs41
-rw-r--r--crates/yt_dlp/src/python_json_decode_failed.error_msg5
-rw-r--r--crates/yt_dlp/src/python_json_decode_failed.error_msg.license9
-rw-r--r--crates/yt_dlp/src/tests.rs89
-rw-r--r--crates/yt_dlp/src/wrapper/info_json.rs827
-rw-r--r--crates/yt_dlp/src/wrapper/mod.rs12
-rw-r--r--crates/yt_dlp/src/wrapper/yt_dlp_options.rs62
13 files changed, 598 insertions, 1666 deletions
diff --git a/crates/yt_dlp/.cargo/config.toml b/crates/yt_dlp/.cargo/config.toml
deleted file mode 100644
index d84f14d..0000000
--- a/crates/yt_dlp/.cargo/config.toml
+++ /dev/null
@@ -1,12 +0,0 @@
-# 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>.
-
-[env]
-PYO3_PYTHON = "/nix/store/7xzk119acyws2c4ysygdv66l0grxkr39-python3-3.11.9-env/bin/python3"
diff --git a/crates/yt_dlp/Cargo.toml b/crates/yt_dlp/Cargo.toml
index b80c70f..ddd5f9b 100644
--- a/crates/yt_dlp/Cargo.toml
+++ b/crates/yt_dlp/Cargo.toml
@@ -10,7 +10,7 @@
[package]
name = "yt_dlp"
-description = "A wrapper around the python yt_dlp library"
+description = "A rust fii wrapper library for the python yt_dlp library"
keywords = []
categories = []
version.workspace = true
@@ -19,19 +19,16 @@ authors.workspace = true
license.workspace = true
repository.workspace = true
rust-version.workspace = true
-publish = false
+publish = true
[dependencies]
-pyo3 = { version = "0.24.0", features = ["auto-initialize"] }
-bytes.workspace = true
+indexmap = { version = "2.9.0", default-features = false }
log.workspace = true
-serde.workspace = true
+rustpython = { git = "https://github.com/RustPython/RustPython.git", features = ["threading", "stdlib", "stdio", "importlib", "ssl"], default-features = false }
serde_json.workspace = true
+thiserror = "2.0.12"
url.workspace = true
-[dev-dependencies]
-tokio.workspace = true
-
[lints]
workspace = true
diff --git a/crates/yt_dlp/src/duration.rs b/crates/yt_dlp/src/duration.rs
deleted file mode 100644
index 19181a5..0000000
--- a/crates/yt_dlp/src/duration.rs
+++ /dev/null
@@ -1,78 +0,0 @@
-// 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>.
-
-// TODO: This file should be de-duplicated with the same file in the 'yt' crate <2024-06-25>
-
-#[derive(Debug, Clone, Copy)]
-pub struct Duration {
- time: u32,
-}
-
-impl From<&str> for Duration {
- fn from(v: &str) -> Self {
- let buf: Vec<_> = v.split(':').take(2).collect();
- Self {
- time: (buf[0].parse::<u32>().expect("Should be a number") * 60)
- + buf[1].parse::<u32>().expect("Should be a number"),
- }
- }
-}
-
-impl From<Option<f64>> for Duration {
- fn from(value: Option<f64>) -> Self {
- Self {
- #[allow(
- clippy::cast_possible_truncation,
- clippy::cast_precision_loss,
- clippy::cast_sign_loss
- )]
- time: value.unwrap_or(0.0).ceil() as u32,
- }
- }
-}
-
-impl std::fmt::Display for Duration {
- fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
- const SECOND: u32 = 1;
- const MINUTE: u32 = 60 * SECOND;
- const HOUR: u32 = 60 * MINUTE;
-
- let base_hour = self.time - (self.time % HOUR);
- let base_min = (self.time % HOUR) - ((self.time % HOUR) % MINUTE);
- let base_sec = (self.time % HOUR) % MINUTE;
-
- let h = base_hour / HOUR;
- let m = base_min / MINUTE;
- let s = base_sec / SECOND;
-
- if self.time == 0 {
- write!(f, "0s")
- } else if h > 0 {
- write!(f, "{h}h {m}m")
- } else {
- write!(f, "{m}m {s}s")
- }
- }
-}
-#[cfg(test)]
-mod test {
- use super::Duration;
-
- #[test]
- fn test_display_duration_1h() {
- let dur = Duration { time: 60 * 60 };
- assert_eq!("1h 0m".to_owned(), dur.to_string());
- }
- #[test]
- fn test_display_duration_30min() {
- let dur = Duration { time: 60 * 30 };
- assert_eq!("30m 0s".to_owned(), dur.to_string());
- }
-}
diff --git a/crates/yt_dlp/src/error.rs b/crates/yt_dlp/src/error.rs
deleted file mode 100644
index 3881f0b..0000000
--- a/crates/yt_dlp/src/error.rs
+++ /dev/null
@@ -1,68 +0,0 @@
-// yt - A fully featured command line YouTube client
-//
-// Copyright (C) 2025 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::{fmt::Display, io};
-
-use pyo3::Python;
-
-#[derive(Debug)]
-#[allow(clippy::module_name_repetitions)]
-pub enum YtDlpError {
- ResponseParseError {
- error: serde_json::error::Error,
- },
- PythonError {
- error: Box<pyo3::PyErr>,
- kind: String,
- },
- IoError {
- error: io::Error,
- },
-}
-
-impl std::error::Error for YtDlpError {}
-
-impl Display for YtDlpError {
- fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
- match self {
- YtDlpError::ResponseParseError { error } => write!(
- f,
- include_str!("./python_json_decode_failed.error_msg"),
- error
- ),
- YtDlpError::PythonError { error, kind: _ } => write!(f, "Python error: {error}"),
- YtDlpError::IoError { error } => write!(f, "Io error: {error}"),
- }
- }
-}
-
-impl From<serde_json::error::Error> for YtDlpError {
- fn from(value: serde_json::error::Error) -> Self {
- Self::ResponseParseError { error: value }
- }
-}
-
-impl From<pyo3::PyErr> for YtDlpError {
- fn from(value: pyo3::PyErr) -> Self {
- Python::with_gil(|py| {
- let kind = value.get_type(py).to_string();
- Self::PythonError {
- error: Box::new(value),
- kind,
- }
- })
- }
-}
-
-impl From<io::Error> for YtDlpError {
- fn from(value: io::Error) -> Self {
- Self::IoError { error: value }
- }
-}
diff --git a/crates/yt_dlp/src/lib.rs b/crates/yt_dlp/src/lib.rs
index c6d9290..34b8a5d 100644
--- a/crates/yt_dlp/src/lib.rs
+++ b/crates/yt_dlp/src/lib.rs
@@ -1,549 +1,541 @@
-// 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>.
+//! The `yt_dlp` interface is completely contained in the [`YoutubeDL`] structure.
-// The pyo3 `pyfunction` proc-macros call unsafe functions internally, which trigger this lint.
-#![allow(unsafe_op_in_unsafe_fn)]
-#![allow(clippy::missing_errors_doc)]
+use std::io::Write;
+use std::mem;
+use std::{env, fs::File, path::PathBuf};
-use std::io::stderr;
-use std::{env, process};
-use std::{fs::File, io::Write};
-
-use std::{path::PathBuf, sync::Once};
-
-use crate::{duration::Duration, logging::setup_logging, wrapper::info_json::InfoJson};
-
-use bytes::Bytes;
-use error::YtDlpError;
-use log::{Level, debug, info, log_enabled};
-use pyo3::types::{PyString, PyTuple, PyTupleMethods};
-use pyo3::{
- Bound, PyAny, PyResult, Python, pyfunction,
- types::{PyAnyMethods, PyDict, PyDictMethods, PyList, PyListMethods, PyModule},
- wrap_pyfunction,
+use indexmap::IndexMap;
+use log::{Level, debug, error, info, log_enabled};
+use logging::setup_logging;
+use rustpython::vm::builtins::PyList;
+use rustpython::{
+ InterpreterConfig,
+ vm::{
+ self, Interpreter, PyObjectRef, PyRef, VirtualMachine,
+ builtins::{PyBaseException, PyDict, PyStr},
+ function::{FuncArgs, KwArgs, PosArgs},
+ },
};
-use serde::Serialize;
-use serde_json::{Map, Value};
use url::Url;
-pub mod duration;
-pub mod error;
-pub mod logging;
-pub mod wrapper;
-
-#[cfg(test)]
-mod tests;
+mod logging;
+pub mod progress_hook;
-/// Synchronisation helper, to ensure that we don't setup the logger multiple times
-static SYNC_OBJ: Once = Once::new();
+#[macro_export]
+macro_rules! json_get {
+ ($value:expr, $name:literal, $into:ident) => {
+ $crate::json_cast!($value.get($name).expect("Should exist"), $into)
+ };
+}
-/// Add a logger to the yt-dlp options.
-/// If you have an logger set (i.e. for rust), than this will log to rust
-///
-/// # Panics
-/// This should never panic.
-pub fn add_logger_and_sig_handler<'a>(
- opts: Bound<'a, PyDict>,
- py: Python<'_>,
-) -> PyResult<Bound<'a, PyDict>> {
- /// Is the specified record to be logged? Returns false for no,
- /// true for yes. Filters can either modify log records in-place or
- /// return a completely different record instance which will replace
- /// the original log record in any future processing of the event.
- #[pyfunction]
- fn filter_error_log(_py: Python<'_>, record: &Bound<'_, PyAny>) -> bool {
- // Filter out all error logs (they are propagated as rust errors)
- let levelname: String = record
- .getattr("levelname")
- .expect("This should exist")
- .extract()
- .expect("This should be a String");
+#[macro_export]
+macro_rules! json_cast {
+ ($value:expr, $into:ident) => {
+ $value.$into().expect(concat!(
+ "Should be able to cast value into ",
+ stringify!($into)
+ ))
+ };
+}
- let return_value = levelname.as_str() != "ERROR";
+/// The core of the `yt_dlp` interface.
+pub struct YoutubeDL {
+ interpreter: Interpreter,
+ youtube_dl_class: PyObjectRef,
+ yt_dlp_module: PyObjectRef,
+ options: serde_json::Map<String, serde_json::Value>,
+}
- if log_enabled!(Level::Debug) && !return_value {
- let message: String = record
- .call_method0("getMessage")
- .expect("This method exists")
- .extract()
- .expect("The message is a string");
+impl std::fmt::Debug for YoutubeDL {
+ fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
+ // TODO(@bpeetz): Use something useful here. <2025-06-13>
+ f.write_str("YoutubeDL")
+ }
+}
- debug!("Swollowed error message: '{message}'");
+impl YoutubeDL {
+ /// Construct this instance from options.
+ ///
+ /// # Panics
+ /// If `yt_dlp` changed their interface.
+ ///
+ /// # Errors
+ /// If a python call fails.
+ pub fn from_options(mut options: YoutubeDLOptions) -> Result<Self, build::Error> {
+ let mut settings = vm::Settings::default();
+ if let Ok(python_path) = env::var("PYTHONPATH") {
+ for path in python_path.split(':') {
+ settings.path_list.push(path.to_owned());
+ }
+ } else {
+ error!(
+ "No PYTHONPATH found or invalid utf8. \
+ This means, that you probably did not \
+ supply the yt_dlp!"
+ );
}
- return_value
- }
- setup_logging(py, "yt_dlp")?;
+ settings.install_signal_handlers = false;
- let logging = PyModule::import(py, "logging")?;
- let ytdl_logger = logging.call_method1("getLogger", ("yt_dlp",))?;
+ // NOTE(@bpeetz): Another value leads to an internal codegen error. <2025-06-13>
+ settings.optimize = 0;
- // Ensure that all events are logged by setting the log level to NOTSET (we filter on rust's side)
- // Also use this static, to ensure that we don't configure the logger every time
- SYNC_OBJ.call_once(|| {
- // Disable the SIGINT (Ctrl+C) handler, python installs.
- // This allows the user to actually stop the application with Ctrl+C.
- // This is here because it can only be run in the main thread and this was here already.
- py.run(
- c"\
-import signal
-signal.signal(signal.SIGINT, signal.SIG_DFL)",
- None,
- None,
- )
- .expect("This code should always work");
+ settings.isolated = true;
- let config_opts = PyDict::new(py);
- config_opts
- .set_item("level", 0)
- .expect("Setting this item should always work");
+ let interpreter = InterpreterConfig::new()
+ .init_stdlib()
+ .settings(settings)
+ .interpreter();
- logging
- .call_method("basicConfig", (), Some(&config_opts))
- .expect("This method exists");
- });
+ let output_options = options.options.clone();
- ytdl_logger.call_method1(
- "addFilter",
- (wrap_pyfunction!(filter_error_log, py).expect("This function can be wrapped"),),
- )?;
+ let (yt_dlp_module, youtube_dl_class) = match interpreter.enter(|vm| {
+ let yt_dlp_module = vm.import("yt_dlp", 0)?;
+ let class = yt_dlp_module.get_attr("YoutubeDL", vm)?;
- // This was taken from `ytcc`, I don't think it is still applicable
- // ytdl_logger.setattr("propagate", false)?;
- // let logging_null_handler = logging.call_method0("NullHandler")?;
- // ytdl_logger.setattr("addHandler", logging_null_handler)?;
+ let maybe_hook = mem::take(&mut options.progress_hook);
+ let opts = options.into_py_dict(vm);
+ if let Some(function) = maybe_hook {
+ opts.get_or_insert(vm, vm.new_pyobj("progress_hooks"), || {
+ let hook: PyObjectRef = vm.new_function("progress_hook", function).into();
+ vm.new_pyobj(vec![hook])
+ })
+ .expect("Should work?");
+ }
- opts.set_item("logger", ytdl_logger).expect("Should work");
+ {
+ // Unconditionally set a logger.
+ // Otherwise, yt_dlp will log to stderr.
- Ok(opts)
-}
+ /// Is the specified record to be logged? Returns false for no,
+ /// true for yes. Filters can either modify log records in-place or
+ /// return a completely different record instance which will replace
+ /// the original log record in any future processing of the event.
+ fn filter_error_log(mut input: FuncArgs, vm: &VirtualMachine) -> bool {
+ let record = input.args.remove(0);
-#[pyfunction]
-#[allow(clippy::too_many_lines)]
-#[allow(clippy::missing_panics_doc)]
-#[allow(clippy::items_after_statements)]
-#[allow(
- clippy::cast_possible_truncation,
- clippy::cast_sign_loss,
- clippy::cast_precision_loss
-)]
-pub fn progress_hook(py: Python<'_>, input: &Bound<'_, PyDict>) -> PyResult<()> {
- // Only add the handler, if the log-level is higher than Debug (this avoids covering debug
- // messages).
- if log_enabled!(Level::Debug) {
- return Ok(());
- }
+ // Filter out all error logs (they are propagated as rust errors)
+ let levelname: PyRef<PyStr> = record
+ .get_attr("levelname", vm)
+ .expect("This should exist")
+ .downcast()
+ .expect("This should be a String");
- // ANSI ESCAPE CODES Wrappers {{{
- // see: https://en.wikipedia.org/wiki/ANSI_escape_code#Control_Sequence_Introducer_commands
- const CSI: &str = "\x1b[";
- fn clear_whole_line() {
- eprint!("{CSI}2K");
- }
- fn move_to_col(x: usize) {
- eprint!("{CSI}{x}G");
- }
- // }}}
+ let return_value = levelname.as_str() != "ERROR";
- let input: Map<String, Value> = serde_json::from_str(&json_dumps(
- py,
- input
- .downcast::<PyAny>()
- .expect("Will always work")
- .to_owned(),
- )?)
- .expect("python's json is valid");
+ if log_enabled!(Level::Debug) && !return_value {
+ let message: String = {
+ let get_message = record.get_attr("getMessage", vm).expect("Is set");
+ let message: PyRef<PyStr> = get_message
+ .call((), vm)
+ .expect("Can be called")
+ .downcast()
+ .expect("Downcasting works");
- macro_rules! get {
- (@interrogate $item:ident, $type_fun:ident, $get_fun:ident, $name:expr) => {{
- let a = $item.get($name).expect(concat!(
- "The field '",
- stringify!($name),
- "' should exist."
- ));
+ message.as_str().to_owned()
+ };
- if a.$type_fun() {
- a.$get_fun().expect(
- "The should have been checked in the if guard, so unpacking here is fine",
- )
- } else {
- panic!(
- "Value {} => \n{}\n is not of type: {}",
- $name,
- a,
- stringify!($type_fun)
- );
- }
- }};
+ debug!("Swollowed error message: '{message}'");
+ }
+ return_value
+ }
- ($type_fun:ident, $get_fun:ident, $name1:expr, $name2:expr) => {{
- let a = get! {@interrogate input, is_object, as_object, $name1};
- let b = get! {@interrogate a, $type_fun, $get_fun, $name2};
- b
- }};
+ let logging = setup_logging(vm, "yt_dlp")?;
+ let ytdl_logger = {
+ let get_logger = logging.get_item("getLogger", vm)?;
+ get_logger.call(("yt_dlp",), vm)?
+ };
- ($type_fun:ident, $get_fun:ident, $name:expr) => {{
- get! {@interrogate input, $type_fun, $get_fun, $name}
- }};
- }
+ {
+ let args = FuncArgs::new(
+ PosArgs::new(vec![]),
+ KwArgs::new({
+ let mut map = IndexMap::new();
+ // Ensure that all events are logged by setting
+ // the log level to NOTSET (we filter on rust's side)
+ map.insert("level".to_owned(), vm.new_pyobj(0));
+ map
+ }),
+ );
- macro_rules! default_get {
- (@interrogate $item:ident, $default:expr, $get_fun:ident, $name:expr) => {{
- let a = if let Some(field) = $item.get($name) {
- field.$get_fun().unwrap_or($default)
- } else {
- $default
- };
- a
- }};
+ let basic_config = logging.get_item("basicConfig", vm)?;
+ basic_config.call(args, vm)?;
+ }
- ($get_fun:ident, $default:expr, $name1:expr, $name2:expr) => {{
- let a = get! {@interrogate input, is_object, as_object, $name1};
- let b = default_get! {@interrogate a, $default, $get_fun, $name2};
- b
- }};
+ {
+ let add_filter = ytdl_logger.get_attr("addFilter", vm)?;
+ add_filter.call(
+ (vm.new_function("yt_dlp_error_filter", filter_error_log),),
+ vm,
+ )?;
+ }
- ($get_fun:ident, $default:expr, $name:expr) => {{
- default_get! {@interrogate input, $default, $get_fun, $name}
- }};
- }
+ opts.set_item("logger", ytdl_logger, vm)?;
+ }
- macro_rules! c {
- ($color:expr, $format:expr) => {
- format!("\x1b[{}m{}\x1b[0m", $color, $format)
+ let youtube_dl_class = class.call((opts,), vm)?;
+
+ Ok::<_, PyRef<PyBaseException>>((yt_dlp_module, youtube_dl_class))
+ }) {
+ Ok(ok) => ok,
+ Err(err) => {
+ interpreter.finalize(Some(err));
+ return Err(build::Error::Python);
+ }
};
- }
- fn format_bytes(bytes: u64) -> String {
- let bytes = Bytes::new(bytes);
- bytes.to_string()
+ Ok(Self {
+ interpreter,
+ youtube_dl_class,
+ yt_dlp_module,
+ options: output_options,
+ })
}
- fn format_speed(speed: f64) -> String {
- #[allow(clippy::cast_possible_truncation, clippy::cast_sign_loss)]
- let bytes = Bytes::new(speed.floor() as u64);
- format!("{bytes}/s")
+ /// # Panics
+ ///
+ /// If `yt_dlp` changed their location or type of `__version__`.
+ pub fn version(&self) -> String {
+ let str_ref: PyRef<PyStr> = self.interpreter.enter_and_expect(
+ |vm| {
+ let version_module = self.yt_dlp_module.get_attr("version", vm)?;
+ let version = version_module.get_attr("__version__", vm)?;
+ let version = version.downcast().expect("This should always be a string");
+ Ok(version)
+ },
+ "yt_dlp version location has changed",
+ );
+ str_ref.to_string()
}
- let get_title = || -> String {
- match get! {is_string, as_str, "info_dict", "ext"} {
- "vtt" => {
- format!(
- "Subtitles ({})",
- default_get! {as_str, "<No Subtitle Language>", "info_dict", "name"}
- )
- }
- "webm" | "mp4" | "mp3" | "m4a" => {
- default_get! { as_str, "<No title>", "info_dict", "title"}.to_owned()
- }
- other => panic!("The extension '{other}' is not yet implemented"),
- }
- };
-
- match get! {is_string, as_str, "status"} {
- "downloading" => {
- let elapsed = default_get! {as_f64, 0.0f64, "elapsed"};
- let eta = default_get! {as_f64, 0.0, "eta"};
- let speed = default_get! {as_f64, 0.0, "speed"};
+ /// Download a given list of URLs.
+ /// Returns the paths they were downloaded to.
+ ///
+ /// # Errors
+ /// If one of the downloads error.
+ pub fn download(&self, urls: &[Url]) -> Result<Vec<PathBuf>, extract_info::Error> {
+ let mut out_paths = Vec::with_capacity(urls.len());
- let downloaded_bytes = get! {is_u64, as_u64, "downloaded_bytes"};
- let (total_bytes, bytes_is_estimate): (u64, &'static str) = {
- let total_bytes = default_get!(as_u64, 0, "total_bytes");
- if total_bytes == 0 {
- let maybe_estimate = default_get!(as_u64, 0, "total_bytes_estimate");
+ for url in urls {
+ info!("Started downloading url: '{url}'");
+ let info_json = self.extract_info(url, true, true)?;
- if maybe_estimate == 0 {
- // The download speed should be in bytes per second and the eta in seconds.
- // Thus multiplying them gets us the raw bytes (which were estimated by `yt_dlp`, from their `info.json`)
- let bytes_still_needed = (speed * eta).ceil() as u64;
-
- (downloaded_bytes + bytes_still_needed, "~")
- } else {
- (maybe_estimate, "~")
- }
- } else {
- (total_bytes, "")
- }
- };
- let percent: f64 = {
- if total_bytes == 0 {
- 100.0
- } else {
- (downloaded_bytes as f64 / total_bytes as f64) * 100.0
- }
+ // Try to work around yt-dlp type weirdness
+ let result_string = if let Some(filename) = info_json.get("filename") {
+ PathBuf::from(json_cast!(filename, as_str))
+ } else {
+ PathBuf::from(json_get!(
+ json_cast!(
+ json_get!(info_json, "requested_downloads", as_array)[0],
+ as_object
+ ),
+ "filename",
+ as_str
+ ))
};
- clear_whole_line();
- move_to_col(1);
-
- eprint!(
- "'{}' [{}/{} at {}] -> [{} of {}{} {}] ",
- c!("34;1", get_title()),
- c!("33;1", Duration::from(Some(elapsed))),
- c!("33;1", Duration::from(Some(eta))),
- c!("32;1", format_speed(speed)),
- c!("31;1", format_bytes(downloaded_bytes)),
- c!("31;1", bytes_is_estimate),
- c!("31;1", format_bytes(total_bytes)),
- c!("36;1", format!("{:.02}%", percent))
- );
- stderr().flush()?;
- }
- "finished" => {
- eprintln!("-> Finished downloading.");
+ out_paths.push(result_string);
+ info!("Finished downloading url");
}
- "error" => {
- // TODO: This should probably return an Err. But I'm not so sure where the error would
- // bubble up to (i.e., who would catch it) <2025-01-21>
- eprintln!("-> Error while downloading: {}", get_title());
- process::exit(1);
- }
- other => unreachable!("'{other}' should not be a valid state!"),
- };
- Ok(())
-}
-
-pub fn add_hooks<'a>(opts: Bound<'a, PyDict>, py: Python<'_>) -> PyResult<Bound<'a, PyDict>> {
- if let Some(hooks) = opts.get_item("progress_hooks")? {
- let hooks = hooks.downcast::<PyList>()?;
- hooks.append(wrap_pyfunction!(progress_hook, py)?)?;
-
- opts.set_item("progress_hooks", hooks)?;
- } else {
- // No hooks are set yet
- let hooks_list = PyList::new(py, &[wrap_pyfunction!(progress_hook, py)?])?;
-
- opts.set_item("progress_hooks", hooks_list)?;
+ Ok(out_paths)
}
- Ok(opts)
-}
+ /// `extract_info(self, url, download=True, ie_key=None, extra_info=None, process=True, force_generic_extractor=False)`
+ ///
+ /// Extract and return the information dictionary of the URL
+ ///
+ /// Arguments:
+ /// - `url` URL to extract
+ ///
+ /// Keyword arguments:
+ /// :`download` Whether to download videos
+ /// :`process` Whether to resolve all unresolved references (URLs, playlist items).
+ /// Must be True for download to work
+ ///
+ /// # Panics
+ /// If expectations about python fail to hold.
+ ///
+ /// # Errors
+ /// If python operations fail.
+ pub fn extract_info(
+ &self,
+ url: &Url,
+ download: bool,
+ process: bool,
+ ) -> Result<InfoJson, extract_info::Error> {
+ match self.interpreter.enter(|vm| {
+ let pos_args = PosArgs::new(vec![vm.new_pyobj(url.to_string())]);
-/// Take the result of the ie (may be modified) and resolve all unresolved
-/// references (URLs, playlist items).
-///
-/// It will also download the videos if 'download'.
-/// Returns the resolved `ie_result`.
-#[allow(clippy::missing_panics_doc)]
-pub fn process_ie_result(
- yt_dlp_opts: &Map<String, Value>,
- ie_result: InfoJson,
- download: bool,
-) -> Result<InfoJson, YtDlpError> {
- Python::with_gil(|py| -> Result<InfoJson, YtDlpError> {
- let opts = json_map_to_py_dict(yt_dlp_opts, py)?;
+ let kw_args = KwArgs::new({
+ let mut map = IndexMap::new();
+ map.insert("download".to_owned(), vm.new_pyobj(download));
+ map.insert("process".to_owned(), vm.new_pyobj(process));
+ map
+ });
- let instance = get_yt_dlp(py, opts)?;
+ let fun_args = FuncArgs::new(pos_args, kw_args);
- let args = {
- let ie_result = json_loads_str(py, ie_result)?;
- (ie_result,)
- };
+ let inner = self.youtube_dl_class.get_attr("extract_info", vm)?;
+ let result = inner
+ .call_with_args(fun_args, vm)?
+ .downcast::<PyDict>()
+ .expect("This is a dict");
- let kwargs = PyDict::new(py);
- kwargs.set_item("download", download)?;
-
- let result = instance
- .call_method("process_ie_result", args, Some(&kwargs))?
- .downcast_into::<PyDict>()
- .expect("This is a dict");
-
- let result_str = json_dumps(py, result.into_any())?;
-
- serde_json::from_str(&result_str).map_err(Into::into)
- })
-}
-
-/// `extract_info(self, url, download=True, ie_key=None, extra_info=None, process=True, force_generic_extractor=False)`
-///
-/// Extract and return the information dictionary of the URL
-///
-/// Arguments:
-/// @param url URL to extract
-///
-/// Keyword arguments:
-/// @param download Whether to download videos
-/// @param process Whether to resolve all unresolved references (URLs, playlist items).
-/// Must be True for download to work
-/// @param `ie_key` Use only the extractor with this key
-///
-/// @param `extra_info` Dictionary containing the extra values to add to the info (For internal use only)
-/// @`force_generic_extractor` Force using the generic extractor (Deprecated; use `ie_key`='Generic')
-#[allow(clippy::missing_panics_doc)]
-pub fn extract_info(
- yt_dlp_opts: &Map<String, Value>,
- url: &Url,
- download: bool,
- process: bool,
-) -> Result<InfoJson, YtDlpError> {
- Python::with_gil(|py| -> Result<InfoJson, YtDlpError> {
- let opts = json_map_to_py_dict(yt_dlp_opts, py)?;
+ // Resolve the generator object
+ if let Ok(generator) = result.get_item("entries", vm) {
+ if generator.payload_is::<PyList>() {
+ // already resolved. Do nothing
+ } else {
+ let max_backlog = self.options.get("playlistend").map_or(10000, |value| {
+ usize::try_from(value.as_u64().expect("Works")).expect("Should work")
+ });
- let instance = get_yt_dlp(py, opts)?;
- let args = (url.as_str(),);
+ let mut out = vec![];
+ let next = generator.get_attr("__next__", vm)?;
+ while let Ok(output) = next.call((), vm) {
+ out.push(output);
- let kwargs = PyDict::new(py);
- kwargs.set_item("download", download)?;
- kwargs.set_item("process", process)?;
+ if out.len() == max_backlog {
+ break;
+ }
+ }
+ result.set_item("entries", vm.new_pyobj(out), vm)?;
+ }
+ }
- let result = instance
- .call_method("extract_info", args, Some(&kwargs))?
- .downcast_into::<PyDict>()
- .expect("This is a dict");
+ let result = {
+ let sanitize = self.youtube_dl_class.get_attr("sanitize_info", vm)?;
+ let value = sanitize.call((result,), vm)?;
- // Resolve the generator object
- if let Some(generator) = result.get_item("entries")? {
- if generator.is_instance_of::<PyList>() {
- // already resolved. Do nothing
- } else {
- let max_backlog = yt_dlp_opts.get("playlistend").map_or(10000, |value| {
- usize::try_from(value.as_u64().expect("Works")).expect("Should work")
- });
+ value.downcast::<PyDict>().expect("This should stay a dict")
+ };
- let mut out = vec![];
- while let Ok(output) = generator.call_method0("__next__") {
- out.push(output);
+ let result_json = json_dumps(result, vm);
- if out.len() == max_backlog {
- break;
- }
+ if let Ok(confirm) = env::var("YT_STORE_INFO_JSON") {
+ if confirm == "yes" {
+ let mut file = File::create("output.info.json").unwrap();
+ write!(
+ file,
+ "{}",
+ serde_json::to_string_pretty(&serde_json::Value::Object(
+ result_json.clone()
+ ))
+ .expect("Valid json")
+ )
+ .unwrap();
}
- result.set_item("entries", out)?;
}
- }
-
- let result_str = json_dumps(py, result.into_any())?;
- if let Ok(confirm) = env::var("YT_STORE_INFO_JSON") {
- if confirm == "yes" {
- let mut file = File::create("output.info.json")?;
- write!(file, "{result_str}").unwrap();
+ Ok::<_, PyRef<PyBaseException>>(result_json)
+ }) {
+ Ok(ok) => Ok(ok),
+ Err(err) => {
+ self.interpreter.enter(|vm| {
+ vm.print_exception(err);
+ });
+ Err(extract_info::Error::Python)
}
}
+ }
- serde_json::from_str(&result_str).map_err(Into::into)
- })
-}
+ /// Take the (potentially modified) result of the information extractor (i.e.,
+ /// [`Self::extract_info`] with `process` and `download` set to false)
+ /// and resolve all unresolved references (URLs,
+ /// playlist items).
+ ///
+ /// It will also download the videos if 'download' is true.
+ /// Returns the resolved `ie_result`.
+ ///
+ /// # Panics
+ /// If expectations about python fail to hold.
+ ///
+ /// # Errors
+ /// If python operations fail.
+ pub fn process_ie_result(
+ &self,
+ ie_result: InfoJson,
+ download: bool,
+ ) -> Result<InfoJson, process_ie_result::Error> {
+ match self.interpreter.enter(|vm| {
+ let pos_args = PosArgs::new(vec![vm.new_pyobj(json_loads(ie_result, vm))]);
-/// # Panics
-/// Only if python fails to return a valid URL.
-pub fn unsmuggle_url(smug_url: &Url) -> PyResult<Url> {
- Python::with_gil(|py| {
- let utils = get_yt_dlp_utils(py)?;
- let url = utils
- .call_method1("unsmuggle_url", (smug_url.as_str(),))?
- .downcast::<PyTuple>()?
- .get_item(0)?;
+ let kw_args = KwArgs::new({
+ let mut map = IndexMap::new();
+ map.insert("download".to_owned(), vm.new_pyobj(download));
+ map
+ });
- let url: Url = url
- .downcast::<PyString>()?
- .to_string()
- .parse()
- .expect("Python should be able to return a valid url");
+ let fun_args = FuncArgs::new(pos_args, kw_args);
- Ok(url)
- })
-}
+ let inner = self.youtube_dl_class.get_attr("process_ie_result", vm)?;
+ let result = inner
+ .call_with_args(fun_args, vm)?
+ .downcast::<PyDict>()
+ .expect("This is a dict");
-/// Download a given list of URLs.
-/// Returns the paths they were downloaded to.
-///
-/// # Panics
-/// Only if `yt_dlp` changes their `info_json` schema.
-pub fn download(
- urls: &[Url],
- download_options: &Map<String, Value>,
-) -> Result<Vec<PathBuf>, YtDlpError> {
- let mut out_paths = Vec::with_capacity(urls.len());
+ let result = {
+ let sanitize = self.youtube_dl_class.get_attr("sanitize_info", vm)?;
+ let value = sanitize.call((result,), vm)?;
- for url in urls {
- info!("Started downloading url: '{}'", url);
- let info_json = extract_info(download_options, url, true, true)?;
+ value.downcast::<PyDict>().expect("This should stay a dict")
+ };
- // Try to work around yt-dlp type weirdness
- let result_string = if let Some(filename) = info_json.filename {
- filename
- } else {
- info_json.requested_downloads.expect("This must exist")[0]
- .filename
- .clone()
- };
+ let result_json = json_dumps(result, vm);
- out_paths.push(result_string);
- info!("Finished downloading url: '{}'", url);
+ Ok::<_, PyRef<PyBaseException>>(result_json)
+ }) {
+ Ok(ok) => Ok(ok),
+ Err(err) => {
+ self.interpreter.enter(|vm| {
+ vm.print_exception(err);
+ });
+ Err(process_ie_result::Error::Python)
+ }
+ }
}
-
- Ok(out_paths)
}
-fn json_map_to_py_dict<'a>(
- map: &Map<String, Value>,
- py: Python<'a>,
-) -> PyResult<Bound<'a, PyDict>> {
- let json_string = serde_json::to_string(&map).expect("This must always work");
+#[allow(missing_docs)]
+pub mod process_ie_result {
+ #[derive(Debug, thiserror::Error, Clone, Copy)]
+ pub enum Error {
+ #[error("Python threw an exception")]
+ Python,
+ }
+}
+#[allow(missing_docs)]
+pub mod extract_info {
+ #[derive(Debug, thiserror::Error, Clone, Copy)]
+ pub enum Error {
+ #[error("Python threw an exception")]
+ Python,
+ }
+}
- let python_dict = json_loads(py, json_string)?;
+pub type InfoJson = serde_json::Map<String, serde_json::Value>;
+pub type ProgressHookFunction = fn(input: FuncArgs, vm: &VirtualMachine);
- Ok(python_dict)
+/// Options, that are used to customize the download behaviour.
+///
+/// In the future, this might get a Builder api.
+///
+/// See `help(yt_dlp.YoutubeDL())` from python for a full list of available options.
+#[derive(Default, Debug)]
+pub struct YoutubeDLOptions {
+ options: serde_json::Map<String, serde_json::Value>,
+ progress_hook: Option<ProgressHookFunction>,
}
-fn json_dumps(py: Python<'_>, input: Bound<'_, PyAny>) -> PyResult<String> {
- // json.dumps(yt_dlp.sanitize_info(input))
+impl YoutubeDLOptions {
+ #[must_use]
+ pub fn new() -> Self {
+ Self {
+ options: serde_json::Map::new(),
+ progress_hook: None,
+ }
+ }
+
+ #[must_use]
+ pub fn set(self, key: impl Into<String>, value: impl Into<serde_json::Value>) -> Self {
+ let mut options = self.options;
+ options.insert(key.into(), value.into());
+
+ Self {
+ options,
+ progress_hook: self.progress_hook,
+ }
+ }
- let yt_dlp = get_yt_dlp(py, PyDict::new(py))?;
- let sanitized_result = yt_dlp.call_method1("sanitize_info", (input,))?;
+ #[must_use]
+ pub fn with_progress_hook(self, progress_hook: ProgressHookFunction) -> Self {
+ if let Some(_previous_hook) = self.progress_hook {
+ todo!()
+ } else {
+ Self {
+ options: self.options,
+ progress_hook: Some(progress_hook),
+ }
+ }
+ }
- let json = PyModule::import(py, "json")?;
- let dumps = json.getattr("dumps")?;
+ /// # Errors
+ /// If the underlying [`YoutubeDL::from_options`] errors.
+ pub fn build(self) -> Result<YoutubeDL, build::Error> {
+ YoutubeDL::from_options(self)
+ }
- let output = dumps.call1((sanitized_result,))?;
+ #[must_use]
+ pub fn from_json_options(options: serde_json::Map<String, serde_json::Value>) -> Self {
+ Self {
+ options,
+ progress_hook: None,
+ }
+ }
- let output_str = output.extract::<String>()?;
+ #[must_use]
+ pub fn get(&self, key: &str) -> Option<&serde_json::Value> {
+ self.options.get(key)
+ }
- Ok(output_str)
+ fn into_py_dict(self, vm: &VirtualMachine) -> PyRef<PyDict> {
+ json_loads(self.options, vm)
+ }
}
-fn json_loads_str<T: Serialize>(py: Python<'_>, input: T) -> PyResult<Bound<'_, PyDict>> {
- let string = serde_json::to_string(&input).expect("Correct json must be pased");
+#[allow(missing_docs)]
+pub mod build {
+ #[derive(Debug, thiserror::Error)]
+ pub enum Error {
+ #[error("Python threw an exception")]
+ Python,
- json_loads(py, string)
+ #[error("Io error: {0}")]
+ Io(#[from] std::io::Error),
+ }
}
-fn json_loads(py: Python<'_>, input: String) -> PyResult<Bound<'_, PyDict>> {
- // json.loads(input)
+fn json_loads(
+ input: serde_json::Map<String, serde_json::Value>,
+ vm: &VirtualMachine,
+) -> PyRef<PyDict> {
+ let json = vm.import("json", 0).expect("Module exists");
+ let loads = json.get_attr("loads", vm).expect("Method exists");
+ let self_str = serde_json::to_string(&serde_json::Value::Object(input)).expect("Vaild json");
+ let dict = loads
+ .call((self_str,), vm)
+ .expect("Vaild json is always a valid dict");
- let json = PyModule::import(py, "json")?;
- let dumps = json.getattr("loads")?;
+ dict.downcast().expect("Should always be a dict")
+}
- let output = dumps.call1((input,))?;
+/// # Panics
+/// If expectation about python operations fail.
+pub fn json_dumps(
+ input: PyRef<PyDict>,
+ vm: &VirtualMachine,
+) -> serde_json::Map<String, serde_json::Value> {
+ let json = vm.import("json", 0).expect("Module exists");
+ let dumps = json.get_attr("dumps", vm).expect("Method exists");
+ let dict = dumps
+ .call((input,), vm)
+ .map_err(|err| vm.print_exception(err))
+ .expect("Might not always work, but for our dicts it works");
- Ok(output
- .downcast::<PyDict>()
- .expect("This should always be a PyDict")
- .clone())
-}
+ let string: PyRef<PyStr> = dict.downcast().expect("Should always be a string");
-fn get_yt_dlp_utils(py: Python<'_>) -> PyResult<Bound<'_, PyAny>> {
- let yt_dlp = PyModule::import(py, "yt_dlp")?;
- let utils = yt_dlp.getattr("utils")?;
+ let real_string = string.to_str().expect("Should be valid utf8");
- Ok(utils)
-}
-fn get_yt_dlp<'a>(py: Python<'a>, opts: Bound<'a, PyDict>) -> PyResult<Bound<'a, PyAny>> {
- // Unconditionally set a logger
- let opts = add_logger_and_sig_handler(opts, py)?;
- let opts = add_hooks(opts, py)?;
+ // {
+ // let mut file = File::create("debug.dump.json").unwrap();
+ // write!(file, "{}", real_string).unwrap();
+ // }
- let yt_dlp = PyModule::import(py, "yt_dlp")?;
- let youtube_dl = yt_dlp.call_method1("YoutubeDL", (opts,))?;
+ let value: serde_json::Value = serde_json::from_str(real_string).expect("Should be valid json");
- Ok(youtube_dl)
+ match value {
+ serde_json::Value::Object(map) => map,
+ _ => unreachable!("These should not be json.dumps output"),
+ }
}
diff --git a/crates/yt_dlp/src/logging.rs b/crates/yt_dlp/src/logging.rs
index e731502..5cb4c1d 100644
--- a/crates/yt_dlp/src/logging.rs
+++ b/crates/yt_dlp/src/logging.rs
@@ -10,34 +10,66 @@
// This file is taken from: https://github.com/dylanbstorey/pyo3-pylogger/blob/d89e0d6820ebc4f067647e3b74af59dbc4941dd5/src/lib.rs
// It is licensed under the Apache 2.0 License, copyright up to 2024 by Dylan Storey
-// It was modified by Benedikt Peetz 2024
-
-// The pyo3 `pyfunction` proc-macros call unsafe functions internally, which trigger this lint.
-#![allow(unsafe_op_in_unsafe_fn)]
-
-use std::ffi::CString;
+// It was modified by Benedikt Peetz 2024, 2025
use log::{Level, MetadataBuilder, Record, logger};
-use pyo3::{
- Bound, PyAny, PyResult, Python,
- prelude::{PyAnyMethods, PyListMethods, PyModuleMethods},
- pyfunction, wrap_pyfunction,
+use rustpython::vm::{
+ PyObjectRef, PyRef, PyResult, VirtualMachine,
+ builtins::{PyInt, PyList, PyStr},
+ convert::ToPyObject,
+ function::FuncArgs,
};
/// Consume a Python `logging.LogRecord` and emit a Rust `Log` instead.
-#[allow(clippy::needless_pass_by_value)]
-#[pyfunction]
-fn host_log(record: Bound<'_, PyAny>, rust_target: &str) -> PyResult<()> {
- let level = record.getattr("levelno")?;
- let message = record.getattr("getMessage")?.call0()?.to_string();
- let pathname = record.getattr("pathname")?.to_string();
- let lineno = record
- .getattr("lineno")?
- .to_string()
- .parse::<u32>()
- .expect("This should always be a u32");
+fn host_log(mut input: FuncArgs, vm: &VirtualMachine) -> PyResult<()> {
+ let record = input.args.remove(0);
+ let rust_target = {
+ let base: PyRef<PyStr> = input.args.remove(0).downcast().expect("Should be a string");
+ base.as_str().to_owned()
+ };
+
+ let level = {
+ let level: PyRef<PyInt> = record
+ .get_attr("levelno", vm)?
+ .downcast()
+ .expect("Should always be an int");
+ level.as_u32_mask()
+ };
+ let message = {
+ let get_message = record.get_attr("getMessage", vm)?;
+ let message: PyRef<PyStr> = get_message
+ .call((), vm)?
+ .downcast()
+ .expect("Downcasting works");
+
+ message.as_str().to_owned()
+ };
+
+ let pathname = {
+ let pathname: PyRef<PyStr> = record
+ .get_attr("pathname", vm)?
+ .downcast()
+ .expect("Is a string");
+
+ pathname.as_str().to_owned()
+ };
+
+ let lineno = {
+ let lineno: PyRef<PyInt> = record
+ .get_attr("lineno", vm)?
+ .downcast()
+ .expect("Is a number");
- let logger_name = record.getattr("name")?.to_string();
+ lineno.as_u32_mask()
+ };
+
+ let logger_name = {
+ let name: PyRef<PyStr> = record
+ .get_attr("name", vm)?
+ .downcast()
+ .expect("Should be a string");
+ name.as_str().to_owned()
+ };
let full_target: Option<String> = if logger_name.trim().is_empty() || logger_name == "root" {
None
@@ -48,25 +80,25 @@ fn host_log(record: Bound<'_, PyAny>, rust_target: &str) -> PyResult<()> {
Some(format!("{rust_target}::{logger_name}"))
};
- let target = full_target.as_deref().unwrap_or(rust_target);
+ let target = full_target.as_deref().unwrap_or(&rust_target);
// error
- let error_metadata = if level.ge(40u8)? {
+ let error_metadata = if level >= 40 {
MetadataBuilder::new()
.target(target)
.level(Level::Error)
.build()
- } else if level.ge(30u8)? {
+ } else if level >= 30 {
MetadataBuilder::new()
.target(target)
.level(Level::Warn)
.build()
- } else if level.ge(20u8)? {
+ } else if level >= 20 {
MetadataBuilder::new()
.target(target)
.level(Level::Info)
.build()
- } else if level.ge(10u8)? {
+ } else if level >= 10 {
MetadataBuilder::new()
.target(target)
.level(Level::Debug)
@@ -98,13 +130,24 @@ fn host_log(record: Bound<'_, PyAny>, rust_target: &str) -> PyResult<()> {
/// # Panics
/// Only if internal assertions fail.
#[allow(clippy::module_name_repetitions)]
-pub fn setup_logging(py: Python<'_>, target: &str) -> PyResult<()> {
- let logging = py.import("logging")?;
+pub(super) fn setup_logging(vm: &VirtualMachine, target: &str) -> PyResult<PyObjectRef> {
+ let logging = vm.import("logging", 0)?;
- logging.setattr("host_log", wrap_pyfunction!(host_log, &logging)?)?;
+ let scope = vm.new_scope_with_builtins();
- py.run(
- CString::new(format!(
+ for (key, value) in logging.dict().expect("Should be a dict") {
+ let key: PyRef<PyStr> = key.downcast().expect("Is a string");
+
+ scope.globals.set_item(key.as_str(), value, vm)?;
+ }
+ scope
+ .globals
+ .set_item("host_log", vm.new_function("host_log", host_log).into(), vm)?;
+
+ let local_scope = scope.clone();
+ vm.run_code_string(
+ local_scope,
+ format!(
r#"
class HostHandler(Handler):
def __init__(self, level=0):
@@ -119,15 +162,36 @@ def basicConfig(*pargs, **kwargs):
kwargs["handlers"] = [HostHandler()]
return oldBasicConfig(*pargs, **kwargs)
"#
- ))
- .expect("This is hardcoded")
- .as_c_str(),
- Some(&logging.dict()),
- None,
+ )
+ .as_str(),
+ "<embedded logging inintializing code>".to_owned(),
)?;
- let all = logging.index()?;
- all.append("HostHandler")?;
+ let all: PyRef<PyList> = logging
+ .get_attr("__all__", vm)?
+ .downcast()
+ .expect("Is a list");
+ all.borrow_vec_mut().push(vm.new_pyobj("HostHandler"));
- Ok(())
+ // {
+ // let logging_dict = logging.dict().expect("Exists");
+ //
+ // for (key, val) in scope.globals {
+ // let key: PyRef<PyStr> = key.downcast().expect("Is a string");
+ //
+ // if !logging_dict.contains_key(key.as_str(), vm) {
+ // logging_dict.set_item(key.as_str(), val, vm)?;
+ // }
+ // }
+ //
+ // for (key, val) in scope.locals {
+ // let key: PyRef<PyStr> = key.downcast().expect("Is a string");
+ //
+ // if !logging_dict.contains_key(key.as_str(), vm) {
+ // logging_dict.set_item(key.as_str(), val, vm)?;
+ // }
+ // }
+ // }
+
+ Ok(scope.globals.to_pyobject(vm))
}
diff --git a/crates/yt_dlp/src/progress_hook.rs b/crates/yt_dlp/src/progress_hook.rs
new file mode 100644
index 0000000..7a7628a
--- /dev/null
+++ b/crates/yt_dlp/src/progress_hook.rs
@@ -0,0 +1,41 @@
+#[macro_export]
+macro_rules! mk_python_function {
+ ($name:ident, $new_name:ident) => {
+ pub fn $new_name(
+ mut args: $crate::progress_hook::rustpython::vm::function::FuncArgs,
+ vm: &$crate::progress_hook::rustpython::vm::VirtualMachine,
+ ) {
+ use $crate::progress_hook::rustpython;
+
+ let input = {
+ let dict: rustpython::vm::PyRef<rustpython::vm::builtins::PyDict> = args
+ .args
+ .remove(0)
+ .downcast()
+ .expect("The progress hook is always called with these args");
+ let new_dict = rustpython::vm::builtins::PyDict::new_ref(&vm.ctx);
+ dict.into_iter()
+ .filter_map(|(name, value)| {
+ let real_name: rustpython::vm::PyRefExact<rustpython::vm::builtins::PyStr> =
+ name.downcast_exact(vm).expect("Is a string");
+ let name_str = real_name.to_str().expect("Is a string");
+ if name_str.starts_with('_') {
+ None
+ } else {
+ Some((name_str.to_owned(), value))
+ }
+ })
+ .for_each(|(key, value)| {
+ new_dict
+ .set_item(&key, value, vm)
+ .expect("This is a transpositions, should always be valid");
+ });
+
+ $crate::json_dumps(new_dict, vm)
+ };
+ $name(input).expect("Shall not fail!");
+ }
+ };
+}
+
+pub use rustpython;
diff --git a/crates/yt_dlp/src/python_json_decode_failed.error_msg b/crates/yt_dlp/src/python_json_decode_failed.error_msg
deleted file mode 100644
index d10688e..0000000
--- a/crates/yt_dlp/src/python_json_decode_failed.error_msg
+++ /dev/null
@@ -1,5 +0,0 @@
-Failed to decode yt-dlp's response: {}
-
-This is probably a bug.
-Try running the command again with the `YT_STORE_INFO_JSON=yes` environment variable set
-and maybe debug it further via `yt check info-json output.info.json`.
diff --git a/crates/yt_dlp/src/python_json_decode_failed.error_msg.license b/crates/yt_dlp/src/python_json_decode_failed.error_msg.license
deleted file mode 100644
index 7813eb6..0000000
--- a/crates/yt_dlp/src/python_json_decode_failed.error_msg.license
+++ /dev/null
@@ -1,9 +0,0 @@
-yt - A fully featured command line YouTube client
-
-Copyright (C) 2025 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>.
diff --git a/crates/yt_dlp/src/tests.rs b/crates/yt_dlp/src/tests.rs
deleted file mode 100644
index 91b6626..0000000
--- a/crates/yt_dlp/src/tests.rs
+++ /dev/null
@@ -1,89 +0,0 @@
-// 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::sync::LazyLock;
-
-use serde_json::{Value, json};
-use url::Url;
-
-static YT_OPTS: LazyLock<serde_json::Map<String, Value>> = LazyLock::new(|| {
- match json!({
- "playliststart": 1,
- "playlistend": 10,
- "noplaylist": false,
- "extract_flat": false,
- }) {
- Value::Object(obj) => obj,
- _ => unreachable!("This json is hardcoded"),
- }
-});
-
-#[tokio::test]
-#[ignore = "This test hangs forever"]
-async fn test_extract_info_video() {
- let info = crate::extract_info(
- &YT_OPTS,
- &Url::parse("https://www.youtube.com/watch?v=dbjPnXaacAU").expect("Is valid."),
- false,
- false,
- )
- .await
- .map_err(|err| format!("Encountered error: '{err}'"))
- .unwrap();
-
- println!("{info:#?}");
-}
-
-#[tokio::test]
-#[ignore = "This test hangs forever"]
-async fn test_extract_info_url() {
- let err = crate::extract_info(
- &YT_OPTS,
- &Url::parse("https://google.com").expect("Is valid."),
- false,
- false,
- )
- .await
- .map_err(|err| format!("Encountered error: '{err}'"))
- .unwrap();
-
- println!("{err:#?}");
-}
-
-#[tokio::test]
-#[ignore = "This test hangs forever"]
-async fn test_extract_info_playlist() {
- let err = crate::extract_info(
- &YT_OPTS,
- &Url::parse("https://www.youtube.com/@TheGarriFrischer/videos").expect("Is valid."),
- false,
- true,
- )
- .await
- .map_err(|err| format!("Encountered error: '{err}'"))
- .unwrap();
-
- println!("{err:#?}");
-}
-#[tokio::test]
-#[ignore = "This test hangs forever"]
-async fn test_extract_info_playlist_full() {
- let err = crate::extract_info(
- &YT_OPTS,
- &Url::parse("https://www.youtube.com/@NixOS-Foundation/videos").expect("Is valid."),
- false,
- true,
- )
- .await
- .map_err(|err| format!("Encountered error: '{err}'"))
- .unwrap();
-
- println!("{err:#?}");
-}
diff --git a/crates/yt_dlp/src/wrapper/info_json.rs b/crates/yt_dlp/src/wrapper/info_json.rs
deleted file mode 100644
index ea73d26..0000000
--- a/crates/yt_dlp/src/wrapper/info_json.rs
+++ /dev/null
@@ -1,827 +0,0 @@
-// 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>.
-
-// `yt_dlp` named them like this.
-#![allow(clippy::pub_underscore_fields)]
-
-use std::{collections::HashMap, path::PathBuf};
-
-use pyo3::{Bound, PyResult, Python, types::PyDict};
-use serde::{Deserialize, Deserializer, Serialize};
-use serde_json::Value;
-use url::Url;
-
-use crate::json_loads_str;
-
-type Todo = String;
-type Extractor = String;
-type ExtractorKey = 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 {
- #[serde(skip_serializing_if = "Option::is_none")]
- pub __files_to_move: Option<FilesToMove>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub __last_playlist_index: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub __post_extractor: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub __x_forwarded_for_ip: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub _filename: Option<PathBuf>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub _format_sort_fields: Option<Vec<String>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub _has_drm: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub _type: Option<InfoType>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub _version: Option<Version>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub abr: Option<f64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub acodec: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub age_limit: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub artists: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub aspect_ratio: Option<f64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub asr: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub audio_channels: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub audio_ext: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub automatic_captions: Option<HashMap<String, Vec<Caption>>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub availability: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub average_rating: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub categories: Option<Vec<String>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub channel: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub channel_follower_count: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub channel_id: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub channel_is_verified: Option<bool>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub channel_url: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub chapters: Option<Vec<Chapter>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub comment_count: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub comments: Option<Vec<Comment>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub concurrent_view_count: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub container: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub description: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub direct: Option<bool>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub display_id: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub downloader_options: Option<DownloaderOptions>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub duration: Option<f64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub duration_string: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub dynamic_range: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub entries: Option<Vec<InfoJson>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub episode: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub episode_number: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub epoch: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub ext: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub extractor: Option<Extractor>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub extractor_key: Option<ExtractorKey>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub filename: Option<PathBuf>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub filesize: Option<u64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub filesize_approx: Option<u64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub format: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub format_id: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub format_index: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub format_note: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub formats: Option<Vec<Format>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub fps: Option<f64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub fulltitle: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub genre: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub genres: Option<Vec<String>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub has_drm: Option<bool>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub heatmap: Option<Vec<HeatMapEntry>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub height: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub hls_aes: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub http_headers: Option<HttpHeader>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub id: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub ie_key: Option<ExtractorKey>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub is_live: Option<bool>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub language: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub language_preference: Option<i32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub license: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub like_count: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub live_status: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub location: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub manifest_url: Option<Url>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub media_type: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub modified_date: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub n_entries: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub original_url: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playable_in_embed: Option<bool>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_autonumber: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_channel: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_channel_id: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_count: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_id: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_index: Option<u64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_title: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_uploader: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_uploader_id: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub playlist_webpage_url: Option<Url>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub preference: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub protocol: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub quality: Option<f64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub release_date: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub release_timestamp: Option<u64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub release_year: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub repost_count: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub requested_downloads: Option<Vec<RequestedDownloads>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub requested_entries: Option<Vec<u32>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub requested_formats: Option<Vec<Format>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub requested_subtitles: Option<HashMap<String, Subtitle>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub resolution: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub season: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub season_number: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub series: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub source_preference: Option<i32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub sponsorblock_chapters: Option<Vec<SponsorblockChapter>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub start_time: Option<f64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub stretched_ratio: Option<Todo>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub subtitles: Option<HashMap<String, Vec<Caption>>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub tags: Option<Vec<String>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub tbr: Option<f64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub thumbnail: Option<Url>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub thumbnails: Option<Vec<ThumbNail>>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub timestamp: Option<f64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub title: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub upload_date: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub uploader: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub uploader_id: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub uploader_url: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub url: Option<Url>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub vbr: Option<f64>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub vcodec: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub video_ext: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub view_count: Option<u32>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub was_live: Option<bool>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub webpage_url: Option<Url>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub webpage_url_basename: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub webpage_url_domain: Option<String>,
-
- #[serde(skip_serializing_if = "Option::is_none")]
- pub width: Option<u32>,
-}
-
-#[derive(Debug, Deserialize, Serialize, PartialEq)]
-#[serde(deny_unknown_fields)]
-#[allow(missing_copy_implementations)]
-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: Option<f64>,
- pub asr: Option<u32>,
- pub audio_channels: Option<u32>,
- pub audio_ext: Option<String>,
- pub chapters: Option<Vec<SponsorblockChapter>>,
- pub duration: Option<f64>,
- pub dynamic_range: Option<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: Option<String>,
- pub fps: Option<f64>,
- pub has_drm: Option<bool>,
- pub height: Option<u32>,
- pub http_headers: Option<HttpHeader>,
- pub infojson_filename: PathBuf,
- pub language: Option<String>,
- pub manifest_url: Option<Url>,
- pub protocol: String,
- pub quality: Option<i64>,
- pub requested_formats: Option<Vec<Format>>,
- pub resolution: String,
- pub tbr: f64,
- pub url: Option<Url>,
- pub vbr: f64,
- pub vcodec: String,
- pub video_ext: Option<String>,
- pub width: Option<u32>,
-}
-
-#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd)]
-#[serde(deny_unknown_fields)]
-pub struct Subtitle {
- pub ext: SubtitleExt,
- pub filepath: PathBuf,
- pub filesize: Option<u64>,
- pub fragment_base_url: Option<Url>,
- pub fragments: Option<Vec<Fragment>>,
- pub manifest_url: Option<Url>,
- pub name: Option<String>,
- pub protocol: Option<Todo>,
- pub url: Url,
-}
-
-#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd, Ord, Eq, Clone, Copy)]
-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, PartialOrd)]
-#[serde(deny_unknown_fields)]
-pub struct Caption {
- pub ext: SubtitleExt,
- pub filepath: Option<PathBuf>,
- pub filesize: Option<u64>,
- pub fragments: Option<Vec<SubtitleFragment>>,
- pub fragment_base_url: Option<Url>,
- pub manifest_url: Option<Url>,
- pub name: Option<String>,
- pub protocol: Option<String>,
- pub url: String,
- pub video_id: Option<String>,
-}
-
-#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd)]
-#[serde(deny_unknown_fields)]
-pub struct SubtitleFragment {
- path: PathBuf,
- duration: Option<f64>,
-}
-
-#[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, Clone, Copy)]
-#[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, Clone, Copy)]
-#[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)]
-#[allow(missing_copy_implementations)]
-pub struct HeatMapEntry {
- pub start_time: f64,
- pub end_time: f64,
- pub value: f64,
-}
-
-#[derive(Debug, Deserialize, Serialize, PartialEq, PartialOrd, Ord, Eq, Clone, Copy)]
-#[serde(deny_unknown_fields)]
-pub enum InfoType {
- #[serde(alias = "playlist")]
- #[serde(rename(serialize = "playlist"))]
- Playlist,
-
- #[serde(alias = "url")]
- #[serde(rename(serialize = "url"))]
- Url,
-
- #[serde(alias = "video")]
- #[serde(rename(serialize = "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 {
- #[must_use]
- 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)]
-#[allow(clippy::struct_excessive_bools)]
-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: Option<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 is_live: 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)]
-#[allow(missing_copy_implementations)]
-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 duration: Option<f64>,
- pub fragment_count: Option<usize>,
- pub path: Option<PathBuf>,
- pub url: Option<Url>,
-}
-
-impl InfoJson {
- pub fn to_py_dict(self, py: Python<'_>) -> PyResult<Bound<'_, PyDict>> {
- let output: Bound<'_, PyDict> = json_loads_str(py, self)?;
- Ok(output)
- }
-}
diff --git a/crates/yt_dlp/src/wrapper/mod.rs b/crates/yt_dlp/src/wrapper/mod.rs
deleted file mode 100644
index 3fe3247..0000000
--- a/crates/yt_dlp/src/wrapper/mod.rs
+++ /dev/null
@@ -1,12 +0,0 @@
-// 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>.
-
-pub mod info_json;
-// pub mod yt_dlp_options;
diff --git a/crates/yt_dlp/src/wrapper/yt_dlp_options.rs b/crates/yt_dlp/src/wrapper/yt_dlp_options.rs
deleted file mode 100644
index 25595b5..0000000
--- a/crates/yt_dlp/src/wrapper/yt_dlp_options.rs
+++ /dev/null
@@ -1,62 +0,0 @@
-// 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 pyo3::{Bound, PyResult, Python, types::PyDict};
-use serde::Serialize;
-
-use crate::json_loads;
-
-#[derive(Serialize, Clone)]
-pub struct YtDlpOptions {
- pub playliststart: u32,
- pub playlistend: u32,
- pub noplaylist: bool,
- pub extract_flat: ExtractFlat,
- // pub extractor_args: ExtractorArgs,
- // pub format: String,
- // pub fragment_retries: u32,
- // #[serde(rename(serialize = "getcomments"))]
- // pub get_comments: bool,
- // #[serde(rename(serialize = "ignoreerrors"))]
- // pub ignore_errors: bool,
- // pub retries: u32,
- // #[serde(rename(serialize = "writeinfojson"))]
- // pub write_info_json: bool,
- // pub postprocessors: Vec<serde_json::Map<String, serde_json::Value>>,
-}
-
-#[derive(Serialize, Copy, Clone)]
-pub enum ExtractFlat {
- #[serde(rename(serialize = "in_playlist"))]
- InPlaylist,
-
- #[serde(rename(serialize = "discard_in_playlist"))]
- DiscardInPlaylist,
-}
-
-#[derive(Serialize, Clone)]
-pub struct ExtractorArgs {
- pub youtube: YoutubeExtractorArgs,
-}
-
-#[derive(Serialize, Clone)]
-pub struct YoutubeExtractorArgs {
- comment_sort: Vec<String>,
- max_comments: Vec<String>,
-}
-
-impl YtDlpOptions {
- pub fn to_py_dict(self, py: Python) -> PyResult<Bound<PyDict>> {
- let string = serde_json::to_string(&self).expect("This should always work");
-
- let output: Bound<PyDict> = json_loads(py, string)?;
- Ok(output)
- }
-}