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/core/metadata_service.py

230 lines
8.3 KiB

"""
Metadata Service - Hot-swappable Spotify/iTunes provider
Automatically uses Spotify when authenticated, falls back to iTunes when not.
Provides unified interface for all metadata operations.
"""
from typing import List, Optional, Dict, Any, Literal
from core.spotify_client import SpotifyClient
from core.itunes_client import iTunesClient
from utils.logging_config import get_logger
logger = get_logger("metadata_service")
MetadataProvider = Literal["spotify", "itunes", "auto"]
class MetadataService:
"""
Unified metadata service that seamlessly switches between Spotify and iTunes.
Usage:
service = MetadataService()
tracks = service.search_tracks("Radiohead OK Computer")
# Uses Spotify if authenticated, otherwise iTunes
"""
def __init__(self, preferred_provider: MetadataProvider = "auto"):
"""
Initialize metadata service.
Args:
preferred_provider: "spotify", "itunes", or "auto" (default)
- "auto": Use Spotify if authenticated, else iTunes
- "spotify": Always use Spotify (may fail if not authenticated)
- "itunes": Always use iTunes
"""
self.preferred_provider = preferred_provider
self.spotify = SpotifyClient()
self.itunes = iTunesClient()
self._log_initialization()
def _log_initialization(self):
"""Log initialization status"""
spotify_status = "✅ Authenticated" if self.spotify.is_spotify_authenticated() else "❌ Not authenticated"
itunes_status = "✅ Available" if self.itunes.is_authenticated() else "❌ Not available"
logger.info(f"MetadataService initialized - Spotify: {spotify_status}, iTunes: {itunes_status}")
logger.info(f"Preferred provider: {self.preferred_provider}")
def get_active_provider(self) -> str:
"""
Get the currently active metadata provider.
Returns:
"spotify" or "itunes"
"""
if self.preferred_provider == "spotify":
return "spotify"
elif self.preferred_provider == "itunes":
return "itunes"
else: # auto
# Use is_spotify_authenticated() to check actual Spotify auth status
# (is_authenticated() always returns True due to iTunes fallback)
return "spotify" if self.spotify.is_spotify_authenticated() else "itunes"
def _get_client(self):
"""Get the appropriate client based on provider selection"""
provider = self.get_active_provider()
if provider == "spotify":
if not self.spotify.is_spotify_authenticated():
logger.warning("Spotify requested but not authenticated, falling back to iTunes")
return self.itunes
return self.spotify
else:
return self.itunes
# ==================== Search Methods ====================
def search_tracks(self, query: str, limit: int = 20) -> List:
"""
Search for tracks using active provider.
Args:
query: Search query
limit: Maximum results
Returns:
List of Track objects
"""
client = self._get_client()
provider = self.get_active_provider()
logger.debug(f"Searching tracks with {provider}: '{query}'")
return client.search_tracks(query, limit)
def search_artists(self, query: str, limit: int = 20) -> List:
"""
Search for artists using active provider.
Args:
query: Search query
limit: Maximum results
Returns:
List of Artist objects
"""
client = self._get_client()
provider = self.get_active_provider()
logger.debug(f"Searching artists with {provider}: '{query}'")
return client.search_artists(query, limit)
def search_albums(self, query: str, limit: int = 20) -> List:
"""
Search for albums using active provider.
Args:
query: Search query
limit: Maximum results
Returns:
List of Album objects
"""
client = self._get_client()
provider = self.get_active_provider()
logger.debug(f"Searching albums with {provider}: '{query}'")
return client.search_albums(query, limit)
# ==================== Detail Fetching ====================
def get_track_details(self, track_id: str) -> Optional[Dict[str, Any]]:
"""Get detailed track information"""
client = self._get_client()
return client.get_track_details(track_id)
def get_album(self, album_id: str) -> Optional[Dict[str, Any]]:
"""Get album information"""
client = self._get_client()
return client.get_album(album_id)
def get_album_tracks(self, album_id: str) -> Optional[Dict[str, Any]]:
"""Get all tracks from an album"""
client = self._get_client()
provider = self.get_active_provider()
logger.debug(f"Fetching album tracks with {provider}: {album_id}")
return client.get_album_tracks(album_id)
def get_artist(self, artist_id: str) -> Optional[Dict[str, Any]]:
"""Get artist information"""
client = self._get_client()
return client.get_artist(artist_id)
def get_artist_albums(self, artist_id: str, album_type: str = "album,single", limit: int = 50) -> List:
"""Get artist's albums/discography"""
client = self._get_client()
provider = self.get_active_provider()
logger.debug(f"Fetching artist albums with {provider}: {artist_id}")
return client.get_artist_albums(artist_id, album_type, limit)
def get_track_features(self, track_id: str) -> Optional[Dict[str, Any]]:
"""
Get track audio features (Spotify only).
Returns None for iTunes.
"""
client = self._get_client()
return client.get_track_features(track_id)
# ==================== User Library (Spotify only) ====================
def get_user_playlists(self) -> List:
"""Get user playlists (Spotify only)"""
if self.spotify.is_spotify_authenticated():
return self.spotify.get_user_playlists()
logger.warning("User playlists only available with Spotify authentication")
return []
def get_saved_tracks(self) -> List:
"""Get user's saved/liked tracks (Spotify only)"""
if self.spotify.is_spotify_authenticated():
return self.spotify.get_saved_tracks()
logger.warning("Saved tracks only available with Spotify authentication")
return []
def get_saved_tracks_count(self) -> int:
"""Get count of user's saved tracks (Spotify only)"""
if self.spotify.is_spotify_authenticated():
return self.spotify.get_saved_tracks_count()
return 0
# ==================== Utility Methods ====================
def is_authenticated(self) -> bool:
"""Check if any provider is available"""
return self.spotify.is_spotify_authenticated() or self.itunes.is_authenticated()
def get_provider_info(self) -> Dict[str, Any]:
"""Get information about available providers"""
return {
"active_provider": self.get_active_provider(),
"spotify_authenticated": self.spotify.is_spotify_authenticated(),
"itunes_available": self.itunes.is_authenticated(),
"preferred_provider": self.preferred_provider,
"can_access_user_data": self.spotify.is_spotify_authenticated(),
}
def reload_config(self):
"""Reload configuration for both clients"""
logger.info("Reloading metadata service configuration")
self.spotify.reload_config()
self.itunes.reload_config()
self._log_initialization()
# Convenience singleton instance
_metadata_service_instance: Optional[MetadataService] = None
def get_metadata_service() -> MetadataService:
"""
Get global metadata service instance (singleton pattern).
Returns:
MetadataService instance
"""
global _metadata_service_instance
if _metadata_service_instance is None:
_metadata_service_instance = MetadataService()
return _metadata_service_instance