From 4fc2454598e903781c18f734e04ee227986c8456 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=A9rome=20Eertmans?= Date: Wed, 22 May 2024 14:00:02 +0200 Subject: [PATCH 1/3] feat(lib): smarter files reversing Implement a smarter generation of reversed files by splitting the video into smaller segments. Closes #434 --- manim_slides/slide/base.py | 11 ++++-- manim_slides/utils.py | 68 ++++++++++++++++++++++++++++++++++++-- 2 files changed, 75 insertions(+), 4 deletions(-) diff --git a/manim_slides/slide/base.py b/manim_slides/slide/base.py index 4c35b0d4..f059984d 100644 --- a/manim_slides/slide/base.py +++ b/manim_slides/slide/base.py @@ -475,10 +475,11 @@ def _save_slides(self, use_cache: bool = True) -> None: for pre_slide_config in tqdm( self._slides, - desc=f"Concatenating animation files to '{scene_files_folder}' and generating reversed animations", + desc=f"Concatenating animations to '{scene_files_folder}' and generating reversed animations", leave=self._leave_progress_bar, ascii=True if platform.system() == "Windows" else None, disable=not self._show_progress_bar, + unit=" slides", ): slide_files = files[pre_slide_config.slides_slice] @@ -492,7 +493,13 @@ def _save_slides(self, use_cache: bool = True) -> None: # We only reverse video if it was not present if not use_cache or not rev_file.exists(): - reverse_video_file(dst_file, rev_file) + reverse_video_file( + dst_file, + rev_file, + leave=self._leave_progress_bar, + ascii=True if platform.system() == "Windows" else None, + disable=not self._show_progress_bar, + ) slides.append( SlideConfig.from_pre_slide_config_and_files( diff --git a/manim_slides/utils.py b/manim_slides/utils.py index 53ac5c8e..89fb095b 100644 --- a/manim_slides/utils.py +++ b/manim_slides/utils.py @@ -2,9 +2,12 @@ import os import tempfile from collections.abc import Iterator +from multiprocessing import Pool from pathlib import Path +from typing import Any, Optional import av +from tqdm import tqdm from .logger import logger @@ -89,8 +92,9 @@ def link_nodes(*nodes: av.filter.context.FilterContext) -> None: c.link_to(n) -def reverse_video_file(src: Path, dest: Path) -> None: +def reverse_video_file_in_one_chunk(src_and_dest: tuple[Path, Path]) -> None: """Reverses a video file, writing the result to `dest`.""" + src, dest = src_and_dest with ( av.open(str(src)) as input_container, av.open(str(dest), mode="w") as output_container, @@ -120,8 +124,68 @@ def reverse_video_file(src: Path, dest: Path) -> None: for _ in range(frames_count): frame = graph.pull() - frame.pict_type = 5 # Otherwise we get a warning saying it is changed + frame.pict_type = "NONE" # Otherwise we get a warning saying it is changed output_container.mux(output_stream.encode(frame)) for packet in output_stream.encode(): output_container.mux(packet) + + +def reverse_video_file( + src: Path, + dest: Path, + max_segment_duration: float = 1, + processes: Optional[int] = None, + **tqdm_kwargs: Any, +) -> None: + """Reverses a video file, writing the result to `dest`.""" + with av.open(str(src)) as input_container: # Fast path if file is short enough + input_stream = input_container.streams.video[0] + if input_stream.duration: + if ( + float(input_stream.duration * input_stream.time_base) + <= max_segment_duration + ): + return reverse_video_file_in_one_chunk((src, dest)) + else: + logger.debug( + f"Could not determine duration of {src}, falling back to segmentation." + ) + + with tempfile.TemporaryDirectory() as tmpdirname: + tmpdir = Path(tmpdirname) + with av.open( + str(tmpdir / "%04d.mp4"), + "w", + format="segment", + options={"segment_time": str(max_segment_duration)}, + ) as output_container: + output_stream = output_container.add_stream( + template=input_stream, + ) + + for packet in input_container.demux(input_stream): + if packet.dts is None: + continue + + packet.stream = output_stream + output_container.mux(packet) + + src_files = list(tmpdir.iterdir()) + rev_files = [ + src_file.with_stem("rev_" + src_file.stem) for src_file in src_files + ] + + with Pool(processes, maxtasksperchild=1) as pool: + for _ in tqdm( + pool.imap_unordered( + reverse_video_file_in_one_chunk, zip(src_files, rev_files) + ), + desc="Reversing large file by cutting it in segments", + total=len(src_files), + unit=" files", + **tqdm_kwargs, + ): + pass # We just consume the iterator + + concatenate_video_files(rev_files[::-1], dest) From 971e5479f11bfb96c9fbdf5acf8e2d970b96779e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=A9rome=20Eertmans?= Date: Wed, 22 May 2024 21:30:54 +0200 Subject: [PATCH 2/3] chore(lib): change default length --- manim_slides/utils.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/manim_slides/utils.py b/manim_slides/utils.py index 89fb095b..4c41683d 100644 --- a/manim_slides/utils.py +++ b/manim_slides/utils.py @@ -134,7 +134,7 @@ def reverse_video_file_in_one_chunk(src_and_dest: tuple[Path, Path]) -> None: def reverse_video_file( src: Path, dest: Path, - max_segment_duration: float = 1, + max_segment_duration: float = 4, processes: Optional[int] = None, **tqdm_kwargs: Any, ) -> None: From c57ce6c812a73a6723a126d3bcc2dab3803fc496 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=A9rome=20Eertmans?= Date: Thu, 5 Sep 2024 15:02:10 +0200 Subject: [PATCH 3/3] chore: use suffix --- manim_slides/utils.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/manim_slides/utils.py b/manim_slides/utils.py index 28bef784..ff6df88f 100644 --- a/manim_slides/utils.py +++ b/manim_slides/utils.py @@ -155,7 +155,7 @@ def reverse_video_file( with tempfile.TemporaryDirectory() as tmpdirname: tmpdir = Path(tmpdirname) with av.open( - str(tmpdir / "%04d.mp4"), + str(tmpdir / f"%04d.{src.suffix}"), "w", format="segment", options={"segment_time": str(max_segment_duration)},