deezer-py/deezer/__init__.py
2021-12-29 18:26:08 +01:00

200 lines
7.8 KiB
Python

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.4"
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"],
'loved_tracks': child.get("LOVEDTRACKS_ID")
})
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"],
'loved_tracks': user_data["USER"].get("LOVEDTRACKS_ID")
})
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.startswith("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