mirror of
https://github.com/SickGear/SickGear.git
synced 2025-01-23 01:43:43 +00:00
806 lines
38 KiB
Python
806 lines
38 KiB
Python
|
# coding=utf-8
|
||
|
import threading
|
||
|
import warnings
|
||
|
warnings.filterwarnings('ignore', module=r'.*fuz.*', message='.*Sequence.*')
|
||
|
warnings.filterwarnings('ignore', module=r'.*connectionpool.*', message='.*certificate verification.*')
|
||
|
|
||
|
import test_lib as test
|
||
|
|
||
|
from datetime import date as org_date, datetime as org_datetime
|
||
|
import datetime
|
||
|
import gc
|
||
|
import hashlib
|
||
|
import lzma
|
||
|
import pickle
|
||
|
import sys
|
||
|
import unittest
|
||
|
import os.path
|
||
|
import shutil
|
||
|
sys.path.insert(1, os.path.abspath('..'))
|
||
|
|
||
|
import sickgear
|
||
|
from sickgear.indexers.indexer_config import TVINFO_TVDB, TVINFO_TMDB, TVINFO_TVMAZE, TVINFO_TRAKT, TVINFO_IMDB, \
|
||
|
TVINFO_TRAKT_SLUG
|
||
|
from lib.tvinfo_base import TVInfoPerson as TVInfoPerson_lib, TVInfoImage as TVInfoImage_lib, \
|
||
|
TVInfoSocialIDs as TVInfoSocialIDs_lib, TVInfoCharacter as TVInfoCharacter_lib, TVInfoShow as TVInfoShow_lib, \
|
||
|
TVInfoIDs as TVInfoIDs_lib, CastList as CastList_lib, CrewList as CrewList_lib, \
|
||
|
TVInfoEpisode as TVInfoEpisode_lib, TVInfoSeason as TVInfoSeason_lib, \
|
||
|
TVInfoNetwork as TVInfoNetwork_lib
|
||
|
from tvinfo_base import TVInfoPerson, TVInfoImage, TVInfoSocialIDs, TVInfoCharacter, TVInfoShow, TVInfoIDs, CastList, \
|
||
|
CrewList, RoleTypes, TVInfoEpisode, TVInfoSeason, TVInfoNetwork
|
||
|
import requests
|
||
|
from lib.api_tvdb.tvdb_api import Tvdb
|
||
|
|
||
|
# noinspection PyUnreachableCode
|
||
|
if False:
|
||
|
from typing import Any, List, Optional
|
||
|
|
||
|
NoneType = type(None)
|
||
|
threading_Lock = (type(threading.Lock()), type(threading.RLock()))
|
||
|
file_dir = os.path.dirname(__file__)
|
||
|
sickgear.CACHE_DIR = os.path.join(file_dir, 'cache')
|
||
|
os.makedirs(sickgear.CACHE_DIR, exist_ok=True)
|
||
|
|
||
|
test_base_name = os.path.splitext(os.path.basename(__file__))[0]
|
||
|
mock_data_dir = os.path.join(file_dir, 'mock_data', test_base_name)
|
||
|
os.makedirs(mock_data_dir, exist_ok=True)
|
||
|
cast_types = [t for t in RoleTypes.reverse.keys() if t < RoleTypes.crew_limit]
|
||
|
crew_types = [t for t in RoleTypes.reverse.keys() if t >= RoleTypes.crew_limit]
|
||
|
|
||
|
# settings for mock file creation
|
||
|
disable_content_creation = True
|
||
|
only_new_urls_data_creation = True
|
||
|
delete_unused_mock_files = False
|
||
|
|
||
|
# other settings
|
||
|
pickle_protocol = 3 # needed for python 3.7 compatibility
|
||
|
used_files = {'browse_start_date.data'}
|
||
|
|
||
|
|
||
|
def _make_filename(file_str, file_ext, extra_strs=None):
|
||
|
# type: (str, str, Optional[List[str]]) -> str
|
||
|
hash_256 = hashlib.sha256()
|
||
|
hash_256.update(file_str.encode('utf-8'))
|
||
|
if isinstance(extra_strs, list):
|
||
|
for _e in extra_strs:
|
||
|
hash_256.update(_e.encode('utf-8'))
|
||
|
return '%s%s' % (hash_256.hexdigest(), file_ext)
|
||
|
|
||
|
|
||
|
def _mock_get(*args, **kwargs):
|
||
|
url = (1 < len(args) and args[1]) or kwargs.get('url')
|
||
|
resp = requests.Response()
|
||
|
resp.url = url
|
||
|
resp.status_code = 200
|
||
|
resp._content = ''
|
||
|
resp.encoding = 'UTF-8'
|
||
|
resp.headers.update({'Content-Type': 'application/json; charset=UTF-8'})
|
||
|
filename = _make_filename(
|
||
|
url, '.data', extra_strs=['%s%s' % (_k, _v) for _k, _v in (args[0].params or {}).items()
|
||
|
if _k not in ('api_key')])
|
||
|
data_file = os.path.join(mock_data_dir, filename)
|
||
|
used_files.add(filename)
|
||
|
if (disable_content_creation or only_new_urls_data_creation) and os.path.isfile(data_file):
|
||
|
with lzma.open(data_file, 'rb') as fd:
|
||
|
resp._content = fd.read()
|
||
|
return resp
|
||
|
elif not disable_content_creation:
|
||
|
kw = kwargs.copy()
|
||
|
kw.update({'params': args[0].params, 'headers': args[0].headers})
|
||
|
resp = requests.get(*args[1:], **kw)
|
||
|
with lzma.open(data_file, 'wb') as fd:
|
||
|
fd.write(resp.content)
|
||
|
else:
|
||
|
print('error getting: %s' % url)
|
||
|
return resp
|
||
|
|
||
|
|
||
|
def _mock_post(*args, **kwargs):
|
||
|
url = (1 < len(args) and args[1]) or kwargs.get('url')
|
||
|
resp = requests.Response()
|
||
|
resp.status_code = 200
|
||
|
resp._content = ''
|
||
|
resp.encoding = 'UTF-8'
|
||
|
resp.headers['Content-Type'] = 'text/html; charset=utf-8'
|
||
|
return resp
|
||
|
|
||
|
|
||
|
def _mock_get_new_token(*args, **kwargs):
|
||
|
return {'token': 'testtoken', 'datetime': datetime.datetime.now()}
|
||
|
|
||
|
|
||
|
browse_start_date_filename = os.path.join(mock_data_dir, 'browse_start_date.data')
|
||
|
if disable_content_creation:
|
||
|
with open(browse_start_date_filename, 'rt', encoding='UTF-8') as f:
|
||
|
browse_start_date = f.read()
|
||
|
else:
|
||
|
with open(browse_start_date_filename, 'wt', encoding='UTF-8') as f:
|
||
|
browse_start_date = org_datetime.now().strftime('%Y-%m-%d')
|
||
|
f.write(browse_start_date)
|
||
|
|
||
|
|
||
|
class _FakeDate(datetime.date):
|
||
|
@classmethod
|
||
|
def today(cls):
|
||
|
d = org_date.fromisoformat(browse_start_date)
|
||
|
return cls(d.year, d.month, d.day)
|
||
|
|
||
|
|
||
|
class _FakeDateTime(datetime.datetime):
|
||
|
@classmethod
|
||
|
def now(cls, *args, **kwargs):
|
||
|
d = org_datetime.fromisoformat(browse_start_date)
|
||
|
return cls(d.year, d.month, d.day, d.hour, d.minute, d.second)
|
||
|
|
||
|
|
||
|
datetime_date_backup = datetime.date
|
||
|
datetime_datetime_backup = datetime.datetime
|
||
|
|
||
|
|
||
|
def _save_pickle_file(f_name, save_obj):
|
||
|
# type: (str, object) -> None
|
||
|
used_files.add(f_name)
|
||
|
full_filename = os.path.join(mock_data_dir, f_name)
|
||
|
datetime.date = datetime_date_backup
|
||
|
datetime.datetime = datetime_datetime_backup
|
||
|
try:
|
||
|
with lzma.open(full_filename, 'wb') as f:
|
||
|
pickle.dump(save_obj, f, protocol=pickle_protocol)
|
||
|
finally:
|
||
|
datetime.date = _FakeDate
|
||
|
datetime.datetime = _FakeDateTime
|
||
|
|
||
|
|
||
|
def _load_pickle_file(f_name):
|
||
|
# type: (str) -> Any
|
||
|
used_files.add(f_name)
|
||
|
full_filename = os.path.join(mock_data_dir, f_name)
|
||
|
datetime.date = datetime_date_backup
|
||
|
datetime.datetime = datetime_datetime_backup
|
||
|
try:
|
||
|
with lzma.open(full_filename, 'rb') as f:
|
||
|
return pickle.load(f)
|
||
|
finally:
|
||
|
datetime.date = _FakeDate
|
||
|
datetime.datetime = _FakeDateTime
|
||
|
|
||
|
|
||
|
backup_session_get = requests.sessions.Session.get
|
||
|
backup_session_post = requests.sessions.Session.post
|
||
|
|
||
|
|
||
|
def _check_types(obj, check_list):
|
||
|
# type: (object, List) -> None
|
||
|
for _p, _pt, _st in check_list:
|
||
|
if not isinstance(getattr(obj, _p), _pt):
|
||
|
raise AssertionError('property: [%s] as unexpected type [%s), expected [%s]' %
|
||
|
(_p, type(getattr(obj, _p)), _pt))
|
||
|
if None is not _st:
|
||
|
if isinstance(getattr(obj, _p), list):
|
||
|
for _o in getattr(obj, _p):
|
||
|
if not isinstance(_o, _st):
|
||
|
raise AssertionError('property: [%s] as unexpected type [%s), expected [%s]' %
|
||
|
(_p, type(_o), _st))
|
||
|
elif isinstance(getattr(obj, _p), dict):
|
||
|
for _k, _v in getattr(obj, _p).items():
|
||
|
if not isinstance(_k, _st[0]):
|
||
|
raise AssertionError('property key: [%s] as unexpected type [%s), expected [%s]' %
|
||
|
(_p, type(_k), _st[0]))
|
||
|
if not isinstance(_v, _st[1]):
|
||
|
raise AssertionError('property key: [%s] as unexpected type [%s), expected [%s]' %
|
||
|
(_p, type(_v), _st[0]))
|
||
|
|
||
|
|
||
|
def _property_type_checker(obj, checked_objs=None):
|
||
|
# type: (object, bool) -> None
|
||
|
|
||
|
# make aure each object is only checked once
|
||
|
checked_objs = checked_objs or []
|
||
|
obj_id = id(obj)
|
||
|
if obj_id in checked_objs:
|
||
|
return
|
||
|
checked_objs.append(obj_id)
|
||
|
|
||
|
# make sure type check uses original types
|
||
|
datetime.date = datetime_date_backup
|
||
|
datetime.datetime = datetime_datetime_backup
|
||
|
|
||
|
try:
|
||
|
# special checks for TVInfoPerson property types
|
||
|
if isinstance(obj, (TVInfoPerson, TVInfoPerson_lib)):
|
||
|
_check_types(obj, [
|
||
|
('images', list, (TVInfoImage, TVInfoImage_lib)),
|
||
|
('social_ids', (TVInfoSocialIDs, TVInfoSocialIDs_lib), None),
|
||
|
('characters', list, (TVInfoCharacter, TVInfoCharacter_lib)),
|
||
|
('name', str, None),
|
||
|
('id', (int, NoneType), None),
|
||
|
('image', (str, NoneType), None),
|
||
|
('thumb_url', (str, NoneType), None),
|
||
|
('gender', (int, NoneType), None),
|
||
|
('bio', (str, NoneType), None),
|
||
|
('birthdate', (datetime.date, NoneType), None),
|
||
|
('deathdate', (datetime.date, NoneType), None),
|
||
|
('country', (str, NoneType), None),
|
||
|
('country_code', (str, NoneType), None),
|
||
|
('country_timezone', (str, NoneType), None),
|
||
|
('ids', (TVInfoIDs, TVInfoIDs_lib), None),
|
||
|
('social_ids', (TVInfoSocialIDs, TVInfoSocialIDs_lib), None),
|
||
|
('birthplace', (str, NoneType), None),
|
||
|
('deathplace', (str, NoneType), None),
|
||
|
('url', (str, NoneType), None),
|
||
|
('height', (int, float, NoneType), None),
|
||
|
('nicknames', set, str),
|
||
|
('real_name', (str, NoneType), None),
|
||
|
('akas', set, str),
|
||
|
])
|
||
|
for _o in obj.characters:
|
||
|
_property_type_checker(_o, checked_objs=checked_objs)
|
||
|
# special checks for TVInfoCharacter property types
|
||
|
elif isinstance(obj, (TVInfoCharacter, TVInfoCharacter_lib)):
|
||
|
_check_types(obj, [
|
||
|
('person', (list, NoneType), (TVInfoPerson, TVInfoPerson_lib)),
|
||
|
('ti_show', (TVInfoShow, TVInfoShow_lib, NoneType), None),
|
||
|
('voice', (bool, NoneType), None),
|
||
|
('plays_self', (bool, NoneType), None),
|
||
|
('regular', (bool, NoneType), None),
|
||
|
('start_year', (int, NoneType), None),
|
||
|
('end_year', (int, NoneType), None),
|
||
|
('name', (str, NoneType), None),
|
||
|
('episode_count', (int, NoneType), None),
|
||
|
('guest_episodes_numbers', dict, [int, list]),
|
||
|
('name', (str, NoneType), None),
|
||
|
('id', (int, NoneType), None),
|
||
|
('image', (str, NoneType), None),
|
||
|
('thumb_url', (str, NoneType), None),
|
||
|
('gender', (int, NoneType), None),
|
||
|
('bio', (str, NoneType), None),
|
||
|
('birthdate', (datetime.date, NoneType), None),
|
||
|
('deathdate', (datetime.date, NoneType), None),
|
||
|
('country', (str, NoneType), None),
|
||
|
('country_code', (str, NoneType), None),
|
||
|
('country_timezone', (str, NoneType), None),
|
||
|
('ids', (TVInfoIDs, TVInfoIDs_lib), None),
|
||
|
])
|
||
|
if isinstance(obj.person, list):
|
||
|
for _p in obj.person:
|
||
|
if not isinstance(_p, (TVInfoPerson, TVInfoPerson_lib)):
|
||
|
raise AssertionError('invalid person object in character object [%s]' % obj)
|
||
|
_property_type_checker(_p, checked_objs=checked_objs)
|
||
|
|
||
|
# for some api the show can't directly be loaded for character credits
|
||
|
if isinstance(obj.ti_show, (TVInfoShow, TVInfoShow_lib)) and obj.ti_show.show_loaded:
|
||
|
_property_type_checker(obj.ti_show, checked_objs=checked_objs)
|
||
|
# special checks for TVInfoShow property types
|
||
|
elif isinstance(obj, (TVInfoShow, TVInfoShow_lib)):
|
||
|
_check_types(obj, [
|
||
|
('id', int, None),
|
||
|
('seriesname', str, None),
|
||
|
('poster_loaded', bool, None),
|
||
|
('banner_loaded', bool, None),
|
||
|
('fanart_loaded', bool, None),
|
||
|
('season_images_loaded', bool, None),
|
||
|
('seasonwide_images_loaded', bool, None),
|
||
|
('actors_loaded', bool, None),
|
||
|
('show_not_found', bool, None),
|
||
|
('ids', (TVInfoIDs, TVInfoIDs_lib), None),
|
||
|
('social_ids', (TVInfoSocialIDs, TVInfoSocialIDs_lib), None),
|
||
|
('slug', (str, NoneType), None),
|
||
|
('seriesid', int, None),
|
||
|
('aliases', list, str),
|
||
|
('classification', (str, NoneType), None),
|
||
|
('genre', (str, NoneType), None),
|
||
|
('genre', (str, NoneType), None),
|
||
|
('genre_list', list, str),
|
||
|
('actors', list, dict),
|
||
|
('cast', (CastList, CastList_lib), None),
|
||
|
('crew', (CrewList, CrewList_lib), None),
|
||
|
('show_type', list, str),
|
||
|
('networks', list, (TVInfoNetwork, TVInfoNetwork_lib)),
|
||
|
('network', (str, NoneType), None),
|
||
|
('network_id', (int, NoneType), None),
|
||
|
('network_country', (str, NoneType), None),
|
||
|
('network_country_code', (str, NoneType), None),
|
||
|
('network_is_stream', (bool, NoneType), None),
|
||
|
('runtime', (int, NoneType), None),
|
||
|
('language', (str, NoneType), None),
|
||
|
('spoken_languages', list, str),
|
||
|
('official_site', (str, NoneType), None),
|
||
|
('imdb_id', (str, NoneType), None),
|
||
|
('zap2itid', (str, NoneType), None),
|
||
|
('airs_dayofweek', (str, NoneType), None),
|
||
|
('airs_time', (str, NoneType), None),
|
||
|
('time', (datetime.time, NoneType), None),
|
||
|
('added', (str, NoneType), None),
|
||
|
('addedby', (str, NoneType), None),
|
||
|
('siteratingcount', (int, NoneType), None),
|
||
|
('lastupdated', (int, str, NoneType), None),
|
||
|
('contentrating', (str, NoneType), None),
|
||
|
('rating', (int, float, NoneType), None),
|
||
|
('status', (str, NoneType), None),
|
||
|
('overview', str, None),
|
||
|
('poster', (str, NoneType), None),
|
||
|
('poster_thumb', (str, NoneType), None),
|
||
|
('banner', (str, NoneType), None),
|
||
|
('banner_thumb', (str, NoneType), None),
|
||
|
('fanart', (str, NoneType), None),
|
||
|
('banners', dict, None),
|
||
|
('images', dict, [int, list]),
|
||
|
('updated_timestamp', (int, NoneType), None),
|
||
|
('popularity', (int, float, NoneType), None),
|
||
|
('vote_count', (int, NoneType), None),
|
||
|
('vote_average', (int, float, NoneType), None),
|
||
|
('origin_countries', list, str),
|
||
|
('alt_ep_numbering', dict, [str, dict]),
|
||
|
('watcher_count', (int, NoneType), None),
|
||
|
('play_count', (int, NoneType), None),
|
||
|
('collected_count', (int, NoneType), None),
|
||
|
('collector_count', (int, NoneType), None),
|
||
|
('next_season_airdate', (str, NoneType), None),
|
||
|
('trailers', dict, [str, str]),
|
||
|
('requested_language', str, None),
|
||
|
])
|
||
|
for _ct, _cl in obj.cast.items():
|
||
|
if _ct not in cast_types:
|
||
|
raise AssertionError('invalid CastType in show object')
|
||
|
for _c in _cl:
|
||
|
_property_type_checker(_c, checked_objs=checked_objs)
|
||
|
|
||
|
for _ct, _cl in obj.crew.items():
|
||
|
if _ct not in crew_types:
|
||
|
raise AssertionError('invalid CrewType in show object')
|
||
|
for _c in _cl:
|
||
|
_property_type_checker(_c, checked_objs=checked_objs)
|
||
|
|
||
|
# iter though episode objects and test them
|
||
|
for _s, _sl in obj.items():
|
||
|
if not isinstance(_s, int):
|
||
|
raise AssertionError('invalid season number in show object [%s]' % _s)
|
||
|
if not isinstance(_sl, (TVInfoSeason, TVInfoSeason_lib)):
|
||
|
raise AssertionError('invalid season object in show object [%s]' % _s)
|
||
|
_property_type_checker(_sl, checked_objs=checked_objs)
|
||
|
# special checks for TVInfoSeason property types
|
||
|
elif isinstance(obj, (TVInfoSeason, TVInfoSeason_lib)):
|
||
|
_check_types(obj, [
|
||
|
('id', (int, NoneType), None),
|
||
|
('show', (TVInfoShow, TVInfoShow_lib), None),
|
||
|
('number', int, None),
|
||
|
('name', (str, NoneType), None),
|
||
|
('actors', list, None),
|
||
|
('cast', (CastList, CastList_lib), None),
|
||
|
('network', (str, NoneType), None),
|
||
|
('network_id', (int, NoneType), None),
|
||
|
('network_timezone', (str, NoneType), None),
|
||
|
('network_country', (str, NoneType), None),
|
||
|
('network_country_code', (str, NoneType), None),
|
||
|
('network_is_stream', (bool, NoneType), None),
|
||
|
('ordered', (int, NoneType), None),
|
||
|
('start_date', (str, NoneType), None),
|
||
|
('end_date', (str, NoneType), None),
|
||
|
('poster', (str, NoneType), None),
|
||
|
('summery', (str, NoneType), None),
|
||
|
('episode_order', (int, NoneType), None),
|
||
|
])
|
||
|
|
||
|
for _ct, _cl in obj.cast.items():
|
||
|
if _ct not in cast_types:
|
||
|
raise AssertionError('invalid CastType in season object')
|
||
|
for _c in _cl:
|
||
|
_property_type_checker(_c, checked_objs=checked_objs)
|
||
|
|
||
|
for _en, _eo in obj.items():
|
||
|
if not isinstance(_en, int):
|
||
|
raise AssertionError('invalid episode number in show object [%s]' % _en)
|
||
|
if not isinstance(_eo, (TVInfoEpisode, TVInfoEpisode_lib)):
|
||
|
raise AssertionError('invalid episdode object in season object [%s]' % _en)
|
||
|
_property_type_checker(_eo, checked_objs=checked_objs)
|
||
|
# special checks for TVInfoEpisode property types
|
||
|
elif isinstance(obj, (TVInfoEpisode, TVInfoEpisode_lib)):
|
||
|
_check_types(obj, [
|
||
|
('id', int, None),
|
||
|
('episodename', (str, NoneType), None),
|
||
|
('seriesid', (int, NoneType), None),
|
||
|
('season', (TVInfoSeason, TVInfoSeason_lib), None),
|
||
|
('seasonnumber', int, None),
|
||
|
('episodenumber', int, None),
|
||
|
('absolute_number', (int, NoneType), None),
|
||
|
('is_special', (bool, NoneType), None),
|
||
|
('actors', list, dict),
|
||
|
('gueststars', (str, NoneType), None),
|
||
|
('gueststars_list', list, str),
|
||
|
('cast', (CastList, CastList_lib), None),
|
||
|
('directors', list, str),
|
||
|
('writer', (str, NoneType), None),
|
||
|
('writers', list, str),
|
||
|
('crew', (CrewList, CrewList_lib), None),
|
||
|
('overview', str, None),
|
||
|
('productioncode', (str, NoneType), None),
|
||
|
('showurl', (str, NoneType), None),
|
||
|
('lastupdated', (int, str, NoneType), None),
|
||
|
('dvddiscid', (str, NoneType), None),
|
||
|
('dvd_season', (int, NoneType), None),
|
||
|
('dvd_episodenumber', (int, NoneType), None),
|
||
|
('dvdchapter', (int, NoneType), None),
|
||
|
('firstaired', (str, NoneType), None),
|
||
|
('airtime', (datetime.time, NoneType), None),
|
||
|
('runtime', (int, NoneType), None),
|
||
|
('timestamp', (int, float, NoneType), None),
|
||
|
('network', (str, NoneType), None),
|
||
|
('network_id', (int, NoneType), None),
|
||
|
('network_timezone', (str, NoneType), None),
|
||
|
('network_country', (str, NoneType), None),
|
||
|
('network_country_code', (str, NoneType), None),
|
||
|
('network_is_stream', (bool, NoneType), None),
|
||
|
('filename', (str, NoneType), None),
|
||
|
('lastupdatedby', (int, str, NoneType), None),
|
||
|
('airsafterseason', (int, NoneType), None),
|
||
|
('airsbeforeseason', (int, NoneType), None),
|
||
|
('airsbeforeepisode', (int, NoneType), None),
|
||
|
('imdb_id', (str, NoneType), None),
|
||
|
('contentrating', (str, NoneType), None),
|
||
|
('thumbadded', (str, NoneType), None),
|
||
|
('rating', (int, float, NoneType), None),
|
||
|
('vote_count', (int, NoneType), None),
|
||
|
('siteratingcount', (int, NoneType), None),
|
||
|
('show', (TVInfoShow, TVInfoShow_lib, NoneType), None),
|
||
|
('alt_nums', dict, [str, dict]),
|
||
|
('finale_type', (int, NoneType), None),
|
||
|
])
|
||
|
if isinstance(obj.show, (TVInfoShow, TVInfoShow_lib)):
|
||
|
_property_type_checker(obj.show, checked_objs=checked_objs)
|
||
|
|
||
|
_property_type_checker(obj.season, checked_objs=checked_objs)
|
||
|
|
||
|
for _ct, _cl in obj.cast.items():
|
||
|
if _ct not in cast_types:
|
||
|
raise AssertionError('invalid CastType in episode object')
|
||
|
for _c in _cl:
|
||
|
_property_type_checker(_c, checked_objs=checked_objs)
|
||
|
|
||
|
for _ct, _cl in obj.crew.items():
|
||
|
if _ct not in crew_types:
|
||
|
raise AssertionError('invalid CrewType in episode object')
|
||
|
for _c in _cl:
|
||
|
_property_type_checker(_c, checked_objs=checked_objs)
|
||
|
finally:
|
||
|
datetime.date = _FakeDate
|
||
|
datetime.datetime = _FakeDateTime
|
||
|
|
||
|
|
||
|
def _compare_helper(obj_a, obj_b):
|
||
|
_property_type_checker(obj_a)
|
||
|
|
||
|
for prop, value in vars(obj_a).items():
|
||
|
if isinstance(value, threading_Lock):
|
||
|
continue
|
||
|
try:
|
||
|
assert value == getattr(obj_b, prop)
|
||
|
except AssertionError as e:
|
||
|
print('property [%s] different: obj_a [%s], obj_b [%s)' % (prop, value, getattr(obj_b, prop)))
|
||
|
raise e
|
||
|
except (BaseException, Exception) as e:
|
||
|
print(e)
|
||
|
raise e
|
||
|
|
||
|
|
||
|
person_tests = [
|
||
|
# {'p_id': 346941, 'tvid': TVINFO_TVDB}, # Katherine McNamara
|
||
|
{'p_id': 968006, 'tvid': TVINFO_TMDB}, # Katherine McNamara
|
||
|
{'p_id': 15776, 'tvid': TVINFO_TVMAZE}, # Katherine McNamara
|
||
|
{'p_id': 260345, 'tvid': TVINFO_TRAKT}, # Katherine McNamara
|
||
|
{'p_id': 3031063, 'tvid': TVINFO_IMDB}, # Katherine McNamara
|
||
|
]
|
||
|
|
||
|
show_tests = [
|
||
|
{'prodid': 295837, 'tvid': TVINFO_TVDB}, # Shadowhunters
|
||
|
{'prodid': 63210, 'tvid': TVINFO_TMDB}, # Shadowhunters
|
||
|
{'prodid': 2158, 'tvid': TVINFO_TVMAZE}, # Shadowhunters
|
||
|
]
|
||
|
|
||
|
search_tests = [
|
||
|
# tvdb tests
|
||
|
{'kwargs': {'name': 'Shadowhunters'}, 'search_tvid': TVINFO_TVDB}, # Shadowhunters
|
||
|
{'kwargs': {'name': 'Wednesday'}, 'search_tvid': TVINFO_TVDB}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_TVDB: 295837}}, 'search_tvid': TVINFO_TVDB}, # Shadowhunters
|
||
|
# {'kwargs': {'ids': {TVINFO_IMDB: 4145054}}, 'search_tvid': TVINFO_TVDB}, # Shadowhunters
|
||
|
# {'kwargs': {'ids': {TVINFO_TMDB: 119051}}, 'search_tvid': TVINFO_TVDB}, # Wednesday
|
||
|
# {'kwargs': {'ids': {TVINFO_TVMAZE: 53647}}, 'search_tvid': TVINFO_TVDB}, # Wednesday
|
||
|
# trakt tests
|
||
|
{'kwargs': {'name': 'Shadowhunters'}, 'search_tvid': TVINFO_TRAKT}, # Shadowhunters
|
||
|
{'kwargs': {'name': 'Wednesday'}, 'search_tvid': TVINFO_TRAKT}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_TVDB: 295837}}, 'search_tvid': TVINFO_TRAKT}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_IMDB: 4145054}}, 'search_tvid': TVINFO_TRAKT}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_TMDB: 119051}}, 'search_tvid': TVINFO_TRAKT}, # Wednesday
|
||
|
{'kwargs': {'ids': {TVINFO_TRAKT_SLUG: 'walker-independence'}}, 'search_tvid': TVINFO_TRAKT}, # Walker: Independence
|
||
|
# tmdb tests
|
||
|
{'kwargs': {'name': 'Shadowhunters'}, 'search_tvid': TVINFO_TMDB}, # Shadowhunters
|
||
|
{'kwargs': {'name': 'Wednesday'}, 'search_tvid': TVINFO_TMDB}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_TVDB: 295837}}, 'search_tvid': TVINFO_TMDB}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_IMDB: 4145054}}, 'search_tvid': TVINFO_TMDB}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_TMDB: 119051}}, 'search_tvid': TVINFO_TMDB}, # Wednesday
|
||
|
# tvmaze tests
|
||
|
{'kwargs': {'name': 'Shadowhunters'}, 'search_tvid': TVINFO_TVMAZE}, # Shadowhunters
|
||
|
{'kwargs': {'name': 'Wednesday'}, 'search_tvid': TVINFO_TVMAZE}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_TVDB: 295837}}, 'search_tvid': TVINFO_TVMAZE}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_IMDB: 4145054}}, 'search_tvid': TVINFO_TVMAZE}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_TVMAZE: 2158}}, 'search_tvid': TVINFO_TVMAZE}, # Shadowhunters
|
||
|
{'kwargs': {'ids': {TVINFO_TVMAZE: 53647}}, 'search_tvid': TVINFO_TVMAZE}, # Wednesday
|
||
|
]
|
||
|
|
||
|
|
||
|
person_search_tests = [
|
||
|
# tvdb tests
|
||
|
{'kwargs': {'name': 'Katherine McNamara'}, 'search_tvid': TVINFO_TVDB}, # Katherine McNamara
|
||
|
# {'kwargs': {'ids': {TVINFO_TVDB: 346941}}, 'search_tvid': TVINFO_TVDB}, # Katherine McNamara
|
||
|
# {'kwargs': {'ids': {TVINFO_IMDB: 3031063}}, 'search_tvid': TVINFO_TVDB}, # Katherine McNamara
|
||
|
# {'kwargs': {'ids': {TVINFO_TMDB: 968006}}, 'search_tvid': TVINFO_TVDB}, # Katherine McNamara
|
||
|
# {'kwargs': {'ids': {TVINFO_TVMAZE: 15776}}, 'search_tvid': TVINFO_TVDB}, # Katherine McNamara
|
||
|
# trakt tests
|
||
|
{'kwargs': {'name': 'Katherine McNamara'}, 'search_tvid': TVINFO_TRAKT}, # Katherine McNamara
|
||
|
# {'kwargs': {'ids': {TVINFO_TVDB: 346941}}, 'search_tvid': TVINFO_TRAKT}, # Katherine McNamara
|
||
|
{'kwargs': {'ids': {TVINFO_IMDB: 3031063}}, 'search_tvid': TVINFO_TRAKT}, # Katherine McNamara
|
||
|
{'kwargs': {'ids': {TVINFO_TMDB: 968006}}, 'search_tvid': TVINFO_TRAKT}, # Katherine McNamara
|
||
|
# tmdb tests
|
||
|
{'kwargs': {'name': 'Katherine McNamara'}, 'search_tvid': TVINFO_TMDB}, # Katherine McNamara
|
||
|
# {'kwargs': {'ids': {TVINFO_TVDB: 346941}}, 'search_tvid': TVINFO_TMDB}, # Katherine McNamara
|
||
|
{'kwargs': {'ids': {TVINFO_IMDB: 3031063}}, 'search_tvid': TVINFO_TMDB}, # Katherine McNamara
|
||
|
{'kwargs': {'ids': {TVINFO_TMDB: 968006}}, 'search_tvid': TVINFO_TMDB}, # Katherine McNamara
|
||
|
# tvmaze tests
|
||
|
{'kwargs': {'name': 'Katherine McNamara'}, 'search_tvid': TVINFO_TVMAZE}, # Katherine McNamara
|
||
|
# {'kwargs': {'ids': {TVINFO_TVDB: 346941}}, 'search_tvid': TVINFO_TVMAZE}, # Katherine McNamara
|
||
|
# {'kwargs': {'ids': {TVINFO_IMDB: 3031063}}, 'search_tvid': TVINFO_TVMAZE}, # Katherine McNamara
|
||
|
{'kwargs': {'ids': {TVINFO_TVMAZE: 15776}}, 'search_tvid': TVINFO_TVMAZE}, # Katherine McNamara
|
||
|
]
|
||
|
|
||
|
|
||
|
browser_extra_args = {
|
||
|
TVINFO_TVDB: {
|
||
|
'get_top_rated': [{'year': 2022}, {'year': 2020}, {'year': 2010}, {'in_last_year': True}]
|
||
|
},
|
||
|
TVINFO_TVMAZE: {},
|
||
|
TVINFO_TMDB: {
|
||
|
'get_trending': [{'time_window': 'days'}, {'time_window': 'week'}],
|
||
|
'get_similar': [{'tvid': 4145054}],
|
||
|
'get_recommended_for_show': [{'tvid': 4145054}],
|
||
|
},
|
||
|
TVINFO_TRAKT: {
|
||
|
'get_similar': [{'tvid': 99113}],
|
||
|
'get_recommended': [{'period': 'weekly'}, {'period': 'daily'}, {'period': 'monthly'}, {'period': 'yearly'},
|
||
|
{'period': 'all'}],
|
||
|
'get_most_collected': [{'period': 'weekly'}, {'period': 'daily'}, {'period': 'monthly'}, {'period': 'yearly'},
|
||
|
{'period': 'all'}],
|
||
|
'get_most_watched': [{'period': 'weekly'}, {'period': 'daily'}, {'period': 'monthly'}, {'period': 'yearly'},
|
||
|
{'period': 'all'}],
|
||
|
'get_most_played': [{'period': 'weekly'}, {'period': 'daily'}, {'period': 'monthly'}, {'period': 'yearly'},
|
||
|
{'period': 'all'}],
|
||
|
},
|
||
|
}
|
||
|
|
||
|
|
||
|
class TVInfoTests(test.SickbeardTestDBCase):
|
||
|
|
||
|
@classmethod
|
||
|
def setUpClass(cls) -> None:
|
||
|
test.teardown_test_db()
|
||
|
super(TVInfoTests, cls).setUpClass()
|
||
|
datetime.date = _FakeDate
|
||
|
datetime.datetime = _FakeDateTime
|
||
|
requests.sessions.Session.get = _mock_get
|
||
|
if disable_content_creation:
|
||
|
Tvdb.get_new_token = _mock_get_new_token
|
||
|
requests.sessions.Session.post = _mock_post
|
||
|
|
||
|
@classmethod
|
||
|
def tearDownClass(cls):
|
||
|
super(TVInfoTests, cls).tearDownClass()
|
||
|
files = {_f.name for _f in os.scandir(mock_data_dir) if _f.is_file()}
|
||
|
unused_files = files - used_files
|
||
|
if delete_unused_mock_files:
|
||
|
for _u_f in unused_files:
|
||
|
full_filename = os.path.join(mock_data_dir, _u_f)
|
||
|
try:
|
||
|
os.remove(full_filename)
|
||
|
except (BaseException, Exception) as e:
|
||
|
print('errror deleting: [%s], error: %s' % (full_filename, e))
|
||
|
if unused_files:
|
||
|
print('unused files: %s' % unused_files)
|
||
|
print('reset mock methods')
|
||
|
datetime.date = datetime_date_backup
|
||
|
datetime.datetime = datetime_datetime_backup
|
||
|
requests.sessions.Session.get = backup_session_get
|
||
|
requests.sessions.Session.post = backup_session_post
|
||
|
|
||
|
def tearDown(self):
|
||
|
super(TVInfoTests, self).tearDown()
|
||
|
try:
|
||
|
gc.collect(2)
|
||
|
shutil.rmtree(sickgear.CACHE_DIR)
|
||
|
except (BaseException, Exception):
|
||
|
pass
|
||
|
|
||
|
def test_person_data(self):
|
||
|
for t_c in person_tests:
|
||
|
tvid = t_c['tvid']
|
||
|
p_id = t_c['p_id']
|
||
|
print('testing person: %s: %s' % (sickgear.TVInfoAPI(tvid).name, p_id))
|
||
|
|
||
|
tvinfo_config = sickgear.TVInfoAPI(tvid).api_params.copy()
|
||
|
tvinfo_config['cache'] = False
|
||
|
|
||
|
t = sickgear.TVInfoAPI(tvid).setup(**tvinfo_config)
|
||
|
t.clear_cache()
|
||
|
person_obj = t.get_person(p_id, get_show_credits=True, get_images=True)
|
||
|
filename = _make_filename('%s_%s_person' % (tvid, p_id), '.obj_data')
|
||
|
for _c in person_obj.characters:
|
||
|
_c.ti_show.load_data()
|
||
|
if os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_p_o = _load_pickle_file(filename) # type: TVInfoPerson
|
||
|
elif not disable_content_creation:
|
||
|
_save_pickle_file(filename, person_obj)
|
||
|
_p_o = person_obj
|
||
|
try:
|
||
|
assert None is not person_obj and None is not _p_o
|
||
|
_compare_helper(person_obj, _p_o)
|
||
|
except (BaseException, Exception) as e:
|
||
|
if not disable_content_creation and os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_save_pickle_file(filename, person_obj)
|
||
|
_p_o = person_obj
|
||
|
assert None is not person_obj and None is not _p_o
|
||
|
_compare_helper(person_obj, _p_o)
|
||
|
else:
|
||
|
raise e
|
||
|
|
||
|
def test_show_data(self):
|
||
|
for t_c in show_tests:
|
||
|
tvid = t_c['tvid']
|
||
|
prodid = t_c['prodid']
|
||
|
print('testing show: %s: %s' % (sickgear.TVInfoAPI(tvid).name, prodid))
|
||
|
|
||
|
tvinfo_config = sickgear.TVInfoAPI(tvid).api_params.copy()
|
||
|
tvinfo_config['cache'] = False
|
||
|
|
||
|
t = sickgear.TVInfoAPI(tvid).setup(**tvinfo_config)
|
||
|
t.clear_cache()
|
||
|
show_obj = t.get_show(prodid, load_episodes=True, banners=True, posters=True, seasons=True,
|
||
|
seasonwides=True, fanart=True, actors=True)
|
||
|
filename = _make_filename('%s_%s_show' % (tvid, prodid), '.obj_data')
|
||
|
if os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_s_o = _load_pickle_file(filename) # type: TVInfoShow
|
||
|
elif not disable_content_creation:
|
||
|
_save_pickle_file(filename, show_obj)
|
||
|
_s_o = show_obj
|
||
|
try:
|
||
|
assert None is not show_obj and None is not _s_o
|
||
|
_compare_helper(show_obj, _s_o)
|
||
|
except (BaseException, Exception) as e:
|
||
|
if not disable_content_creation and os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_save_pickle_file(filename, show_obj)
|
||
|
_s_o = show_obj
|
||
|
assert None is not show_obj and None is not _s_o
|
||
|
_compare_helper(show_obj, _s_o)
|
||
|
else:
|
||
|
raise e
|
||
|
|
||
|
def test_person_search(self):
|
||
|
for t_c in person_search_tests:
|
||
|
search_tvid = t_c['search_tvid']
|
||
|
print('testing [%s] person search for: %s' % (sickgear.TVInfoAPI(search_tvid).name, t_c['kwargs']))
|
||
|
|
||
|
tvinfo_config = sickgear.TVInfoAPI(search_tvid).api_params.copy()
|
||
|
tvinfo_config['cache'] = False
|
||
|
|
||
|
t = sickgear.TVInfoAPI(search_tvid).setup(**tvinfo_config)
|
||
|
t.clear_cache()
|
||
|
search_results = t.search_person(**t_c['kwargs'])
|
||
|
filename = _make_filename('%s_%s_person_search' % (search_tvid, ','.join(
|
||
|
['%s%s' % (_k, _v) for _k, _v in t_c['kwargs'].items()])), '.obj_data')
|
||
|
if os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_s_o = _load_pickle_file(filename) # type: List[TVInfoPerson]
|
||
|
elif not disable_content_creation:
|
||
|
_save_pickle_file(filename, search_results)
|
||
|
_s_o = search_results
|
||
|
try:
|
||
|
assert None is not search_results and None is not _s_o
|
||
|
for _i, _r in enumerate(search_results):
|
||
|
_compare_helper(_r, _s_o[_i])
|
||
|
except (BaseException, Exception) as e:
|
||
|
if not disable_content_creation and os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_save_pickle_file(filename, search_results)
|
||
|
_s_o = search_results
|
||
|
assert None is not search_results and None is not _s_o
|
||
|
for _i, _r in enumerate(search_results):
|
||
|
_compare_helper(_r, _s_o[_i])
|
||
|
else:
|
||
|
raise e
|
||
|
|
||
|
def test_show_search(self):
|
||
|
for t_c in search_tests:
|
||
|
search_tvid = t_c['search_tvid']
|
||
|
print('testing [%s] search for: %s' % (sickgear.TVInfoAPI(search_tvid).name, t_c['kwargs']))
|
||
|
|
||
|
tvinfo_config = sickgear.TVInfoAPI(search_tvid).api_params.copy()
|
||
|
tvinfo_config['cache'] = False
|
||
|
|
||
|
t = sickgear.TVInfoAPI(search_tvid).setup(**tvinfo_config)
|
||
|
t.clear_cache()
|
||
|
search_results = t.search_show(**t_c['kwargs'])
|
||
|
filename = _make_filename('%s_%s_search' % (search_tvid, ','.join(
|
||
|
['%s%s' % (_k, _v) for _k, _v in t_c['kwargs'].items()])), '.obj_data')
|
||
|
if os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_s_o = _load_pickle_file(filename) # type: List[TVInfoShow]
|
||
|
elif not disable_content_creation:
|
||
|
_save_pickle_file(filename, search_results)
|
||
|
_s_o = search_results
|
||
|
try:
|
||
|
assert None is not search_results and None is not _s_o
|
||
|
for _i, _r in enumerate(search_results):
|
||
|
_compare_helper(_r, _s_o[_i])
|
||
|
except (BaseException, Exception) as e:
|
||
|
if not disable_content_creation and os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_save_pickle_file(filename, search_results)
|
||
|
_s_o = search_results
|
||
|
assert None is not search_results and None is not _s_o
|
||
|
for _i, _r in enumerate(search_results):
|
||
|
_compare_helper(_r, _s_o[_i])
|
||
|
else:
|
||
|
raise e
|
||
|
|
||
|
def test_browse_endpoints(self):
|
||
|
for _tvid in sickgear.TVInfoAPI().all_sources:
|
||
|
tvinfo_config = sickgear.TVInfoAPI(_tvid).api_params.copy()
|
||
|
tvinfo_config['cache'] = False
|
||
|
|
||
|
t = sickgear.TVInfoAPI(_tvid).setup(**tvinfo_config)
|
||
|
t.clear_cache()
|
||
|
for _m in (('get_premieres', {}), ('get_anticipated', {}),
|
||
|
('get_recommended', {}), ('get_most_collected', {}), ('get_most_watched', {}),
|
||
|
('get_most_played', {}), ('get_returning', {}), ('discover', {}), ('get_new_seasons', {}),
|
||
|
('get_new_shows', {}), ('get_top_rated', {}), ('get_popular', {}), ('get_trending', {}),
|
||
|
('get_recommended_for_show', {}), ('get_similar', {})):
|
||
|
_method = getattr(t, _m[0], None)
|
||
|
# check if method is implemented in tvid source
|
||
|
if not _method or 'TVInfoBase' in _method.__qualname__:
|
||
|
continue
|
||
|
for _e_kw in browser_extra_args.get(_tvid, {}).get(_m[0], [{}]) or [{}]:
|
||
|
kw = _m[1].copy()
|
||
|
kw.update(_e_kw)
|
||
|
print('testing %s: %s%s' % (sickgear.TVInfoAPI(_tvid).name, _m[0],
|
||
|
('', ' with parameter: %s' % kw)[0 != len(kw)]))
|
||
|
results = _method(**kw)
|
||
|
filename = _make_filename('%s_%s_%s' % (
|
||
|
_tvid, ''.join('%s%s' % (_k, _v) for _k, _v in kw.items()), _m[0]), '.obj_data')
|
||
|
if os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_s_o = _load_pickle_file(filename) # type: List[TVInfoShow]
|
||
|
elif not disable_content_creation:
|
||
|
_save_pickle_file(filename, results)
|
||
|
_s_o = results
|
||
|
|
||
|
try:
|
||
|
assert None is not results and None is not _s_o
|
||
|
for _i, _r in enumerate(results):
|
||
|
_compare_helper(_r, _s_o[_i])
|
||
|
except (BaseException, Exception) as e:
|
||
|
if not disable_content_creation and os.path.isfile(os.path.join(mock_data_dir, filename)):
|
||
|
_save_pickle_file(filename, results)
|
||
|
_s_o = results
|
||
|
assert None is not results and None is not _s_o
|
||
|
for _i, _r in enumerate(results):
|
||
|
_compare_helper(_r, _s_o[_i])
|
||
|
else:
|
||
|
raise e
|
||
|
|
||
|
|
||
|
if '__main__' == __name__:
|
||
|
|
||
|
print('===============================')
|
||
|
print('STARTING - TV Info TESTS')
|
||
|
print('===============================')
|
||
|
|
||
|
try:
|
||
|
suite = unittest.TestLoader().loadTestsFromTestCase(TVInfoTests)
|
||
|
unittest.TextTestRunner(verbosity=2).run(suite)
|
||
|
finally:
|
||
|
requests.sessions.Session.get = backup_session_get
|
||
|
requests.sessions.Session.post = backup_session_post
|
||
|
datetime.date = datetime_date_backup
|
||
|
datetime.datetime = datetime_datetime_backup
|