Files
OqeeRewind/utils/utilities.py
2025-12-20 12:52:35 +01:00

70 lines
2.3 KiB
Python

"""Utility functions for OqeeRewind, including verification, merging, and decryption."""
import os
import sys
import subprocess
import shutil
from utils.logging_config import logger
def verify_mp4ff():
"""Verify if mp4ff-decrypt is installed and available in PATH."""
if shutil.which("mp4ff-decrypt") is None:
logger.error("mp4ff-decrypt is not installed or not in PATH.")
logger.info("Please install it using:")
logger.info("go install github.com/Eyevinn/mp4ff/cmd/mp4ff-decrypt@latest")
sys.exit(1)
return True
def verify_cmd(path: str) -> bool:
"""Verify if the file provided at path is valid and exists, otherwise log error and exit."""
if not os.path.exists(path):
logger.error("File does not exist: %s", path)
sys.exit(1)
if not os.path.isfile(path):
logger.error("Path is not a file: %s", path)
sys.exit(1)
return True
def merge_segments(input_folder: str, track_id: str, output_file: str):
"""Merge downloaded segments into a single file using ffmpeg."""
segment_folder = os.path.join(input_folder, f"segments_{track_id}")
segment_files = sorted(
[f for f in os.listdir(segment_folder) if f.endswith(".m4s")],
key=lambda x: int(x.split(".")[0]),
)
with open(output_file, "wb") as outfile:
for fname in segment_files:
with open(f"{segment_folder}/{fname}", "rb") as infile:
outfile.write(infile.read())
logger.info("Merged segments into %s", output_file)
def decrypt(input_file, init_path, output_file, key):
"""Decrypt a media file using mp4ff-decrypt.
Args:
input_file: Path to the input encrypted file.
init_path: Path to the initialization file.
output_file: Path to the output decrypted file.
key: The decryption key in KID:KEY format.
Returns:
True if decryption succeeded, False otherwise.
"""
key = key.split(":")[1]
result = subprocess.run(
["mp4ff-decrypt", "-init", init_path, "-key", key, input_file, output_file],
capture_output=True,
text=True,
check=False,
)
if result.returncode == 0:
logger.info("Decrypted %s to %s", input_file, output_file)
return True
logger.error("Decryption failed: %s", result.stderr)
return False