mirror of https://github.com/Nezreka/SoulSync.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
135 lines
5.2 KiB
135 lines
5.2 KiB
"""Import post-processing guards and quarantine helpers."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import json
|
|
import os
|
|
from datetime import datetime
|
|
from pathlib import Path
|
|
from typing import Any, Dict, Optional
|
|
|
|
from config.settings import config_manager
|
|
from core.imports.context import (
|
|
get_import_clean_artist,
|
|
get_import_clean_title,
|
|
get_import_context_artist,
|
|
get_import_original_search,
|
|
get_import_track_info,
|
|
normalize_import_context,
|
|
)
|
|
from core.imports.file_ops import safe_move_file
|
|
from database.music_database import MusicDatabase
|
|
from utils.logging_config import get_logger
|
|
|
|
|
|
logger = get_logger("imports.guards")
|
|
|
|
|
|
def _get_config_manager():
|
|
return config_manager
|
|
|
|
|
|
def move_to_quarantine(file_path: str, context: dict, reason: str, automation_engine=None, *, trigger: str = "unknown") -> str:
|
|
"""Move a file to the quarantine folder and write a metadata sidecar.
|
|
|
|
`trigger` identifies which check fired (`integrity` / `acoustid` /
|
|
`bit_depth` / `unknown`) and is persisted in the sidecar so
|
|
one-click Approve can set the matching `_skip_quarantine_check`
|
|
bypass when re-running the pipeline.
|
|
|
|
Sidecar also persists a JSON-safe snapshot of the full `context`
|
|
dict via `serialize_quarantine_context`, enabling in-place approve
|
|
without losing the matched-track metadata. Legacy sidecars (written
|
|
before this expansion) lack the `context` field — Approve falls
|
|
back to `recover_to_staging` for those.
|
|
"""
|
|
from core.imports.quarantine import serialize_quarantine_context
|
|
|
|
download_dir = _get_config_manager().get("soulseek.download_path", "./downloads")
|
|
quarantine_dir = Path(download_dir) / "ss_quarantine"
|
|
quarantine_dir.mkdir(parents=True, exist_ok=True)
|
|
|
|
timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
|
|
original_name = Path(file_path).stem
|
|
file_ext = Path(file_path).suffix
|
|
|
|
quarantine_filename = f"{timestamp}_{original_name}{file_ext}.quarantined"
|
|
quarantine_path = quarantine_dir / quarantine_filename
|
|
|
|
safe_move_file(file_path, str(quarantine_path))
|
|
|
|
metadata_path = quarantine_dir / f"{timestamp}_{original_name}.json"
|
|
context = normalize_import_context(context)
|
|
original_search = get_import_original_search(context)
|
|
artist_context = get_import_context_artist(context)
|
|
|
|
metadata = {
|
|
"original_filename": Path(file_path).name,
|
|
"quarantine_reason": reason,
|
|
"timestamp": datetime.now().isoformat(),
|
|
"expected_track": get_import_clean_title(context, default=original_search.get("title", "Unknown")),
|
|
"expected_artist": get_import_clean_artist(context, default=(artist_context.get("name", "") if isinstance(artist_context, dict) else "Unknown")),
|
|
"context_key": context.get("context_key", "unknown"),
|
|
"trigger": trigger,
|
|
"context": serialize_quarantine_context(context),
|
|
}
|
|
|
|
try:
|
|
with open(metadata_path, "w", encoding="utf-8") as f:
|
|
json.dump(metadata, f, indent=2, ensure_ascii=False)
|
|
except Exception as exc:
|
|
logger.warning("Failed to write quarantine metadata: %s", exc)
|
|
|
|
logger.warning("File quarantined: %s - Reason: %s", quarantine_path, reason)
|
|
|
|
if automation_engine:
|
|
try:
|
|
ti = context.get("track_info", {})
|
|
artists = ti.get("artists", [])
|
|
artist_name = ""
|
|
if artists:
|
|
first = artists[0]
|
|
artist_name = first.get("name", str(first)) if isinstance(first, dict) else str(first)
|
|
automation_engine.emit(
|
|
"download_quarantined",
|
|
{
|
|
"artist": artist_name,
|
|
"title": ti.get("name", ""),
|
|
"reason": reason or "Unknown",
|
|
},
|
|
)
|
|
except Exception as e:
|
|
logger.debug("emit download_quarantined failed: %s", e)
|
|
|
|
return str(quarantine_path)
|
|
|
|
|
|
def check_flac_bit_depth(file_path: str, context: dict) -> Optional[str]:
|
|
"""Return a rejection message if a FLAC file violates the configured bit depth."""
|
|
if not context.get("_audio_quality", "").startswith("FLAC"):
|
|
return None
|
|
|
|
quality_profile = MusicDatabase().get_quality_profile()
|
|
flac_config = quality_profile.get("qualities", {}).get("flac", {})
|
|
flac_pref = flac_config.get("bit_depth", "any")
|
|
if flac_pref == "any":
|
|
return None
|
|
|
|
actual_bits = context["_audio_quality"].replace("FLAC ", "").replace("bit", "")
|
|
if actual_bits == flac_pref:
|
|
return None
|
|
|
|
flac_fallback = flac_config.get("bit_depth_fallback", True)
|
|
downsample_enabled = _get_config_manager().get("lossy_copy.downsample_hires", False)
|
|
track_info = context.get("track_info", {})
|
|
track_name = track_info.get("name", os.path.basename(file_path))
|
|
|
|
if flac_fallback or downsample_enabled:
|
|
if downsample_enabled:
|
|
logger.info("[FLAC Downsample] Accepted %s-bit FLAC (will be downsampled to %s-bit): %s", actual_bits, flac_pref, track_name)
|
|
else:
|
|
logger.warning("[FLAC Fallback] Accepted %s-bit FLAC (preferred %s-bit): %s", actual_bits, flac_pref, track_name)
|
|
return None
|
|
|
|
return f"FLAC bit depth mismatch: file is {actual_bits}-bit, preference is {flac_pref}-bit"
|