73d7da440f
Two related gaps surfaced from "uploaded a track, nothing changed / no status": - A track could stay stuck on `pending` forever (an unexpected worker error rolled back the run without recording anything), and `failed` carried no reason. Add `tracks.metadata_error` + `tracks.enriched_at` (migration), stamp the outcome in apply_enrichment, add TrackRepository.mark_enrichment_failed, wrap enrich_task to persist crashes as `failed` in a fresh session, and emit a human-readable no-match reason. Expose metadata_error/enriched_at in TrackOut. - The tag-first merge let junk embedded tags (e.g. "Music Track"/"Sound_13958") override even a 0.99-confidence AcoustID match. Add acoustid_trust_score (default 0.85): above it the acoustic identity wins for title/artist/album/ year, tags are fallback; below it, tag-first as before. Add a license-free real-file fixture (Scarlet Fire / Otis McDonald) whose junk tags AcoustID overrides, with an always-on tag-reader test plus fpcalc/AcoustID/ network-gated identity + full-pipeline tests (skip on host, run in the container). Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
35 lines
709 B
Python
35 lines
709 B
Python
"""Track and Artist domain entities."""
|
|
|
|
import datetime as dt
|
|
import uuid
|
|
from dataclasses import dataclass
|
|
|
|
|
|
@dataclass(frozen=True, slots=True)
|
|
class Artist:
|
|
id: uuid.UUID
|
|
name: str
|
|
created_at: dt.datetime
|
|
updated_at: dt.datetime
|
|
|
|
|
|
@dataclass(frozen=True, slots=True)
|
|
class Track:
|
|
id: uuid.UUID
|
|
title: str
|
|
artist_id: uuid.UUID
|
|
album_id: uuid.UUID | None
|
|
storage_uri: str
|
|
file_format: str
|
|
file_size: int
|
|
source: str
|
|
source_id: str
|
|
duration_seconds: int | None
|
|
genre: str | None
|
|
year: int | None
|
|
metadata_status: str
|
|
metadata_error: str | None
|
|
enriched_at: dt.datetime | None
|
|
created_at: dt.datetime
|
|
updated_at: dt.datetime
|