import re import requests import json from deezer.gw import GW from deezer.api import API from deezer.errors import DeezerError, WrongLicense, WrongGeolocation __version__ = "1.3.0" class TrackFormats(): """Number associtation for formats""" FLAC = 9 MP3_320 = 3 MP3_128 = 1 MP4_RA3 = 15 MP4_RA2 = 14 MP4_RA1 = 13 DEFAULT = 8 LOCAL = 0 class Deezer: def __init__(self): self.http_headers = { "User-Agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) " \ "Chrome/79.0.3945.130 Safari/537.36" } self.session = requests.Session() self.logged_in = False self.current_user = {} self.childs = [] self.selected_account = 0 self.api = API(self.session, self.http_headers) self.gw = GW(self.session, self.http_headers) def get_session(self): return { 'logged_in': self.logged_in, 'current_user': self.current_user, 'childs': self.childs, 'selected_account': self.selected_account, 'cookies': self.session.cookies.get_dict() } def set_session(self, data): self.logged_in = data['logged_in'] self.current_user = data['current_user'] self.childs = data['childs'] self.selected_account = data['selected_account'] self.session = requests.Session() self.session.cookies.update(data['cookies']) def login(self, email, password, re_captcha_token, child=0): if child: child = int(child) # Check if user already logged in user_data = self.gw.get_user_data() if not user_data or user_data and len(user_data.keys()) == 0: self.logged_in = False return False if user_data['USER']['USER_ID'] == 0: self.logged_in = False return False # Get the checkFormLogin check_form_login = user_data['checkFormLogin'] login = self.session.post( "https://www.deezer.com/ajax/action.php", data={ 'type': 'login', 'mail': email, 'password': password, 'checkFormLogin': check_form_login, 'reCaptchaToken': re_captcha_token }, headers={'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8', **self.http_headers} ) # Check if user logged in if 'success' not in login.text: self.logged_in = False return False user_data = self.gw.get_user_data() self._post_login(user_data) self.change_account(child) self.logged_in = True return True def login_via_arl(self, arl, child=0): arl = arl.strip() if child: child = int(child) cookie_obj = requests.cookies.create_cookie( domain='.deezer.com', name='arl', value=arl, path="/", rest={'HttpOnly': True} ) self.session.cookies.set_cookie(cookie_obj) user_data = self.gw.get_user_data() # Check if user logged in if not user_data or user_data and len(user_data.keys()) == 0: self.logged_in = False return False if user_data["USER"]["USER_ID"] == 0: self.logged_in = False return False self._post_login(user_data) self.change_account(child) self.logged_in = True return True def _post_login(self, user_data): self.childs = [] family = user_data["USER"]["MULTI_ACCOUNT"]["ENABLED"] and not user_data["USER"]["MULTI_ACCOUNT"]["IS_SUB_ACCOUNT"] if family: childs = self.gw.get_child_accounts() for child in childs: if child['EXTRA_FAMILY']['IS_LOGGABLE_AS']: self.childs.append({ 'id': child["USER_ID"], 'name': child["BLOG_NAME"], 'picture': child.get("USER_PICTURE", ""), 'license_token': user_data["USER"]["OPTIONS"]["license_token"], 'can_stream_hq': user_data["USER"]["OPTIONS"]["web_hq"] or user_data["USER"]["OPTIONS"]["mobile_hq"], 'can_stream_lossless': user_data["USER"]["OPTIONS"]["web_lossless"] or user_data["USER"]["OPTIONS"]["mobile_lossless"], 'country': user_data["USER"]["OPTIONS"]["license_country"], 'language': user_data["USER"]["SETTING"]["global"]["language"] }) else: self.childs.append({ 'id': user_data["USER"]["USER_ID"], 'name': user_data["USER"]["BLOG_NAME"], 'picture': user_data["USER"].get("USER_PICTURE", ""), 'license_token': user_data["USER"]["OPTIONS"]["license_token"], 'can_stream_hq': user_data["USER"]["OPTIONS"]["web_hq"] or user_data["USER"]["OPTIONS"]["mobile_hq"], 'can_stream_lossless': user_data["USER"]["OPTIONS"]["web_lossless"] or user_data["USER"]["OPTIONS"]["mobile_lossless"], 'country': user_data["USER"]["OPTIONS"]["license_country"], 'language': user_data["USER"]["SETTING"]["global"]["language"] }) def change_account(self, child_n): if len(self.childs)-1 < child_n: child_n = 0 self.current_user = self.childs[child_n] self.selected_account = child_n self.http_headers["Accept-Language"] = re.sub(r"[^0-9A-Za-z *,-.;=]", "", self.current_user['language']) return (self.current_user, self.selected_account) def get_track_url(self, track_token, track_format): tracks = self.get_tracks_url([track_token, ], track_format) if len(tracks) > 0: if isinstance(tracks[0], DeezerError): raise tracks[0] else: return tracks[0] return None def get_tracks_url(self, track_tokens, track_format): if not isinstance(track_tokens, list): track_tokens = [track_tokens, ] if not self.current_user.get('license_token'): return [] if (track_format == "FLAC" or track_format.startswidth("MP4_RA")) and not self.current_user.get('can_stream_lossless') or track_format == "MP3_320" and not self.current_user.get('can_stream_hq'): raise WrongLicense(format) result = [] try: request = self.session.post( "https://media.deezer.com/v1/get_url", json={ 'license_token': self.current_user['license_token'], 'media': [{ 'type': "FULL", 'formats': [ { 'cipher': "BF_CBC_STRIPE", 'format': track_format } ] }], 'track_tokens': track_tokens }, headers = self.http_headers ) request.raise_for_status() response = request.json() except requests.exceptions.HTTPError: return [] if len(response.get('data', [])): for data in response['data']: if 'errors' in data: if data['errors'][0]['code'] == 2002: result.append(WrongGeolocation(self.current_user['country'])) else: result.append(DeezerError(json.dumps(response))) if 'media' in data and len(data['media']): result.append(data['media'][0]['sources'][0]['url']) else: result.append(None) return result