2021-02-11 18:04:05 +00:00
|
|
|
import requests
|
|
|
|
from time import sleep
|
2020-11-12 15:55:24 +00:00
|
|
|
|
2020-11-19 20:04:41 +00:00
|
|
|
import json
|
2020-11-19 20:49:39 +00:00
|
|
|
from deezer.utils import map_artist_album, map_user_track, map_user_artist, map_user_album, map_user_playlist
|
2020-11-19 17:56:40 +00:00
|
|
|
|
2020-11-12 15:55:24 +00:00
|
|
|
class LyricsStatus():
|
|
|
|
"""Explicit Content Lyrics"""
|
|
|
|
|
|
|
|
NOT_EXPLICIT = 0
|
|
|
|
"""Not Explicit"""
|
|
|
|
|
|
|
|
EXPLICIT = 1
|
|
|
|
"""Explicit"""
|
|
|
|
|
|
|
|
UNKNOWN = 2
|
|
|
|
"""Unknown"""
|
|
|
|
|
|
|
|
EDITED = 3
|
|
|
|
"""Edited"""
|
|
|
|
|
|
|
|
PARTIALLY_EXPLICIT = 4
|
|
|
|
"""Partially Explicit (Album "lyrics" only)"""
|
|
|
|
|
|
|
|
PARTIALLY_UNKNOWN = 5
|
|
|
|
"""Partially Unknown (Album "lyrics" only)"""
|
|
|
|
|
|
|
|
NO_ADVICE = 6
|
|
|
|
"""No Advice Available"""
|
|
|
|
|
|
|
|
PARTIALLY_NO_ADVICE = 7
|
|
|
|
"""Partially No Advice Available (Album "lyrics" only)"""
|
|
|
|
|
2020-12-10 16:05:26 +00:00
|
|
|
class PlaylistStatus():
|
|
|
|
PUBLIC = 0
|
|
|
|
PRIVATE = 1
|
|
|
|
COLLABORATIVE = 2
|
|
|
|
|
2021-03-12 14:02:11 +00:00
|
|
|
EMPTY_TRACK_OBJ = {
|
2020-11-12 15:55:24 +00:00
|
|
|
'SNG_ID': 0,
|
|
|
|
'SNG_TITLE': '',
|
|
|
|
'DURATION': 0,
|
|
|
|
'MD5_ORIGIN': 0,
|
|
|
|
'MEDIA_VERSION': 0,
|
|
|
|
'FILESIZE': 0,
|
|
|
|
'ALB_TITLE': "",
|
|
|
|
'ALB_PICTURE': "",
|
|
|
|
'ART_ID': 0,
|
|
|
|
'ART_NAME': ""
|
|
|
|
}
|
|
|
|
|
|
|
|
class GW:
|
|
|
|
def __init__(self, session, headers):
|
|
|
|
self.http_headers = headers
|
|
|
|
self.session = session
|
|
|
|
|
|
|
|
def api_call(self, method, args=None, params=None):
|
|
|
|
if args is None: args = {}
|
|
|
|
if params is None: params = {}
|
|
|
|
p = {'api_version': "1.0",
|
|
|
|
'api_token': 'null' if method == 'deezer.getUserData' else self._get_token(),
|
|
|
|
'input': '3',
|
|
|
|
'method': method}
|
|
|
|
p.update(params)
|
|
|
|
try:
|
2021-03-12 14:02:11 +00:00
|
|
|
result_json = self.session.post(
|
2020-11-12 15:55:24 +00:00
|
|
|
"http://www.deezer.com/ajax/gw-light.php",
|
|
|
|
params=p,
|
|
|
|
timeout=30,
|
|
|
|
json=args,
|
|
|
|
headers=self.http_headers
|
2021-03-12 14:02:11 +00:00
|
|
|
).json()
|
2020-11-12 15:55:24 +00:00
|
|
|
except:
|
2021-02-11 18:04:05 +00:00
|
|
|
sleep(2)
|
2020-11-12 15:55:24 +00:00
|
|
|
return self.api_call(method, args, params)
|
|
|
|
if len(result_json['error']):
|
2021-03-19 15:01:45 +00:00
|
|
|
raise GWAPIError(json.dumps(result_json['error']))
|
2021-03-12 14:02:11 +00:00
|
|
|
return result_json['results']
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def _get_token(self):
|
|
|
|
token_data = self.get_user_data()
|
|
|
|
return token_data['checkForm']
|
|
|
|
|
|
|
|
def get_user_data(self):
|
|
|
|
return self.api_call('deezer.getUserData')
|
|
|
|
|
2020-11-19 20:49:39 +00:00
|
|
|
def get_user_profile_page(self, user_id, tab, limit=10):
|
|
|
|
return self.api_call('deezer.pageProfile', {'user_id': user_id, 'tab': tab, 'nb': limit})
|
|
|
|
|
2020-11-12 15:55:24 +00:00
|
|
|
def get_child_accounts(self):
|
|
|
|
return self.api_call('deezer.getChildAccounts')
|
|
|
|
|
|
|
|
def get_track(self, sng_id):
|
|
|
|
return self.api_call('song.getData', {'sng_id': sng_id})
|
|
|
|
|
|
|
|
def get_track_page(self, sng_id):
|
|
|
|
return self.api_call('deezer.pageTrack', {'sng_id': sng_id})
|
|
|
|
|
|
|
|
def get_track_lyrics(self, sng_id):
|
|
|
|
return self.api_call('song.getLyrics', {'sng_id': sng_id})
|
|
|
|
|
|
|
|
def get_tracks_gw(self, sng_ids):
|
|
|
|
tracks_array = []
|
|
|
|
body = self.api_call('song.getListData', {'sng_ids': sng_ids})
|
|
|
|
errors = 0
|
|
|
|
for i in range(len(sng_ids)):
|
|
|
|
if sng_ids[i] != 0:
|
|
|
|
tracks_array.append(body['data'][i - errors])
|
|
|
|
else:
|
|
|
|
errors += 1
|
2021-03-12 14:02:11 +00:00
|
|
|
tracks_array.append(EMPTY_TRACK_OBJ)
|
2020-11-12 15:55:24 +00:00
|
|
|
return tracks_array
|
|
|
|
|
|
|
|
def get_album(self, alb_id):
|
|
|
|
return self.api_call('album.getData', {'alb_id': alb_id})
|
|
|
|
|
|
|
|
def get_album_page(self, alb_id):
|
|
|
|
return self.api_call('deezer.pageAlbum', {
|
2021-03-12 14:02:11 +00:00
|
|
|
'alb_id': alb_id,
|
|
|
|
'lang': 'en',
|
|
|
|
'header': True,
|
|
|
|
'tab': 0
|
|
|
|
})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def get_album_tracks(self, alb_id):
|
|
|
|
tracks_array = []
|
|
|
|
body = self.api_call('song.getListByAlbum', {'alb_id': alb_id, 'nb': -1})
|
|
|
|
for track in body['data']:
|
2021-03-12 14:02:11 +00:00
|
|
|
track['POSITION'] = body['data'].index(track)
|
2021-04-09 17:28:41 +00:00
|
|
|
tracks_array.append(track)
|
2020-11-12 15:55:24 +00:00
|
|
|
return tracks_array
|
|
|
|
|
|
|
|
def get_artist(self, art_id):
|
|
|
|
return self.api_call('artist.getData', {'art_id': art_id})
|
|
|
|
|
|
|
|
def get_artist_page(self, art_id):
|
|
|
|
return self.api_call('deezer.pageArtist', {
|
2021-03-12 14:02:11 +00:00
|
|
|
'art_id': art_id,
|
|
|
|
'lang': 'en',
|
|
|
|
'header': True,
|
|
|
|
'tab': 0
|
|
|
|
})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def get_artist_top_tracks(self, art_id, limit=100):
|
|
|
|
tracks_array = []
|
|
|
|
body = self.api_call('artist.getTopTrack', {'art_id': art_id, 'nb': limit})
|
|
|
|
for track in body['data']:
|
|
|
|
track['POSITION'] = body['data'].index(track)
|
|
|
|
tracks_array.append(track)
|
|
|
|
return tracks_array
|
|
|
|
|
|
|
|
def get_artist_discography(self, art_id, index=0, limit=25):
|
|
|
|
return self.api_call('album.getDiscography', {
|
2021-03-12 14:02:11 +00:00
|
|
|
'art_id': art_id,
|
|
|
|
"discography_mode":"all",
|
|
|
|
'nb': limit,
|
|
|
|
'nb_songs': 0,
|
|
|
|
'start': index
|
|
|
|
})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def get_playlist(self, playlist_id):
|
|
|
|
return self.api_call('playlist.getData', {'playlist_id': playlist_id})
|
|
|
|
|
|
|
|
def get_playlist_page(self, playlist_id):
|
|
|
|
return self.api_call('deezer.pagePlaylist', {
|
2021-03-12 14:02:11 +00:00
|
|
|
'playlist_id': playlist_id,
|
|
|
|
'lang': 'en',
|
|
|
|
'header': True,
|
|
|
|
'tab': 0
|
|
|
|
})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def get_playlist_tracks(self, playlist_id):
|
|
|
|
tracks_array = []
|
|
|
|
body = self.api_call('playlist.getSongs', {'playlist_id': playlist_id, 'nb': -1})
|
|
|
|
for track in body['data']:
|
|
|
|
track['POSITION'] = body['data'].index(track)
|
|
|
|
tracks_array.append(track)
|
|
|
|
return tracks_array
|
|
|
|
|
2020-12-10 16:05:26 +00:00
|
|
|
def create_playlist(self, title, status=PlaylistStatus.PUBLIC, description=None, songs=[]):
|
|
|
|
newSongs = []
|
|
|
|
for song in songs:
|
|
|
|
newSongs.append([song, 0])
|
|
|
|
return self.api_call('playlist.create', {
|
|
|
|
'title': title,
|
|
|
|
'status': status,
|
|
|
|
'description': description,
|
|
|
|
'songs': newSongs
|
|
|
|
})
|
|
|
|
|
2021-03-12 14:02:11 +00:00
|
|
|
def edit_playlist(self, playlist_id, title, status=None, description=None, songs=[]):
|
2020-12-10 16:05:26 +00:00
|
|
|
newSongs = []
|
|
|
|
for song in songs:
|
|
|
|
newSongs.append([song, 0])
|
|
|
|
return self.api_call('playlist.update', {
|
|
|
|
'playlist_id': playlist_id,
|
|
|
|
'title': title,
|
|
|
|
'status': status,
|
|
|
|
'description': description,
|
|
|
|
'songs': newSongs
|
|
|
|
})
|
|
|
|
|
|
|
|
def add_songs_to_playlist(self, playlist_id, songs, offset=-1):
|
|
|
|
newSongs = []
|
|
|
|
for song in songs:
|
|
|
|
newSongs.append([song, 0])
|
|
|
|
return self.api_call('playlist.addSongs', {
|
|
|
|
'playlist_id': playlist_id,
|
|
|
|
'songs': newSongs,
|
|
|
|
'offset': offset
|
|
|
|
})
|
|
|
|
|
|
|
|
def add_song_to_playlist(self, playlist_id, sng_id, offset=-1):
|
|
|
|
return self.add_songs_to_playlist(playlist_id, [sng_id], offset)
|
|
|
|
|
|
|
|
def remove_songs_from_playlist(self, playlist_id, songs):
|
|
|
|
newSongs = []
|
|
|
|
for song in songs:
|
|
|
|
newSongs.append([song, 0])
|
|
|
|
return self.api_call('playlist.deleteSongs', {
|
|
|
|
'playlist_id': playlist_id,
|
|
|
|
'songs': newSongs
|
|
|
|
})
|
|
|
|
|
|
|
|
def remove_song_from_playlist(self, playlist_id, sng_id):
|
|
|
|
return self.remove_songs_from_playlist(playlist_id, [sng_id])
|
|
|
|
|
|
|
|
def delete_playlist(self, playlist_id):
|
|
|
|
return self.api_call('playlist.delete', {'playlist_id': playlist_id})
|
|
|
|
|
2020-11-12 15:55:24 +00:00
|
|
|
def add_song_to_favorites(self, sng_id):
|
2021-04-09 17:28:41 +00:00
|
|
|
return self.api_call('favorite_song.add', {'SNG_ID': sng_id})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def remove_song_from_favorites(self, sng_id):
|
2021-04-09 17:28:41 +00:00
|
|
|
return self.api_call('favorite_song.remove', {'SNG_ID': sng_id})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def add_album_to_favorites(self, alb_id):
|
2021-04-09 17:28:41 +00:00
|
|
|
return self.api_call('album.addFavorite', {'ALB_ID': alb_id})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def remove_album_from_favorites(self, alb_id):
|
2021-04-09 17:28:41 +00:00
|
|
|
return self.api_call('album.deleteFavorite', {'ALB_ID': alb_id})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def add_artist_to_favorites(self, art_id):
|
2021-04-09 17:28:41 +00:00
|
|
|
return self.api_call('artist.addFavorite', {'ART_ID': art_id})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def remove_artist_from_favorites(self, art_id):
|
2021-04-09 17:28:41 +00:00
|
|
|
return self.api_call('artist.deleteFavorite', {'ART_ID': art_id})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def add_playlist_to_favorites(self, playlist_id):
|
2021-04-09 17:28:41 +00:00
|
|
|
return self.api_call('playlist.addFavorite', {'PARENT_PLAYLIST_ID': playlist_id})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def remove_playlist_from_favorites(self, playlist_id):
|
2021-04-09 17:28:41 +00:00
|
|
|
return self.api_call('playlist.deleteFavorite', {'PLAYLIST_ID': playlist_id})
|
2020-11-12 15:55:24 +00:00
|
|
|
|
|
|
|
def get_page(self, page):
|
|
|
|
params = {
|
|
|
|
'gateway_input': json.dumps({
|
|
|
|
'PAGE': page,
|
|
|
|
'VERSION': '2.3',
|
|
|
|
'SUPPORT': {
|
|
|
|
'grid': [
|
|
|
|
'channel',
|
|
|
|
'album'
|
|
|
|
],
|
|
|
|
'horizontal-grid': [
|
|
|
|
'album'
|
|
|
|
],
|
|
|
|
},
|
|
|
|
'LANG': 'en'
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return self.api_call('page.get', params=params)
|
|
|
|
|
2020-11-19 20:04:41 +00:00
|
|
|
def search(self, query, index=0, limit=10, suggest=True, artist_suggest=True, top_tracks=True):
|
|
|
|
return self.api_call('deezer.pageSearch', {
|
2021-03-12 14:02:11 +00:00
|
|
|
"query": query,
|
|
|
|
"start": index,
|
|
|
|
"nb": limit,
|
|
|
|
"suggest": suggest,
|
|
|
|
"artist_suggest": artist_suggest,
|
|
|
|
"top_tracks": top_tracks
|
|
|
|
})
|
2020-11-19 20:04:41 +00:00
|
|
|
|
2020-11-27 09:39:30 +00:00
|
|
|
def search_music(self, query, type, index=0, limit=10):
|
2021-03-12 14:02:11 +00:00
|
|
|
return self.api_call('search.music', {
|
|
|
|
"query": query,
|
|
|
|
"filter": "ALL",
|
|
|
|
"output": type,
|
|
|
|
"start": index,
|
|
|
|
"nb": limit
|
|
|
|
})
|
2020-11-27 09:39:30 +00:00
|
|
|
|
2020-11-19 17:56:40 +00:00
|
|
|
# Extra calls
|
|
|
|
|
|
|
|
def get_artist_discography_tabs(self, art_id, limit=100):
|
|
|
|
index = 0
|
|
|
|
releases = []
|
|
|
|
result = {'all': []}
|
|
|
|
ids = []
|
2020-11-19 19:42:41 +00:00
|
|
|
|
2020-11-19 17:56:40 +00:00
|
|
|
# Get all releases
|
|
|
|
while True:
|
|
|
|
response = self.get_artist_discography(art_id, index=index, limit=limit)
|
|
|
|
releases += response['data']
|
|
|
|
index += limit
|
|
|
|
if index > response['total']:
|
|
|
|
break
|
|
|
|
|
|
|
|
for release in releases:
|
|
|
|
if release['ALB_ID'] not in ids:
|
|
|
|
ids.append(release['ALB_ID'])
|
|
|
|
obj = map_artist_album(release)
|
|
|
|
if (release['ART_ID'] == art_id or release['ART_ID'] != art_id and release['ROLE_ID'] == 0) and release['ARTISTS_ALBUMS_IS_OFFICIAL']:
|
|
|
|
# Handle all base record types
|
|
|
|
if not obj['record_type'] in result:
|
|
|
|
result[obj['record_type']] = []
|
|
|
|
result[obj['record_type']].append(obj)
|
|
|
|
result['all'].append(obj)
|
|
|
|
else:
|
|
|
|
# Handle albums where the artist is featured
|
|
|
|
if release['ROLE_ID'] == 5:
|
|
|
|
if not 'featured' in result:
|
|
|
|
result['featured'] = []
|
|
|
|
result['featured'].append(obj)
|
|
|
|
# Handle "more" albums
|
|
|
|
elif release['ROLE_ID'] == 0:
|
|
|
|
if not 'more' in result:
|
|
|
|
result['more'] = []
|
|
|
|
result['more'].append(obj)
|
|
|
|
result['all'].append(obj)
|
|
|
|
return result
|
|
|
|
|
2020-11-19 19:42:41 +00:00
|
|
|
def get_track_with_fallback(self, sng_id):
|
|
|
|
body = None
|
|
|
|
if int(sng_id) > 0:
|
|
|
|
try:
|
|
|
|
body = self.get_track_page(sng_id)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
if body:
|
|
|
|
if 'LYRICS' in body:
|
|
|
|
body['DATA']['LYRICS'] = body['LYRICS']
|
|
|
|
body = body['DATA']
|
|
|
|
else:
|
|
|
|
body = self.get_track(sng_id)
|
|
|
|
return body
|
|
|
|
|
2020-11-19 22:13:04 +00:00
|
|
|
def get_user_playlists(self, user_id, limit=25):
|
|
|
|
user_profile_page = self.get_user_profile_page(user_id, 'playlists', limit=limit)
|
2021-03-12 14:02:11 +00:00
|
|
|
blog_name = user_profile_page['DATA']['USER'].get('BLOG_NAME', "Unknown")
|
2020-11-19 21:05:26 +00:00
|
|
|
data = user_profile_page['TAB']['playlists']['data']
|
2020-11-19 20:49:39 +00:00
|
|
|
result = []
|
|
|
|
for playlist in data:
|
2020-11-19 21:05:26 +00:00
|
|
|
result.append(map_user_playlist(playlist, blog_name))
|
2020-11-19 20:49:39 +00:00
|
|
|
return result
|
|
|
|
|
2020-11-19 22:13:04 +00:00
|
|
|
def get_user_albums(self, user_id, limit=25):
|
|
|
|
data = self.get_user_profile_page(user_id, 'albums', limit=limit)['TAB']['albums']['data']
|
2020-11-19 20:49:39 +00:00
|
|
|
result = []
|
|
|
|
for album in data:
|
|
|
|
result.append(map_user_album(album))
|
|
|
|
return result
|
|
|
|
|
2020-11-19 22:13:04 +00:00
|
|
|
def get_user_artists(self, user_id, limit=25):
|
|
|
|
data = self.get_user_profile_page(user_id, 'artists', limit=limit)['TAB']['artists']['data']
|
2020-11-19 20:49:39 +00:00
|
|
|
result = []
|
|
|
|
for artist in data:
|
|
|
|
result.append(map_user_artist(artist))
|
|
|
|
return result
|
|
|
|
|
2020-11-19 22:13:04 +00:00
|
|
|
def get_user_tracks(self, user_id, limit=25):
|
|
|
|
data = self.get_user_profile_page(user_id, 'loved', limit=limit)['TAB']['loved']['data']
|
2020-11-19 20:49:39 +00:00
|
|
|
result = []
|
|
|
|
for track in data:
|
|
|
|
result.append(map_user_track(track))
|
|
|
|
return result
|
|
|
|
|
2021-03-19 15:01:45 +00:00
|
|
|
class GWAPIError(Exception):
|
2020-11-12 15:55:24 +00:00
|
|
|
"""Base class for Deezer exceptions"""
|
|
|
|
pass
|