mirror of
https://github.com/unshackle-dl/unshackle.git
synced 2025-10-23 15:11:08 +00:00
feat: add --no-mux flag to skip muxing tracks into container files
Add --no-mux command-line option to allow downloading individual track
files without muxing them into a container file (.mkv/.mka/.mks).
This addresses use cases where users want to download tracks separately,
such as:
- Downloading only subtitles as individual .srt/.vtt files
- Keeping audio/video/subtitle tracks as separate files
- Converting subtitle formats without creating container files
When --no-mux is used:
- Tracks are saved as individual files with descriptive suffixes
- Video tracks: filename.{codec}.ext
- Audio tracks: filename.{language}.{codec}.ext
- Subtitle tracks: filename.{language}.forced.sdh.ext (as applicable)
- Folder structure respects --no-folder flag
Resolves #21
This commit is contained in:
@@ -258,6 +258,7 @@ class dl:
|
|||||||
@click.option(
|
@click.option(
|
||||||
"--no-source", is_flag=True, default=False, help="Disable the source tag from the output file name and path."
|
"--no-source", is_flag=True, default=False, help="Disable the source tag from the output file name and path."
|
||||||
)
|
)
|
||||||
|
@click.option("--no-mux", is_flag=True, default=False, help="Do not mux tracks into a container file.")
|
||||||
@click.option(
|
@click.option(
|
||||||
"--workers",
|
"--workers",
|
||||||
type=int,
|
type=int,
|
||||||
@@ -484,6 +485,7 @@ class dl:
|
|||||||
no_proxy: bool,
|
no_proxy: bool,
|
||||||
no_folder: bool,
|
no_folder: bool,
|
||||||
no_source: bool,
|
no_source: bool,
|
||||||
|
no_mux: bool,
|
||||||
workers: Optional[int],
|
workers: Optional[int],
|
||||||
downloads: int,
|
downloads: int,
|
||||||
best_available: bool,
|
best_available: bool,
|
||||||
@@ -1139,7 +1141,12 @@ class dl:
|
|||||||
|
|
||||||
muxed_paths = []
|
muxed_paths = []
|
||||||
|
|
||||||
if isinstance(title, (Movie, Episode)):
|
if no_mux:
|
||||||
|
# Skip muxing, handle individual track files
|
||||||
|
for track in title.tracks:
|
||||||
|
if track.path and track.path.exists():
|
||||||
|
muxed_paths.append(track.path)
|
||||||
|
elif isinstance(title, (Movie, Episode)):
|
||||||
progress = Progress(
|
progress = Progress(
|
||||||
TextColumn("[progress.description]{task.description}"),
|
TextColumn("[progress.description]{task.description}"),
|
||||||
SpinnerColumn(finished_text=""),
|
SpinnerColumn(finished_text=""),
|
||||||
@@ -1258,6 +1265,52 @@ class dl:
|
|||||||
# dont mux
|
# dont mux
|
||||||
muxed_paths.append(title.tracks.audio[0].path)
|
muxed_paths.append(title.tracks.audio[0].path)
|
||||||
|
|
||||||
|
if no_mux:
|
||||||
|
# Handle individual track files without muxing
|
||||||
|
final_dir = config.directories.downloads
|
||||||
|
if not no_folder and isinstance(title, (Episode, Song)):
|
||||||
|
# Create folder based on title
|
||||||
|
# Use first available track for filename generation
|
||||||
|
sample_track = title.tracks.videos[0] if title.tracks.videos else (
|
||||||
|
title.tracks.audio[0] if title.tracks.audio else (
|
||||||
|
title.tracks.subtitles[0] if title.tracks.subtitles else None
|
||||||
|
)
|
||||||
|
)
|
||||||
|
if sample_track and sample_track.path:
|
||||||
|
media_info = MediaInfo.parse(sample_track.path)
|
||||||
|
final_dir /= title.get_filename(media_info, show_service=not no_source, folder=True)
|
||||||
|
|
||||||
|
final_dir.mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
|
for track_path in muxed_paths:
|
||||||
|
# Generate appropriate filename for each track
|
||||||
|
media_info = MediaInfo.parse(track_path)
|
||||||
|
base_filename = title.get_filename(media_info, show_service=not no_source)
|
||||||
|
|
||||||
|
# Add track type suffix to filename
|
||||||
|
track = next((t for t in title.tracks if t.path == track_path), None)
|
||||||
|
if track:
|
||||||
|
if isinstance(track, Video):
|
||||||
|
track_suffix = f".{track.codec.name if hasattr(track.codec, 'name') else 'video'}"
|
||||||
|
elif isinstance(track, Audio):
|
||||||
|
lang_suffix = f".{track.language}" if track.language else ""
|
||||||
|
track_suffix = f"{lang_suffix}.{track.codec.name if hasattr(track.codec, 'name') else 'audio'}"
|
||||||
|
elif isinstance(track, Subtitle):
|
||||||
|
lang_suffix = f".{track.language}" if track.language else ""
|
||||||
|
forced_suffix = ".forced" if track.forced else ""
|
||||||
|
sdh_suffix = ".sdh" if track.sdh else ""
|
||||||
|
track_suffix = f"{lang_suffix}{forced_suffix}{sdh_suffix}"
|
||||||
|
else:
|
||||||
|
track_suffix = ""
|
||||||
|
|
||||||
|
final_path = final_dir / f"{base_filename}{track_suffix}{track_path.suffix}"
|
||||||
|
else:
|
||||||
|
final_path = final_dir / f"{base_filename}{track_path.suffix}"
|
||||||
|
|
||||||
|
shutil.move(track_path, final_path)
|
||||||
|
self.log.debug(f"Saved: {final_path.name}")
|
||||||
|
else:
|
||||||
|
# Handle muxed files
|
||||||
for muxed_path in muxed_paths:
|
for muxed_path in muxed_paths:
|
||||||
media_info = MediaInfo.parse(muxed_path)
|
media_info = MediaInfo.parse(muxed_path)
|
||||||
final_dir = config.directories.downloads
|
final_dir = config.directories.downloads
|
||||||
|
|||||||
Reference in New Issue
Block a user