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.
335 lines
9.6 KiB
335 lines
9.6 KiB
from contextlib import contextmanager
|
|
|
|
from core.wishlist import processing
|
|
|
|
|
|
class _FakeLogger:
|
|
def __init__(self):
|
|
self.errors = []
|
|
self.infos = []
|
|
self.warnings = []
|
|
|
|
def error(self, msg):
|
|
self.errors.append(msg)
|
|
|
|
def info(self, msg):
|
|
self.infos.append(msg)
|
|
|
|
def warning(self, msg):
|
|
self.warnings.append(msg)
|
|
|
|
|
|
class _FakeAutomationEngine:
|
|
def __init__(self):
|
|
self.events = []
|
|
|
|
def emit(self, name, payload):
|
|
self.events.append((name, payload))
|
|
|
|
|
|
class _FakeCursor:
|
|
def __init__(self):
|
|
self.calls = []
|
|
|
|
def execute(self, sql, params=None):
|
|
self.calls.append((sql, params))
|
|
|
|
|
|
class _FakeConnection:
|
|
def __init__(self):
|
|
self.cursor_obj = _FakeCursor()
|
|
self.committed = False
|
|
|
|
def cursor(self):
|
|
return self.cursor_obj
|
|
|
|
def commit(self):
|
|
self.committed = True
|
|
|
|
|
|
class _FakeDB:
|
|
def __init__(self):
|
|
self.connection = _FakeConnection()
|
|
|
|
@contextmanager
|
|
def _get_connection(self):
|
|
yield self.connection
|
|
|
|
|
|
class _FakeLock:
|
|
def __enter__(self):
|
|
return self
|
|
|
|
def __exit__(self, exc_type, exc, tb):
|
|
return False
|
|
|
|
|
|
def test_remove_completed_tracks_from_wishlist_calls_remover():
|
|
batch = {"queue": ["a", "b"]}
|
|
download_tasks = {
|
|
"a": {"status": "completed", "track_info": {"name": "Song A"}},
|
|
"b": {"status": "failed", "track_info": {"name": "Song B"}},
|
|
}
|
|
calls = []
|
|
|
|
removed = processing.remove_completed_tracks_from_wishlist(
|
|
batch,
|
|
download_tasks,
|
|
lambda context: calls.append(context),
|
|
logger=_FakeLogger(),
|
|
)
|
|
|
|
assert removed == 1
|
|
assert calls == [{"track_info": {"name": "Song A"}, "original_search_result": {"name": "Song A"}}]
|
|
|
|
|
|
def test_add_cancelled_tracks_to_failed_tracks_builds_entries():
|
|
batch = {"queue": ["a"], "cancelled_tracks": {1}}
|
|
download_tasks = {
|
|
"a": {
|
|
"status": "cancelled",
|
|
"track_index": 1,
|
|
"track_info": {"name": "Song A", "artist": "Artist A", "artists": [{"name": "Artist A"}]},
|
|
"cached_candidates": [{"title": "candidate"}],
|
|
}
|
|
}
|
|
failed = []
|
|
|
|
processed = processing.add_cancelled_tracks_to_failed_tracks(
|
|
batch,
|
|
download_tasks,
|
|
failed,
|
|
logger=_FakeLogger(),
|
|
)
|
|
|
|
assert processed == 1
|
|
assert failed[0]["track_name"] == "Song A"
|
|
assert failed[0]["artist_name"] == "Artist A"
|
|
assert failed[0]["failure_reason"] == "Download cancelled"
|
|
|
|
|
|
def test_recover_uncaptured_failed_tracks_builds_entries():
|
|
batch = {"queue": ["a"]}
|
|
download_tasks = {
|
|
"a": {
|
|
"status": "failed",
|
|
"track_index": 2,
|
|
"track_info": {"name": "Song B", "artist": "Artist B", "artists": [{"name": "Artist B"}]},
|
|
"retry_count": 3,
|
|
"error_message": "boom",
|
|
"cached_candidates": [],
|
|
}
|
|
}
|
|
failed = []
|
|
|
|
recovered = processing.recover_uncaptured_failed_tracks(
|
|
batch,
|
|
download_tasks,
|
|
failed,
|
|
logger=_FakeLogger(),
|
|
)
|
|
|
|
assert recovered == 1
|
|
assert failed[0]["track_name"] == "Song B"
|
|
assert failed[0]["retry_count"] == 3
|
|
assert failed[0]["failure_reason"] == "boom"
|
|
|
|
|
|
def test_finalize_auto_wishlist_completion_toggles_cycle_and_resets_state():
|
|
db = _FakeDB()
|
|
automation_engine = _FakeAutomationEngine()
|
|
resets = []
|
|
activities = []
|
|
summary = {"tracks_added": 2, "total_failed": 5, "errors": 0}
|
|
|
|
result = processing.finalize_auto_wishlist_completion(
|
|
"batch-1",
|
|
summary,
|
|
download_batches={"batch-1": {"current_cycle": "albums"}},
|
|
tasks_lock=_FakeLock(),
|
|
reset_processing_state=lambda: resets.append(True),
|
|
add_activity_item=lambda *args: activities.append(args),
|
|
automation_engine=automation_engine,
|
|
db_factory=lambda: db,
|
|
logger=_FakeLogger(),
|
|
)
|
|
|
|
assert result is summary
|
|
assert resets == [True]
|
|
assert activities == [("", "Wishlist Updated", "2 failed tracks added to wishlist", "Now")]
|
|
assert automation_engine.events == [
|
|
(
|
|
"wishlist_processing_completed",
|
|
{
|
|
"tracks_processed": "5",
|
|
"tracks_found": "2",
|
|
"tracks_failed": "3",
|
|
},
|
|
)
|
|
]
|
|
assert db.connection.committed is True
|
|
assert db.connection.cursor_obj.calls[0][1] == ("singles",)
|
|
|
|
|
|
def test_automatic_wishlist_cleanup_after_db_update_removes_library_matches():
|
|
class _CleanupWishlistService:
|
|
def __init__(self, tracks):
|
|
self.tracks = tracks
|
|
self.removed = []
|
|
|
|
def get_wishlist_tracks_for_download(self, profile_id=1):
|
|
return list(self.tracks)
|
|
|
|
def mark_track_download_result(self, spotify_track_id, success, error_message=None, profile_id=1):
|
|
self.removed.append((spotify_track_id, success, error_message, profile_id))
|
|
return True
|
|
|
|
class _CleanupProfilesDatabase:
|
|
def get_all_profiles(self):
|
|
return [{"id": 1}]
|
|
|
|
class _CleanupMusicDatabase:
|
|
def check_track_exists(self, track_name, artist_name, confidence_threshold=0.7, server_source=None, album=None):
|
|
if track_name == "Song A" and artist_name == "Artist A":
|
|
return {"id": "db-track"}, 0.9
|
|
return None, 0.0
|
|
|
|
wishlist_service = _CleanupWishlistService(
|
|
[
|
|
{
|
|
"name": "Song A",
|
|
"artists": [{"name": "Artist A"}],
|
|
"spotify_track_id": "sp-1",
|
|
"id": "sp-1",
|
|
"album": {"name": "Album A"},
|
|
},
|
|
{
|
|
"name": "Song B",
|
|
"artists": [{"name": "Artist B"}],
|
|
"spotify_track_id": "sp-2",
|
|
"id": "sp-2",
|
|
"album": {"name": "Album B"},
|
|
},
|
|
]
|
|
)
|
|
|
|
removed = processing.automatic_wishlist_cleanup_after_db_update(
|
|
wishlist_service=wishlist_service,
|
|
profiles_database=_CleanupProfilesDatabase(),
|
|
music_database=_CleanupMusicDatabase(),
|
|
active_server="navidrome",
|
|
logger=_FakeLogger(),
|
|
)
|
|
|
|
assert removed == 1
|
|
assert wishlist_service.removed == [("sp-1", True, None, 1)]
|
|
|
|
|
|
class _CleanupProfilesDatabase:
|
|
def get_all_profiles(self):
|
|
return [{"id": 1}]
|
|
|
|
|
|
class _CleanupWishlistService:
|
|
def __init__(self, tracks):
|
|
self._tracks = list(tracks)
|
|
self.removed = []
|
|
|
|
def get_wishlist_tracks_for_download(self, profile_id=1):
|
|
return list(self._tracks)
|
|
|
|
def mark_track_download_result(self, spotify_track_id, success, error_message=None, profile_id=1):
|
|
self.removed.append((spotify_track_id, success, error_message, profile_id))
|
|
return True
|
|
|
|
|
|
class _CleanupMusicDatabase:
|
|
def __init__(self):
|
|
self.track_checks = []
|
|
|
|
def check_track_exists(self, track_name, artist_name, confidence_threshold=0.7, server_source=None, album=None):
|
|
self.track_checks.append((track_name, artist_name, server_source, album))
|
|
if track_name == "Owned Song" and artist_name == "Artist A":
|
|
return {"id": "db-track"}, 0.9
|
|
if track_name == "Broken Song":
|
|
raise RuntimeError("boom")
|
|
return None, 0.0
|
|
|
|
|
|
def test_remove_tracks_already_in_library_skips_enhance_tracks_and_handles_errors():
|
|
wishlist_service = _CleanupWishlistService(
|
|
[
|
|
{
|
|
"name": "Owned Song",
|
|
"artists": ["Artist A"],
|
|
"spotify_track_id": "sp-1",
|
|
"album": {"name": "Album A"},
|
|
},
|
|
{
|
|
"name": "Enhance Song",
|
|
"artists": [{"name": "Artist B"}],
|
|
"spotify_track_id": "sp-2",
|
|
"source_type": "enhance",
|
|
"album": {"name": "Album B"},
|
|
},
|
|
{
|
|
"name": "Broken Song",
|
|
"artists": [{"name": "Artist C"}],
|
|
"spotify_track_id": "sp-3",
|
|
"album": {"name": "Album C"},
|
|
},
|
|
]
|
|
)
|
|
music_db = _CleanupMusicDatabase()
|
|
|
|
removed = processing.remove_tracks_already_in_library(
|
|
wishlist_service,
|
|
_CleanupProfilesDatabase(),
|
|
music_db,
|
|
"navidrome",
|
|
logger=_FakeLogger(),
|
|
skip_track_fn=lambda track: track.get("source_type") == "enhance",
|
|
)
|
|
|
|
assert removed == 1
|
|
assert wishlist_service.removed == [("sp-1", True, None, 1)]
|
|
assert music_db.track_checks == [
|
|
("Owned Song", "Artist A", "navidrome", "Album A"),
|
|
("Broken Song", "Artist C", "navidrome", "Album C"),
|
|
]
|
|
|
|
|
|
def test_finalize_auto_wishlist_completion_with_no_tracks_added_still_resets_state():
|
|
db = _FakeDB()
|
|
automation_engine = _FakeAutomationEngine()
|
|
resets = []
|
|
activities = []
|
|
summary = {"tracks_added": 0, "total_failed": 5, "errors": 0}
|
|
|
|
result = processing.finalize_auto_wishlist_completion(
|
|
"batch-2",
|
|
summary,
|
|
download_batches={"batch-2": {"current_cycle": "singles"}},
|
|
tasks_lock=_FakeLock(),
|
|
reset_processing_state=lambda: resets.append(True),
|
|
add_activity_item=lambda *args: activities.append(args),
|
|
automation_engine=automation_engine,
|
|
db_factory=lambda: db,
|
|
logger=_FakeLogger(),
|
|
)
|
|
|
|
assert result is summary
|
|
assert resets == [True]
|
|
assert activities == []
|
|
assert automation_engine.events == [
|
|
(
|
|
"wishlist_processing_completed",
|
|
{
|
|
"tracks_processed": "5",
|
|
"tracks_found": "0",
|
|
"tracks_failed": "5",
|
|
},
|
|
)
|
|
]
|
|
assert db.connection.committed is True
|