from collections.abc import Callable from tidalapi import Album, Mix, Playlist, Session, Track, UserPlaylist, Video from tidalapi.artist import Artist, Role from tidalapi.media import MediaMetadataTags, Quality from tidalapi.session import SearchTypes from tidalapi.user import LoggedInUser from tidal_dl_ng.constants import FAVORITES, MediaType from tidal_dl_ng.helper.exceptions import MediaUnknown def name_builder_artist(media: Track | Video | Album, delimiter: str = ", ") -> str: """Builds a string of artist names for a track, video, or album. Returns a delimited string of all artist names associated with the given media. Args: media (Track | Video | Album): The media object to extract artist names from. delimiter (str, optional): The delimiter to use between artist names. Defaults to ", ". Returns: str: A delimited string of artist names. """ return delimiter.join(artist.name for artist in media.artists) def name_builder_album_artist(media: Track | Album, first_only: bool = False, delimiter: str = ", ") -> str: """Builds a string of main album artist names for a track or album. Returns a delimited string of main artist names from the album, optionally including only the first main artist. Args: media (Track | Album): The media object to extract artist names from. first_only (bool, optional): If True, only the first main artist is included. Defaults to False. delimiter (str, optional): The delimiter to use between artist names. Defaults to ", ". Returns: str: A delimited string of main album artist names. """ artists_tmp: [str] = [] artists: [Artist] = media.album.artists if isinstance(media, Track) else media.artists for artist in artists: if Role.main in artist.roles: artists_tmp.append(artist.name) if first_only: break return delimiter.join(artists_tmp) def name_builder_title(media: Track | Video | Mix | Playlist | Album | Video) -> str: result: str = ( media.title if isinstance(media, Mix) else media.full_name if hasattr(media, "full_name") else media.name ) return result def name_builder_item(media: Track | Video) -> str: return f"{name_builder_artist(media)} - {name_builder_title(media)}" def get_tidal_media_id(url_or_id_media: str) -> str: id_dirty = url_or_id_media.rsplit("/", 1)[-1] id_media = id_dirty.rsplit("?", 1)[0] return id_media def get_tidal_media_type(url_media: str) -> MediaType | bool: result: MediaType | bool = False url_split = url_media.split("/")[-2] if len(url_split) > 1: media_name = url_media.split("/")[-2] if media_name == "track": result = MediaType.TRACK elif media_name == "video": result = MediaType.VIDEO elif media_name == "album": result = MediaType.ALBUM elif media_name == "playlist": result = MediaType.PLAYLIST elif media_name == "mix": result = MediaType.MIX elif media_name == "artist": result = MediaType.ARTIST return result def url_ending_clean(url: str) -> str: """Checks if a link ends with "/u" or "?u" and removes that part. Args: url (str): The URL to clean. Returns: str: The cleaned URL. """ return url[:-2] if url.endswith("/u") or url.endswith("?u") else url def search_results_all(session: Session, needle: str, types_media: SearchTypes = None) -> dict[str, [SearchTypes]]: limit: int = 300 offset: int = 0 done: bool = False result: dict[str, [SearchTypes]] = {} while not done: tmp_result: dict[str, [SearchTypes]] = session.search( query=needle, models=types_media, limit=limit, offset=offset ) tmp_done: bool = True for key, value in tmp_result.items(): # Append pagination results, if there are any if offset == 0: result = tmp_result tmp_done = False elif bool(value): result[key] += value tmp_done = False # Next page offset += limit done = tmp_done return result def items_results_all( media_list: [Mix | Playlist | Album | Artist], videos_include: bool = True ) -> [Track | Video | Album]: result: [Track | Video | Album] = [] if isinstance(media_list, Mix): result = media_list.items() else: func_get_items_media: [Callable] = [] if isinstance(media_list, Playlist | Album): if videos_include: func_get_items_media.append(media_list.items) else: func_get_items_media.append(media_list.tracks) else: func_get_items_media.append(media_list.get_albums) func_get_items_media.append(media_list.get_ep_singles) result = paginate_results(func_get_items_media) return result def all_artist_album_ids(media_artist: Artist) -> [int | None]: result: [int] = [] func_get_items_media: [Callable] = [media_artist.get_albums, media_artist.get_ep_singles] albums: [Album] = paginate_results(func_get_items_media) for album in albums: result.append(album.id) return result def paginate_results(func_get_items_media: [Callable]) -> [Track | Video | Album | Playlist | UserPlaylist]: result: [Track | Video | Album] = [] for func_media in func_get_items_media: limit: int = 100 offset: int = 0 done: bool = False if func_media.__func__ == LoggedInUser.playlist_and_favorite_playlists: limit: int = 50 while not done: tmp_result: [Track | Video | Album | Playlist | UserPlaylist] = func_media(limit=limit, offset=offset) if bool(tmp_result): result += tmp_result # Get the next page in the next iteration. offset += limit else: done = True return result def user_media_lists(session: Session) -> dict[str, list]: """Fetch user media lists using tidalapi's built-in pagination where available. Returns a dictionary with 'playlists' and 'mixes' keys containing lists of media items. For playlists, includes both Folder and Playlist objects at the root level. Args: session (Session): TIDAL session object. Returns: dict[str, list]: Dictionary with 'playlists' (includes Folder and Playlist) and 'mixes' lists. """ # Use built-in pagination for playlists (root level only) playlists = session.user.favorites.playlists_paginated() # Fetch root-level folders manually (no paginated version available) folders = [] offset = 0 limit = 50 while True: batch = session.user.favorites.playlist_folders(limit=limit, offset=offset, parent_folder_id="root") if not batch: break folders.extend(batch) if len(batch) < limit: break offset += limit # Combine folders and playlists all_playlists = folders + playlists # Get mixes user_mixes = session.mixes().categories[0].items return {"playlists": all_playlists, "mixes": user_mixes} def instantiate_media( session: Session, media_type: type[MediaType.TRACK, MediaType.VIDEO, MediaType.ALBUM, MediaType.PLAYLIST, MediaType.MIX], id_media: str, ) -> Track | Video | Album | Playlist | Mix | Artist: if media_type == MediaType.TRACK: media = session.track(id_media, with_album=True) elif media_type == MediaType.VIDEO: media = session.video(id_media) elif media_type == MediaType.ALBUM: media = session.album(id_media) elif media_type == MediaType.PLAYLIST: media = session.playlist(id_media) elif media_type == MediaType.MIX: media = session.mix(id_media) elif media_type == MediaType.ARTIST: media = session.artist(id_media) else: raise MediaUnknown return media def quality_audio_highest(media: Track | Album) -> Quality: quality: Quality if MediaMetadataTags.hi_res_lossless in media.media_metadata_tags: quality = Quality.hi_res_lossless elif MediaMetadataTags.lossless in media.media_metadata_tags: quality = Quality.high_lossless else: quality = media.audio_quality return quality def favorite_function_factory(tidal, favorite_item: str): function_name: str = FAVORITES[favorite_item]["function_name"] function_list: Callable = getattr(tidal.session.user.favorites, function_name) return function_list