aboutsummaryrefslogtreecommitdiff
path: root/transcoder/src/fragment.rs
blob: 3cb4c40830222afeb48d6a20d8c7b8a2a3638a11 (plain)
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
/*
    This file is part of jellything (https://codeberg.org/metamuffin/jellything)
    which is licensed under the GNU Affero General Public License (version 3); see /COPYING.
    Copyright (C) 2025 metamuffin <metamuffin.org>
*/

use crate::LOCAL_VIDEO_TRANSCODING_TASKS;
use jellybase::{
    cache::{async_cache_file, CachePath},
    common::stream::{StreamContainer, StreamFormatInfo},
};
use log::{debug, info};
use std::process::Stdio;
use tokio::{
    io::copy,
    process::{ChildStdin, Command},
};

// TODO odd video resolutions can cause errors when transcoding to YUV42{0,2}
// TODO with an implementation that cant handle it (SVT-AV1 is such an impl).

pub async fn transcode(
    key: &str,
    format: &StreamFormatInfo,
    container: StreamContainer,
    input: impl FnOnce(ChildStdin),
) -> anyhow::Result<CachePath> {
    async_cache_file(
        &["frag-tc", key, &format!("{format:?}")],
        move |mut output| async move {
            let _permit = LOCAL_VIDEO_TRANSCODING_TASKS.acquire().await?;
            debug!("transcoding fragment with {format:?}");

            let mut args = Vec::<String>::new();

            match format.codec.as_str() {
                "V_AVC" => {}

                _ => unreachable!(),
            }

            info!("encoding with {:?}", args.join(" "));

            let container = match container {
                StreamContainer::WebM => "webm",
                StreamContainer::Matroska => "matroska",
                StreamContainer::WebVTT => "vtt",
                StreamContainer::MPEG4 => "mp4",
                StreamContainer::JVTT => unreachable!(),
            };

            let mut proc = Command::new("ffmpeg")
                .stdin(Stdio::piped())
                .stdout(Stdio::piped())
                .args(["-f", "matroska", "-i", "pipe:0"])
                .args(args)
                .args(["-f", container, "pipe:1"])
                .spawn()?;

            let stdin = proc.stdin.take().unwrap();
            let mut stdout = proc.stdout.take().unwrap();

            input(stdin);
            copy(&mut stdout, &mut output).await?;

            proc.wait().await.unwrap().exit_ok()?;
            info!("done");
            Ok(())
        },
    )
    .await
}