python-plexapi/tools/plex-dummyfiles.py
2024-06-22 11:29:14 -07:00

344 lines
11 KiB
Python

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Plex-DummyFiles creates dummy files for testing with the proper
Plex folder and file naming structure.
"""
import argparse
import os
import re
import shutil
from pathlib import Path
from typing import Any, List, Optional, Tuple, Union
BASE_DIR_PATH = Path(__file__).parents[1].absolute()
STUB_VIDEO_PATH = BASE_DIR_PATH / "tests" / "data" / "video_stub.mp4"
class DummyFiles:
def __init__(self, **kwargs: Any):
self.dummy_file: Path = kwargs['file']
self.root_folder: Path = kwargs['root']
self.title: str = kwargs['title']
self.year: int = kwargs['year']
self.tmdb: Optional[int] = kwargs['tmdb']
self.tvdb: Optional[int] = kwargs['tvdb']
self.imdb: Optional[str] = kwargs['imdb']
self.dry_run: bool = kwargs['dry_run']
self.clean: bool = kwargs['clean']
@property
def external_id(self) -> Optional[str]:
"""Return the external ID of the media."""
if self.tmdb:
return f"tmdb-{self.tmdb}"
if self.tvdb:
return f"tvdb-{self.tvdb}"
if self.imdb:
return f"imdb-{self.imdb}"
return None
def create_folder(self, folder: Path, parent: Optional[Path] = None, level: int = 0) -> None:
"""Create a folder with the path."""
print(f"{'' * level}├─ {folder}{os.sep}")
if parent:
folder = parent / folder
folder = self.root_folder / folder
if not self.dry_run:
if self.clean and folder.exists():
shutil.rmtree(folder)
# No check for illegal characters in folder name
folder.mkdir(parents=True, exist_ok=True)
def create_files(self, files: List[Path], parent: Optional[Path] = None, level: int = 1) -> None:
"""Create a list of files with the given paths."""
for file in files:
print(f"{'' * level}├─ {file}")
if parent:
file = parent / file
file = self.root_folder / file
if not self.dry_run:
# No check for illegal characters in file name
# Will overwrite files if they exist
shutil.copy(self.dummy_file, file)
class DummyMovie(DummyFiles):
def __init__(self, **kwargs: Any):
super().__init__(**kwargs)
versions = kwargs['versions'] or [["", 1]]
self.edition: Optional[str] = kwargs['edition']
self.versions: List[str] = [v[0] for v in versions]
self.parts: List[int] = [v[1] for v in versions]
self.movie_folder: Path = self.create_movie_folder()
self.create_movie_files()
def create_movie_folder(self) -> Path:
"""Create the movie folder with the proper naming structure."""
folder = f"{self.title} ({self.year})"
if self.edition:
folder = f"{folder} {{edition-{self.edition}}}"
if self.external_id:
folder = f"{folder} {{{self.external_id}}}"
movie_folder = Path(folder)
self.create_folder(movie_folder)
return movie_folder
def create_movie_files(self) -> None:
"""Create the list of movie files with the proper naming structure."""
title = f"{self.title} ({self.year})"
_movie_parts: List[List[str]] = []
movie_files: List[Path] = []
for version in self.versions:
if version:
_movie_parts.append([title, f"- {version}"])
else:
_movie_parts.append([title])
if self.edition:
for _movie_part in _movie_parts:
_movie_part.append(f"{{edition-{self.edition}}}")
if self.external_id:
for _movie_part in _movie_parts:
_movie_part.append(f"{{{self.external_id}}}")
for _movie_part, parts in zip(_movie_parts, self.parts):
if parts > 1:
for part in range(1, parts + 1):
_movie_file = f"{' '.join(_movie_part)} - pt{part}{self.dummy_file.suffix}"
movie_files.append(Path(_movie_file))
else:
_movie_file = f"{' '.join(_movie_part)}{self.dummy_file.suffix}"
movie_files.append(Path(_movie_file))
self.create_files(movie_files, parent=self.movie_folder)
class DummyShow(DummyFiles):
def __init__(self, **kwargs: Any):
super().__init__(**kwargs)
self.seasons: List[List[int]] = kwargs['seasons']
self.episodes: List[List[Union[int, List[int], Tuple[int, int]]]] = kwargs['episodes']
self.show_folder: Path = self.create_show_folder()
self.create_episode_files()
def create_show_folder(self) -> Path:
"""Create the show folder with the proper naming structure."""
folder = f"{self.title} ({self.year})"
if self.external_id:
folder = f"{folder} {{{self.external_id}}}"
show_folder = Path(folder)
self.create_folder(show_folder)
return show_folder
def create_episode_files(self) -> None:
"""Create the list of season folders and episode files with the proper naming structure."""
for seasons, episodes in zip(self.seasons, self.episodes):
for season in seasons:
season_folder = Path(f"Season {season:02}")
self.create_folder(season_folder, parent=self.show_folder, level=1)
episode_files: List[Path] = []
for episode in episodes:
if isinstance(episode, tuple):
_episode_file = (
f"{self.title} ({self.year})"
f" - S{season:02}E{episode[0]:02}-E{episode[1]:02}{self.dummy_file.suffix}"
)
episode_files.append(Path(_episode_file))
elif isinstance(episode, list) and episode[1] > 1:
for part in range(1, episode[1] + 1):
_episode_file = (
f"{self.title} ({self.year})"
f" - S{season:02}E{episode[0]:02} - pt{part}{self.dummy_file.suffix}"
)
episode_files.append(Path(_episode_file))
else:
_episode_file = f"{self.title} ({self.year}) - S{season:02}E{episode:02}{self.dummy_file.suffix}"
episode_files.append(Path(_episode_file))
self.create_files(episode_files, parent=self.show_folder / season_folder, level=2)
def validate_folder_path(folder: str) -> Path:
folder_path = Path(folder)
if not folder_path.exists():
raise argparse.ArgumentTypeError(f"Folder does not exist: {folder_path}")
if not folder_path.is_dir():
raise argparse.ArgumentTypeError(f"Path is not a folder: {folder_path}")
return folder_path
def validate_file_path(file: str) -> Path:
file_path = Path(file)
if not file_path.exists():
raise argparse.ArgumentTypeError(f"File does not exist: {file_path}")
if not file_path.is_file():
raise argparse.ArgumentTypeError(f"Path is not a file: {file_path}")
return file_path
def validate_imdb_id(imdb_id: str) -> str:
if re.match(r"tt\d{7,8}", imdb_id):
return imdb_id
raise argparse.ArgumentTypeError(f"Invalid IMDB ID: {imdb_id}")
def validate_versions(
version_str: str,
sep_parts: str = "|",
) -> List[Union[str, int]]:
version_parts = version_str.split(sep_parts)
if len(version_parts) == 1:
return [version_parts[0], 1]
return [version_parts[0], int(version_parts[1])]
def validate_number_ranges(
num_str: str,
sep: str = ",",
sep_range: str = "-",
sep_stack: str = "+",
sep_parts: str = "|",
) -> List[Union[int, List[int], Tuple[int, int]]]:
parsed: List[Union[int, List[int], Tuple[int, int]]] = []
for part in num_str.split(sep):
if sep_range in part:
r1, r2 = [int(i) for i in part.split(sep_range)]
parsed.extend(list(range(r1, r2 + 1)))
elif sep_stack in part:
s1, s2 = [int(i) for i in part.split(sep_stack)]
parsed.append((s1, s2))
elif sep_parts in part:
ep, pt = [int(i) for i in part.split(sep_parts)]
parsed.append([ep, pt])
else:
parsed.append(int(part))
return parsed
if __name__ == "__main__": # noqa: C901
parser = argparse.ArgumentParser(description=__doc__)
parser.add_argument(
"media_type",
help="Type of media to create",
choices=["movie", "show"],
)
parser.add_argument(
"-r",
"--root",
help="Root media folder to create the dummy folders and files",
type=validate_folder_path,
required=True
)
parser.add_argument(
"-t",
"--title",
help="Title of the media",
required=True,
)
parser.add_argument(
"-y",
"--year",
help="Year of the media",
type=int,
required=True,
)
movie_group = parser.add_argument_group("Movie Options")
movie_group.add_argument(
"-ed",
"--edition",
help="Edition title"
)
movie_group.add_argument(
"-vs",
"--versions",
help="Versions and parts to create (| for parts)",
action="append",
type=validate_versions,
)
show_group = parser.add_argument_group("TV Show Options")
show_group.add_argument(
"-sn",
"--seasons",
help="Seasons to create (- for range)",
action="append",
type=validate_number_ranges,
)
show_group.add_argument(
"-ep",
"--episodes",
help="Episodes to create (- for range, + for stacked, | for parts)",
action="append",
type=validate_number_ranges,
)
id_group = parser.add_mutually_exclusive_group()
id_group.add_argument(
"--tmdb",
help="TMDB ID of the media",
type=int,
)
id_group.add_argument(
"--tvdb",
help="TVDB ID of the media",
type=int,
)
id_group.add_argument(
"--imdb",
help="IMDB ID of the media",
type=validate_imdb_id,
)
parser.add_argument(
"-f",
"--file",
help="Path to the dummy video file",
type=validate_file_path,
default=STUB_VIDEO_PATH,
)
parser.add_argument(
"--dry-run",
help="Print the folder and file structure without creating the files",
action="store_true",
)
parser.add_argument(
"--clean",
help="Remove the old files before creating new dummy files",
action="store_true",
)
opts, _ = parser.parse_known_args()
if opts.dry_run:
print("Dry Run: No files will be created")
print(f"{opts.root}{os.sep}")
if opts.media_type == "movie":
DummyMovie(**vars(opts))
elif opts.media_type == "show":
if not opts.seasons or not opts.episodes:
parser.error("Both --seasons and --episodes are required for TV shows")
if len(opts.seasons) != len(opts.episodes):
parser.error("Number of seasons and episodes arguments must match")
if any(not isinstance(season, int) for season_groups in opts.seasons for season in season_groups):
parser.error("Seasons must be a list of integers or integer ranges")
DummyShow(**vars(opts))