189 lines
5.3 KiB
Python
189 lines
5.3 KiB
Python
"""Filesystem application DTOs."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from dataclasses import dataclass
|
|
|
|
|
|
@dataclass
|
|
class CopyMediaResponse:
|
|
"""Response from copying a media file."""
|
|
|
|
status: str
|
|
source: str | None = None
|
|
destination: str | None = None
|
|
filename: str | None = None
|
|
size: int | None = None
|
|
error: str | None = None
|
|
message: str | None = None
|
|
|
|
def to_dict(self) -> dict:
|
|
if self.error:
|
|
return {"status": self.status, "error": self.error, "message": self.message}
|
|
return {
|
|
"status": self.status,
|
|
"source": self.source,
|
|
"destination": self.destination,
|
|
"filename": self.filename,
|
|
"size": self.size,
|
|
}
|
|
|
|
|
|
@dataclass
|
|
class MoveMediaResponse:
|
|
"""Response from moving a media file."""
|
|
|
|
status: str
|
|
source: str | None = None
|
|
destination: str | None = None
|
|
filename: str | None = None
|
|
size: int | None = None
|
|
error: str | None = None
|
|
message: str | None = None
|
|
|
|
def to_dict(self) -> dict:
|
|
if self.error:
|
|
return {"status": self.status, "error": self.error, "message": self.message}
|
|
return {
|
|
"status": self.status,
|
|
"source": self.source,
|
|
"destination": self.destination,
|
|
"filename": self.filename,
|
|
"size": self.size,
|
|
}
|
|
|
|
|
|
|
|
@dataclass
|
|
class PlacedSubtitle:
|
|
"""One subtitle file successfully placed."""
|
|
|
|
source: str
|
|
destination: str
|
|
filename: str
|
|
|
|
def to_dict(self) -> dict:
|
|
return {
|
|
"source": self.source,
|
|
"destination": self.destination,
|
|
"filename": self.filename,
|
|
}
|
|
|
|
|
|
@dataclass
|
|
class UnresolvedTrack:
|
|
"""A subtitle track that needs agent clarification before placement."""
|
|
|
|
raw_tokens: list[str]
|
|
file_path: str | None = None
|
|
file_size_kb: float | None = None
|
|
reason: str = "" # "unknown_language" | "low_confidence"
|
|
|
|
def to_dict(self) -> dict:
|
|
return {
|
|
"raw_tokens": self.raw_tokens,
|
|
"file_path": self.file_path,
|
|
"file_size_kb": self.file_size_kb,
|
|
"reason": self.reason,
|
|
}
|
|
|
|
|
|
@dataclass
|
|
class AvailableSubtitle:
|
|
"""One subtitle track available on an embedded media item."""
|
|
|
|
language: str # ISO 639-2 code
|
|
subtitle_type: str # "standard" | "sdh" | "forced" | "unknown"
|
|
|
|
def to_dict(self) -> dict:
|
|
return {"language": self.language, "type": self.subtitle_type}
|
|
|
|
|
|
@dataclass
|
|
class ManageSubtitlesResponse:
|
|
"""Response from the manage_subtitles use case."""
|
|
|
|
status: str # "ok" | "needs_clarification" | "error"
|
|
video_path: str | None = None
|
|
placed: list[PlacedSubtitle] | None = None
|
|
skipped_count: int = 0
|
|
unresolved: list[UnresolvedTrack] | None = None
|
|
available: list[AvailableSubtitle] | None = None # embedded tracks summary
|
|
error: str | None = None
|
|
message: str | None = None
|
|
|
|
def to_dict(self) -> dict:
|
|
if self.error:
|
|
return {"status": self.status, "error": self.error, "message": self.message}
|
|
result = {
|
|
"status": self.status,
|
|
"video_path": self.video_path,
|
|
"placed": [p.to_dict() for p in (self.placed or [])],
|
|
"placed_count": len(self.placed or []),
|
|
"skipped_count": self.skipped_count,
|
|
}
|
|
if self.unresolved:
|
|
result["unresolved"] = [u.to_dict() for u in self.unresolved]
|
|
result["unresolved_count"] = len(self.unresolved)
|
|
if self.available:
|
|
result["available"] = [a.to_dict() for a in self.available]
|
|
return result
|
|
|
|
|
|
@dataclass
|
|
class CreateSeedLinksResponse:
|
|
"""Response from creating seed links for a torrent."""
|
|
|
|
status: str
|
|
torrent_subfolder: str | None = None
|
|
linked_file: str | None = None
|
|
copied_files: list[str] | None = None
|
|
copied_count: int = 0
|
|
skipped: list[str] | None = None
|
|
error: str | None = None
|
|
message: str | None = None
|
|
|
|
def to_dict(self) -> dict:
|
|
if self.error:
|
|
return {"status": self.status, "error": self.error, "message": self.message}
|
|
return {
|
|
"status": self.status,
|
|
"torrent_subfolder": self.torrent_subfolder,
|
|
"linked_file": self.linked_file,
|
|
"copied_files": self.copied_files or [],
|
|
"copied_count": self.copied_count,
|
|
"skipped": self.skipped or [],
|
|
}
|
|
|
|
|
|
@dataclass
|
|
class ListFolderResponse:
|
|
"""Response from listing a folder."""
|
|
|
|
status: str
|
|
folder_type: str | None = None # SHOULD BE A PROPERTY
|
|
path: str | None = None # NOT NONE - Should be path
|
|
entries: list[str] | None = None # NOT NONE - Empty list of path
|
|
count: int | None = None # USELESS
|
|
error: str | None = None
|
|
message: str | None = None
|
|
|
|
def to_dict(self):
|
|
"""Convert to dict for agent compatibility."""
|
|
result = {"status": self.status}
|
|
|
|
if self.error:
|
|
result["error"] = self.error
|
|
result["message"] = self.message
|
|
else:
|
|
if self.folder_type:
|
|
result["folder_type"] = self.folder_type
|
|
if self.path:
|
|
result["path"] = self.path
|
|
if self.entries is not None:
|
|
result["entries"] = self.entries
|
|
if self.count is not None:
|
|
result["count"] = self.count
|
|
|
|
return result
|