yandex-music-api/yandex_music/utils/request.py

132 行
4.6 KiB
Python
Raw Blame 履歴

このファイルには曖昧(ambiguous)なUnicode文字が含まれています

このファイルには、他の文字と見間違える可能性があるUnicode文字が含まれています。 それが意図的なものと考えられる場合は、この警告を無視して構いません。 それらの文字を表示するにはエスケープボタンを使用します。

import re
import json
import logging
import requests
from yandex_music.utils.response import Response
from yandex_music.exceptions import Unauthorized, BadRequest, NetworkError, YandexMusicError
USER_AGENT = 'Yandex-Music-API'
HEADERS = {
'X-Yandex-Music-Client': 'WindowsPhone/3.20',
}
logging.getLogger('urllib3').setLevel(logging.WARNING)
class Request:
"""Вспомогателньный класс для yandex_music предоставляющий методы для выполнения POST и GET запросов, скачивания
файлов.
Args:
client (:obj:`yandex_music.Client`): Объект класса :class:`yandex_music.Client` представляющий клиент Yandex
Music.
headers (:obj:`dict`, optional): Заголовки передаваемые с каждым запросом.
proxy_url (:obj:`str`, optional): Прокси.
"""
def __init__(self,
client=None,
headers=None,
proxy_url=None):
self.headers = headers or HEADERS.copy()
self.client = self.set_and_return_client(client)
self.proxies = {'http': proxy_url, 'https': proxy_url} if proxy_url else None
def set_authorization(self, token):
self.headers.update({'Authorization': f'OAuth {token}'})
def set_and_return_client(self, client):
self.client = client
if self.client and self.client.token:
self.set_authorization(self.client.token)
return self.client
@staticmethod
def _convert_camel_to_snake(text):
s1 = re.sub('(.)([A-Z][a-z]+)', r'\1_\2', text)
return re.sub('([a-z0-9])([A-Z])', r'\1_\2', s1).lower()
@staticmethod
def _object_hook(obj):
cleaned_object = {}
for key, value in obj.items():
key = Request._convert_camel_to_snake(key.replace('-', '_'))
key = key.replace('client', 'client_')
if len(key) and key[0].isdigit():
key = '_' + key
cleaned_object.update({key: value})
return cleaned_object
def _parse(self, json_data) -> Response:
try:
decoded_s = json_data.decode('utf-8')
data = json.loads(decoded_s, object_hook=Request._object_hook)
except UnicodeDecodeError:
logging.getLogger(__name__).debug(
'Logging raw invalid UTF-8 response:\n%r', json_data)
raise YandexMusicError('Server response could not be decoded using UTF-8')
except (AttributeError, ValueError):
raise YandexMusicError('Invalid server response')
return Response.de_json(data, self.client)
def _request_wrapper(self, *args, **kwargs):
if 'headers' not in kwargs:
kwargs['headers'] = {}
kwargs['headers']['User-Agent'] = USER_AGENT
try:
resp = requests.request(*args, **kwargs)
except requests.Timeout:
raise TimeoutError()
except requests.RequestException as e:
raise NetworkError(e)
if 200 <= resp.status_code <= 299:
return resp
message = self._parse(resp.content).error or 'Unknown HTTPError'
if resp.status_code in (401, 403):
raise Unauthorized(message)
elif resp.status_code == 400:
raise BadRequest(message)
elif resp.status_code in (404, 409, 413):
raise NetworkError(message)
elif resp.status_code == 502:
raise NetworkError('Bad Gateway')
else:
raise NetworkError(f'{message} ({resp.status_code})')
def get(self, url, params=None, timeout=5, *args, **kwargs):
result = self._request_wrapper('GET', url, params=params, headers=self.headers, proxies=self.proxies,
timeout=timeout, *args, **kwargs)
return self._parse(result.content).result
def post(self, url, data=None, timeout=5, *args, **kwargs):
result = self._request_wrapper('POST', url, headers=self.headers, proxies=self.proxies, data=data,
timeout=timeout, *args, **kwargs)
return self._parse(result.content).result
def retrieve(self, url, timeout=5, *args, **kwargs):
return self._request_wrapper('GET', url, proxies=self.proxies, timeout=timeout, *args, **kwargs)
def download(self, url, filename, timeout=5, *args, **kwargs):
result = self.retrieve(url, proxies=self.proxies, timeout=timeout, *args, *kwargs)
with open(filename, 'wb') as f:
f.write(result.content)