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.
SoulSync/tests/test_personalized_generator...

317 lines
12 KiB

"""Boundary tests for the curated / hybrid personalized generators
(`daily_mix`, `fresh_tape`, `archives`, `seasonal_mix`)."""
from __future__ import annotations
import sqlite3
from types import SimpleNamespace
from typing import Any, List
from unittest.mock import MagicMock
import pytest
from core.personalized.generators import archives as _arch_mod
from core.personalized.generators import daily_mix as _dm_mod
from core.personalized.generators import fresh_tape as _ft_mod
from core.personalized.generators import seasonal_mix as _sm_mod
from core.personalized.specs import get_registry
from core.personalized.types import PlaylistConfig
# ─── daily_mix ───────────────────────────────────────────────────────
class _DailyMixService:
"""Stub PersonalizedPlaylistsService for daily_mix tests."""
GENRE_MAPPING = {}
def __init__(self, top_genres=None, genre_tracks=None):
self._top = top_genres or []
self._tracks = genre_tracks or {}
self.calls: List[dict] = []
def get_top_genres_from_library(self, limit):
self.calls.append({'method': 'get_top_genres_from_library', 'limit': limit})
return self._top
def get_genre_playlist(self, genre, limit, **kw):
self.calls.append({'method': 'get_genre_playlist', 'genre': genre, 'limit': limit})
return self._tracks.get(genre, [])
class TestDailyMix:
def test_registered(self):
spec = get_registry().get('daily_mix')
assert spec is not None
assert spec.requires_variant is True
def test_variant_resolver_returns_ranks(self):
spec = get_registry().get('daily_mix')
ranks = spec.variant_resolver(SimpleNamespace(service=_DailyMixService()))
assert ranks == ['1', '2', '3', '4']
def test_resolves_rank_to_top_genre(self):
svc = _DailyMixService(
top_genres=[('Rock', 100), ('Pop', 80), ('Jazz', 30)],
genre_tracks={'Rock': [{'track_name': 'R', 'artist_name': 'A'}]},
)
out = _dm_mod.generate(SimpleNamespace(service=svc), '1', PlaylistConfig(limit=10))
assert len(out) == 1
assert out[0].track_name == 'R'
# Service called for top-genre lookup + genre playlist.
assert {c['method'] for c in svc.calls} == {
'get_top_genres_from_library', 'get_genre_playlist',
}
def test_rank_beyond_top_returns_empty(self):
svc = _DailyMixService(top_genres=[('Rock', 100)]) # only 1 top genre
out = _dm_mod.generate(SimpleNamespace(service=svc), '4', PlaylistConfig())
assert out == []
def test_invalid_variant_raises(self):
deps = SimpleNamespace(service=_DailyMixService())
with pytest.raises(ValueError, match='must be a rank int'):
_dm_mod.generate(deps, 'abc', PlaylistConfig())
# ─── fresh_tape / archives shared shape ─────────────────────────────
class _StubPoolTrack:
def __init__(self, sid, name='T', artist='A', source='spotify'):
self.spotify_track_id = sid
self.itunes_track_id = None
self.deezer_track_id = None
self.track_name = name
self.artist_name = artist
self.album_name = 'Album'
self.album_cover_url = None
self.duration_ms = 200000
self.popularity = 50
self.track_data_json = None
self.source = source
class _CuratedDB:
def __init__(self, curated_ids=None, pool_tracks=None):
self.curated_ids = curated_ids or []
self.pool_tracks = pool_tracks or []
self.requested_keys: List[str] = []
def get_curated_playlist(self, key, profile_id=1):
self.requested_keys.append(key)
return list(self.curated_ids)
def get_discovery_pool_tracks(self, **kwargs):
return list(self.pool_tracks)
def _curated_deps(db):
return SimpleNamespace(
database=db,
get_current_profile_id=lambda: 1,
get_active_discovery_source=lambda: 'spotify',
)
class TestFreshTape:
def test_registered(self):
spec = get_registry().get('fresh_tape')
assert spec is not None
assert spec.requires_variant is False
assert spec.display_name('') == 'Fresh Tape'
def test_returns_empty_when_no_curated_ids(self):
db = _CuratedDB(curated_ids=[])
out = _ft_mod.generate(_curated_deps(db), '', PlaylistConfig())
assert out == []
def test_hydrates_curated_ids_from_pool(self):
db = _CuratedDB(
curated_ids=['sp-1', 'sp-2', 'sp-missing'],
pool_tracks=[
_StubPoolTrack('sp-1', name='Song1', artist='Artist1'),
_StubPoolTrack('sp-2', name='Song2', artist='Artist2'),
],
)
out = _ft_mod.generate(_curated_deps(db), '', PlaylistConfig())
# Missing IDs silently skipped; order preserved.
assert [t.track_name for t in out] == ['Song1', 'Song2']
def test_tries_source_specific_then_fallback_key(self):
# First lookup (source-specific) returns []; second (generic) returns IDs.
class _DB:
def __init__(self):
self.calls = []
self.responses = {
'release_radar_spotify': [],
'release_radar': ['sp-1'],
}
def get_curated_playlist(self, key, profile_id=1):
self.calls.append(key)
return self.responses.get(key, [])
def get_discovery_pool_tracks(self, **kw):
return [_StubPoolTrack('sp-1', name='Hit')]
db = _DB()
out = _ft_mod.generate(_curated_deps(db), '', PlaylistConfig())
assert db.calls == ['release_radar_spotify', 'release_radar']
assert len(out) == 1
def test_respects_limit(self):
db = _CuratedDB(
curated_ids=[f'sp-{i}' for i in range(20)],
pool_tracks=[_StubPoolTrack(f'sp-{i}', name=f'T{i}') for i in range(20)],
)
out = _ft_mod.generate(_curated_deps(db), '', PlaylistConfig(limit=5))
assert len(out) == 5
def test_missing_database_dep_raises(self):
with pytest.raises(RuntimeError, match='missing `database`'):
_ft_mod.generate(SimpleNamespace(), '', PlaylistConfig())
class TestArchives:
def test_registered(self):
spec = get_registry().get('archives')
assert spec is not None
assert spec.display_name('') == 'The Archives'
def test_uses_discovery_weekly_curated_key(self):
db = _CuratedDB(
curated_ids=['sp-1'],
pool_tracks=[_StubPoolTrack('sp-1', name='Discover')],
)
_arch_mod.generate(_curated_deps(db), '', PlaylistConfig())
# Source-specific request fires first; fallback only fires
# when source-specific returns empty. Stub returns IDs on
# every call, so only the first key gets queried.
assert db.requested_keys[0] == 'discovery_weekly_spotify'
# ─── seasonal_mix ───────────────────────────────────────────────────
class _SeasonalService:
def __init__(self, track_ids):
self.track_ids = track_ids
def get_curated_seasonal_playlist(self, season_key, source=None):
return list(self.track_ids)
@pytest.fixture
def seasonal_db(tmp_path):
"""Real sqlite DB with seasonal_tracks rows for hydration."""
p = str(tmp_path / 'seasonal.db')
conn = sqlite3.connect(p)
conn.row_factory = sqlite3.Row
cursor = conn.cursor()
cursor.execute("""
CREATE TABLE seasonal_tracks (
id INTEGER PRIMARY KEY,
season_key TEXT, source TEXT,
spotify_track_id TEXT, track_name TEXT, artist_name TEXT,
album_name TEXT, album_cover_url TEXT,
duration_ms INTEGER, popularity INTEGER, track_data_json TEXT
)
""")
seed = [
('halloween', 'spotify', 'sp-1', 'Spooky', 'Ghost Band', 'Album1', None, 200000, 80, '{"id":"sp-1"}'),
('halloween', 'spotify', 'sp-2', 'Haunted', 'Ghost Band', 'Album2', None, 210000, 70, None),
('halloween', 'spotify', 'sp-extra', 'Extra', 'Other', 'Album3', None, 200000, 60, None),
]
cursor.executemany("""
INSERT INTO seasonal_tracks
(season_key, source, spotify_track_id, track_name, artist_name,
album_name, album_cover_url, duration_ms, popularity, track_data_json)
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
""", seed)
conn.commit()
conn.close()
class _DB:
def __init__(self, path): self.path = path
def _get_connection(self):
c = sqlite3.connect(self.path)
c.row_factory = sqlite3.Row
return c
return _DB(p)
class TestSeasonalMix:
def test_registered(self):
spec = get_registry().get('seasonal_mix')
assert spec is not None
assert spec.requires_variant is True
def test_variant_resolver_returns_seasons(self):
spec = get_registry().get('seasonal_mix')
seasons = spec.variant_resolver(None)
assert 'halloween' in seasons
assert 'christmas' in seasons
def test_no_variant_raises(self, seasonal_db):
deps = SimpleNamespace(
seasonal_service=_SeasonalService(['sp-1']),
database=seasonal_db,
get_active_discovery_source=lambda: 'spotify',
)
with pytest.raises(ValueError, match='requires a season variant'):
_sm_mod.generate(deps, '', PlaylistConfig())
def test_hydrates_curated_ids_in_order(self, seasonal_db):
deps = SimpleNamespace(
seasonal_service=_SeasonalService(['sp-2', 'sp-1']),
database=seasonal_db,
get_active_discovery_source=lambda: 'spotify',
)
out = _sm_mod.generate(deps, 'halloween', PlaylistConfig())
assert [t.track_name for t in out] == ['Haunted', 'Spooky']
def test_missing_track_id_silently_skipped(self, seasonal_db):
deps = SimpleNamespace(
seasonal_service=_SeasonalService(['sp-1', 'sp-not-in-db']),
database=seasonal_db,
get_active_discovery_source=lambda: 'spotify',
)
out = _sm_mod.generate(deps, 'halloween', PlaylistConfig())
assert len(out) == 1
assert out[0].track_name == 'Spooky'
def test_track_data_json_round_trips(self, seasonal_db):
deps = SimpleNamespace(
seasonal_service=_SeasonalService(['sp-1']),
database=seasonal_db,
get_active_discovery_source=lambda: 'spotify',
)
out = _sm_mod.generate(deps, 'halloween', PlaylistConfig())
# sp-1 had JSON; sp-2 had None.
assert out[0].track_data_json == {'id': 'sp-1'}
def test_empty_curated_returns_empty(self, seasonal_db):
deps = SimpleNamespace(
seasonal_service=_SeasonalService([]),
database=seasonal_db,
get_active_discovery_source=lambda: 'spotify',
)
out = _sm_mod.generate(deps, 'halloween', PlaylistConfig())
assert out == []
def test_respects_limit(self, seasonal_db):
deps = SimpleNamespace(
seasonal_service=_SeasonalService(['sp-1', 'sp-2', 'sp-extra']),
database=seasonal_db,
get_active_discovery_source=lambda: 'spotify',
)
out = _sm_mod.generate(deps, 'halloween', PlaylistConfig(limit=2))
assert len(out) == 2
def test_missing_seasonal_service_raises(self):
deps = SimpleNamespace(database=object())
with pytest.raises(RuntimeError, match='missing `seasonal_service`'):
_sm_mod.generate(deps, 'halloween', PlaylistConfig())