|
|
@ -1,3 +1,4 @@
|
|
|
|
|
|
|
|
import json
|
|
|
|
import time
|
|
|
|
import time
|
|
|
|
|
|
|
|
|
|
|
|
import backoff
|
|
|
|
import backoff
|
|
|
@ -22,31 +23,40 @@ class Trakt:
|
|
|
|
|
|
|
|
|
|
|
|
def _make_request(self, url, payload={}, authenticate_user=None, request_type='get'):
|
|
|
|
def _make_request(self, url, payload={}, authenticate_user=None, request_type='get'):
|
|
|
|
headers, authenticate_user = self._headers(authenticate_user)
|
|
|
|
headers, authenticate_user = self._headers(authenticate_user)
|
|
|
|
|
|
|
|
headers['User-Agent'] = 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 ' \
|
|
|
|
|
|
|
|
'(KHTML, like Gecko) Chrome/71.0.3578.80 Safari/537.36'
|
|
|
|
|
|
|
|
|
|
|
|
if authenticate_user:
|
|
|
|
if authenticate_user:
|
|
|
|
url = url.replace('{authenticate_user}', authenticate_user)
|
|
|
|
url = url.replace('{authenticate_user}', authenticate_user)
|
|
|
|
|
|
|
|
|
|
|
|
# make request
|
|
|
|
# make request
|
|
|
|
|
|
|
|
resp_data = ''
|
|
|
|
if request_type == 'delete':
|
|
|
|
if request_type == 'delete':
|
|
|
|
req = requests.delete(url, headers=headers, params=payload, timeout=30)
|
|
|
|
with requests.delete(url, headers=headers, params=payload, timeout=30, stream=True) as req:
|
|
|
|
|
|
|
|
for chunk in req.iter_content(chunk_size=250000, decode_unicode=True):
|
|
|
|
|
|
|
|
if chunk:
|
|
|
|
|
|
|
|
resp_data += chunk
|
|
|
|
else:
|
|
|
|
else:
|
|
|
|
req = requests.get(url, headers=headers, params=payload, timeout=30)
|
|
|
|
with requests.get(url, headers=headers, params=payload, timeout=30, stream=True) as req:
|
|
|
|
|
|
|
|
for chunk in req.iter_content(chunk_size=250000, decode_unicode=True):
|
|
|
|
|
|
|
|
if chunk:
|
|
|
|
|
|
|
|
resp_data += chunk
|
|
|
|
|
|
|
|
|
|
|
|
log.debug("Request URL: %s", req.url)
|
|
|
|
log.debug("Request URL: %s", req.url)
|
|
|
|
log.debug("Request Payload: %s", payload)
|
|
|
|
log.debug("Request Payload: %s", payload)
|
|
|
|
log.debug("Request User: %s", authenticate_user)
|
|
|
|
log.debug("Request User: %s", authenticate_user)
|
|
|
|
log.debug("Response Code: %d", req.status_code)
|
|
|
|
log.debug("Response Code: %d", req.status_code)
|
|
|
|
|
|
|
|
return req, resp_data
|
|
|
|
return req
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@backoff.on_predicate(backoff.expo, lambda x: x is None, max_tries=4, on_backoff=backoff_handler)
|
|
|
|
@backoff.on_predicate(backoff.expo, lambda x: x is None, max_tries=4, on_backoff=backoff_handler)
|
|
|
|
def _make_item_request(self, url, object_name, payload={}):
|
|
|
|
def _make_item_request(self, url, object_name, payload={}):
|
|
|
|
payload = dict_merge(payload, {'extended': 'full'})
|
|
|
|
payload = dict_merge(payload, {'extended': 'full'})
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
try:
|
|
|
|
req = self._make_request(url, payload)
|
|
|
|
req, resp_data = self._make_request(url, payload)
|
|
|
|
|
|
|
|
|
|
|
|
if req.status_code == 200:
|
|
|
|
if req.status_code == 200 and len(resp_data):
|
|
|
|
resp_json = req.json()
|
|
|
|
resp_json = json.loads(resp_data)
|
|
|
|
return resp_json
|
|
|
|
return resp_json
|
|
|
|
elif req.status_code == 401:
|
|
|
|
elif req.status_code == 401:
|
|
|
|
log.error("The authentication to Trakt is revoked. Please re-authenticate.")
|
|
|
|
log.error("The authentication to Trakt is revoked. Please re-authenticate.")
|
|
|
@ -58,7 +68,7 @@ class Trakt:
|
|
|
|
log.exception("Exception retrieving %s: ", object_name)
|
|
|
|
log.exception("Exception retrieving %s: ", object_name)
|
|
|
|
return None
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
@backoff.on_predicate(backoff.expo, lambda x: x is None, max_tries=4, on_backoff=backoff_handler)
|
|
|
|
@backoff.on_predicate(backoff.expo, lambda x: x is None, max_tries=6, on_backoff=backoff_handler)
|
|
|
|
def _make_items_request(self, url, limit, languages, type_name, object_name, authenticate_user=None, payload={},
|
|
|
|
def _make_items_request(self, url, limit, languages, type_name, object_name, authenticate_user=None, payload={},
|
|
|
|
sleep_between=5, genres=None):
|
|
|
|
sleep_between=5, genres=None):
|
|
|
|
if not languages:
|
|
|
|
if not languages:
|
|
|
@ -74,8 +84,34 @@ class Trakt:
|
|
|
|
type_name = type_name.replace('{authenticate_user}', self._user_used_for_authentication(authenticate_user))
|
|
|
|
type_name = type_name.replace('{authenticate_user}', self._user_used_for_authentication(authenticate_user))
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
try:
|
|
|
|
|
|
|
|
resp_data = ''
|
|
|
|
while True:
|
|
|
|
while True:
|
|
|
|
req = self._make_request(url, payload, authenticate_user)
|
|
|
|
attempts = 0
|
|
|
|
|
|
|
|
max_attempts = 6
|
|
|
|
|
|
|
|
retrieve_error = False
|
|
|
|
|
|
|
|
while attempts <= max_attempts:
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
|
|
req, resp_data = self._make_request(url, payload, authenticate_user)
|
|
|
|
|
|
|
|
if resp_data is not None:
|
|
|
|
|
|
|
|
retrieve_error = False
|
|
|
|
|
|
|
|
break
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
log.warning("Failed to retrieve valid response for Trakt %s %s from _make_item_request",
|
|
|
|
|
|
|
|
type_name, object_name)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
except Exception:
|
|
|
|
|
|
|
|
log.exception("Exception retrieving %s %s in _make_item_request: ", type_name, object_name)
|
|
|
|
|
|
|
|
retrieve_error = True
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
attempts += 1
|
|
|
|
|
|
|
|
log.info("Sleeping for %d seconds before making attempt %d/%d", 3 * attempts, attempts + 1,
|
|
|
|
|
|
|
|
max_attempts)
|
|
|
|
|
|
|
|
time.sleep(3 * attempts)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if retrieve_error or not resp_data or not len(resp_data):
|
|
|
|
|
|
|
|
log.error("Failed retrieving %s %s from _make_item_request %d times, aborting...", type_name,
|
|
|
|
|
|
|
|
object_name, attempts)
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
current_page = payload['page']
|
|
|
|
current_page = payload['page']
|
|
|
|
total_pages = 0 if 'X-Pagination-Page-Count' not in req.headers else int(
|
|
|
|
total_pages = 0 if 'X-Pagination-Page-Count' not in req.headers else int(
|
|
|
@ -83,23 +119,28 @@ class Trakt:
|
|
|
|
|
|
|
|
|
|
|
|
log.debug("Response Page: %d of %d", current_page, total_pages)
|
|
|
|
log.debug("Response Page: %d of %d", current_page, total_pages)
|
|
|
|
|
|
|
|
|
|
|
|
if req.status_code == 200:
|
|
|
|
if req.status_code == 200 and len(resp_data):
|
|
|
|
resp_json = req.json()
|
|
|
|
if resp_data.startswith("[{") and resp_data.endswith("}]"):
|
|
|
|
if type_name == 'person' and 'cast' in resp_json:
|
|
|
|
resp_json = json.loads(resp_data)
|
|
|
|
# handle person results
|
|
|
|
|
|
|
|
for item in resp_json['cast']:
|
|
|
|
if type_name == 'person' and 'cast' in resp_json:
|
|
|
|
if item not in processed:
|
|
|
|
# handle person results
|
|
|
|
if object_name.rstrip('s') not in item and 'title' in item:
|
|
|
|
for item in resp_json['cast']:
|
|
|
|
processed.append({object_name.rstrip('s'): item})
|
|
|
|
if item not in processed:
|
|
|
|
else:
|
|
|
|
if object_name.rstrip('s') not in item and 'title' in item:
|
|
|
|
processed.append(item)
|
|
|
|
processed.append({object_name.rstrip('s'): item})
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
processed.append(item)
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
for item in resp_json:
|
|
|
|
|
|
|
|
if item not in processed:
|
|
|
|
|
|
|
|
if object_name.rstrip('s') not in item and 'title' in item:
|
|
|
|
|
|
|
|
processed.append({object_name.rstrip('s'): item})
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
processed.append(item)
|
|
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
else:
|
|
|
|
for item in resp_json:
|
|
|
|
log.warning("Receiving malformed JSON response for page: %d of %d", current_page, total_pages)
|
|
|
|
if item not in processed:
|
|
|
|
|
|
|
|
if object_name.rstrip('s') not in item and 'title' in item:
|
|
|
|
|
|
|
|
processed.append({object_name.rstrip('s'): item})
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
processed.append(item)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# check if we have fetched the last page, break if so
|
|
|
|
# check if we have fetched the last page, break if so
|
|
|
|
if total_pages == 0:
|
|
|
|
if total_pages == 0:
|
|
|
@ -112,6 +153,7 @@ class Trakt:
|
|
|
|
log.info("There are %d pages left to retrieve results from", total_pages - current_page)
|
|
|
|
log.info("There are %d pages left to retrieve results from", total_pages - current_page)
|
|
|
|
payload['page'] += 1
|
|
|
|
payload['page'] += 1
|
|
|
|
time.sleep(sleep_between)
|
|
|
|
time.sleep(sleep_between)
|
|
|
|
|
|
|
|
|
|
|
|
elif req.status_code == 401:
|
|
|
|
elif req.status_code == 401:
|
|
|
|
log.error("The authentication to Trakt is revoked. Please re-authenticate.")
|
|
|
|
log.error("The authentication to Trakt is revoked. Please re-authenticate.")
|
|
|
|
exit()
|
|
|
|
exit()
|
|
|
@ -122,6 +164,7 @@ class Trakt:
|
|
|
|
if len(processed):
|
|
|
|
if len(processed):
|
|
|
|
log.debug("Found %d %s %s", len(processed), type_name, object_name)
|
|
|
|
log.debug("Found %d %s %s", len(processed), type_name, object_name)
|
|
|
|
return processed
|
|
|
|
return processed
|
|
|
|
|
|
|
|
|
|
|
|
return None
|
|
|
|
return None
|
|
|
|
except Exception:
|
|
|
|
except Exception:
|
|
|
|
log.exception("Exception retrieving %s %s: ", type_name, object_name)
|
|
|
|
log.exception("Exception retrieving %s %s: ", type_name, object_name)
|
|
|
@ -130,7 +173,7 @@ class Trakt:
|
|
|
|
def validate_client_id(self):
|
|
|
|
def validate_client_id(self):
|
|
|
|
try:
|
|
|
|
try:
|
|
|
|
# request anticipated shows to validate client_id
|
|
|
|
# request anticipated shows to validate client_id
|
|
|
|
req = self._make_request(
|
|
|
|
req, req_data = self._make_request(
|
|
|
|
url='https://api.trakt.tv/shows/anticipated',
|
|
|
|
url='https://api.trakt.tv/shows/anticipated',
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
@ -142,7 +185,7 @@ class Trakt:
|
|
|
|
return False
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
def remove_recommended_item(self, item_type, trakt_id, authenticate_user=None):
|
|
|
|
def remove_recommended_item(self, item_type, trakt_id, authenticate_user=None):
|
|
|
|
ret = self._make_request(
|
|
|
|
ret, ret_data = self._make_request(
|
|
|
|
url='https://api.trakt.tv/recommendations/%ss/%s' % (item_type, str(trakt_id)),
|
|
|
|
url='https://api.trakt.tv/recommendations/%ss/%s' % (item_type, str(trakt_id)),
|
|
|
|
authenticate_user=authenticate_user,
|
|
|
|
authenticate_user=authenticate_user,
|
|
|
|
request_type='delete'
|
|
|
|
request_type='delete'
|
|
|
|