import requests , time , webbrowser
from modules import util
from modules . util import Failed , TimeoutExpired
from retrying import retry
from ruamel import yaml
logger = util . logger
redirect_uri = " urn:ietf:wg:oauth:2.0:oob "
base_url = " https://api.trakt.tv "
builders = [
" trakt_list " , " trakt_list_details " , " trakt_chart " , " trakt_userlist " , " trakt_boxoffice " , " trakt_recommendations " ,
" trakt_collected_daily " , " trakt_collected_weekly " , " trakt_collected_monthly " , " trakt_collected_yearly " , " trakt_collected_all " ,
" trakt_recommended_daily " , " trakt_recommended_weekly " , " trakt_recommended_monthly " , " trakt_recommended_yearly " , " trakt_recommended_all " ,
" trakt_watched_daily " , " trakt_watched_weekly " , " trakt_watched_monthly " , " trakt_watched_yearly " , " trakt_watched_all " ,
" trakt_collection " , " trakt_popular " , " trakt_trending " , " trakt_watchlist "
]
sorts = [
" rank " , " added " , " title " , " released " , " runtime " , " popularity " ,
" percentage " , " votes " , " random " , " my_rating " , " watched " , " collected "
]
status = [ " returning " , " production " , " planned " , " canceled " , " ended " ]
status_translation = {
" returning " : " returning series " , " production " : " in production " ,
" planned " : " planned " , " canceled " : " canceled " , " ended " : " ended "
}
periods = [ " daily " , " weekly " , " monthly " , " yearly " , " all " ]
id_translation = { " movie " : " movie " , " show " : " show " , " season " : " show " , " episode " : " show " , " person " : " person " , " list " : " list " }
id_types = {
" movie " : ( " tmdb " , " TMDb ID " ) ,
" person " : ( " tmdb " , " TMDb ID " ) ,
" show " : ( " tvdb " , " TVDb ID " ) ,
" season " : ( " tvdb " , " TVDb ID " ) ,
" episode " : ( " tvdb " , " TVDb ID " ) ,
" list " : ( " slug " , " Trakt Slug " )
}
class Trakt :
def __init__ ( self , config , params ) :
self . config = config
self . client_id = params [ " client_id " ]
self . client_secret = params [ " client_secret " ]
self . pin = params [ " pin " ]
self . config_path = params [ " config_path " ]
self . authorization = params [ " authorization " ]
logger . secret ( self . client_secret )
if not self . _save ( self . authorization ) :
if not self . _refresh ( ) :
self . _authorization ( )
self . _slugs = None
self . _movie_genres = None
self . _show_genres = None
self . _movie_languages = None
self . _show_languages = None
self . _movie_countries = None
self . _show_countries = None
self . _movie_certifications = None
self . _show_certifications = None
@property
def slugs ( self ) :
if self . _slugs is None :
items = [ ]
try :
items = [ i [ " ids " ] [ " slug " ] for i in self . _request ( f " /users/me/lists " ) ]
except Failed :
pass
self . _slugs = items
return self . _slugs
@property
def movie_genres ( self ) :
if not self . _movie_genres :
self . _movie_genres = [ g [ " slug " ] for g in self . _request ( " /genres/movies " ) ]
return self . _movie_genres
@property
def show_genres ( self ) :
if not self . _show_genres :
self . _show_genres = [ g [ " slug " ] for g in self . _request ( " /genres/shows " ) ]
return self . _show_genres
@property
def movie_languages ( self ) :
if not self . _movie_languages :
self . _movie_languages = [ g [ " code " ] for g in self . _request ( " /languages/movies " ) ]
return self . _movie_languages
@property
def show_languages ( self ) :
if not self . _show_languages :
self . _show_languages = [ g [ " code " ] for g in self . _request ( " /languages/shows " ) ]
return self . _show_languages
@property
def movie_countries ( self ) :
if not self . _movie_countries :
self . _movie_countries = [ g [ " code " ] for g in self . _request ( " /countries/movies " ) ]
return self . _movie_countries
@property
def show_countries ( self ) :
if not self . _show_countries :
self . _show_countries = [ g [ " code " ] for g in self . _request ( " /countries/shows " ) ]
return self . _show_countries
@property
def movie_certifications ( self ) :
if not self . _movie_certifications :
self . _movie_certifications = [ g [ " slug " ] for g in self . _request ( " /certifications/movies " ) [ " us " ] ]
return self . _movie_certifications
@property
def show_certifications ( self ) :
if not self . _show_certifications :
self . _show_certifications = [ g [ " slug " ] for g in self . _request ( " /certifications/shows " ) [ " us " ] ]
return self . _show_certifications
def _authorization ( self ) :
if self . pin :
pin = self . pin
else :
url = f " https://trakt.tv/oauth/authorize?response_type=code&redirect_uri= { redirect_uri } &client_id= { self . client_id } "
logger . info ( f " Navigate to: { url } " )
logger . info ( " If you get an OAuth error your client_id or client_secret is invalid " )
webbrowser . open ( url , new = 2 )
try : pin = util . logger_input ( " Trakt pin (case insensitive) " , timeout = 300 ) . strip ( )
except TimeoutExpired : raise Failed ( " Input Timeout: Trakt pin required. " )
if not pin : raise Failed ( " Trakt Error: Trakt pin required. " )
json = {
" code " : pin ,
" client_id " : self . client_id ,
" client_secret " : self . client_secret ,
" redirect_uri " : redirect_uri ,
" grant_type " : " authorization_code "
}
response = self . config . post ( f " { base_url } /oauth/token " , json = json , headers = { " Content-Type " : " application/json " } )
if response . status_code != 200 :
raise Failed ( " Trakt Error: Invalid trakt pin. If you ' re sure you typed it in correctly your client_id or client_secret may be invalid " )
elif not self . _save ( response . json ( ) ) :
raise Failed ( " Trakt Error: New Authorization Failed " )
def _check ( self , authorization = None ) :
token = self . authorization [ ' access_token ' ] if authorization is None else authorization [ ' access_token ' ]
headers = {
" Content-Type " : " application/json " ,
" Authorization " : f " Bearer { token } " ,
" trakt-api-version " : " 2 " ,
" trakt-api-key " : self . client_id
}
logger . secret ( token )
response = self . config . get ( f " { base_url } /users/settings " , headers = headers )
return response . status_code == 200
def _refresh ( self ) :
if self . authorization and " refresh_token " in self . authorization and self . authorization [ " refresh_token " ] :
logger . info ( " Refreshing Access Token... " )
json = {
" refresh_token " : self . authorization [ " refresh_token " ] ,
" client_id " : self . client_id ,
" client_secret " : self . client_secret ,
" redirect_uri " : redirect_uri ,
" grant_type " : " refresh_token "
}
response = self . config . post ( f " { base_url } /oauth/token " , json = json , headers = { " Content-Type " : " application/json " } )
if response . status_code != 200 :
return False
return self . _save ( response . json ( ) )
return False
def _save ( self , authorization ) :
if authorization and self . _check ( authorization ) :
if self . authorization != authorization and not self . config . read_only :
yaml . YAML ( ) . allow_duplicate_keys = True
config , ind , bsi = yaml . util . load_yaml_guess_indent ( open ( self . config_path , encoding = " utf-8 " ) )
config [ " trakt " ] [ " pin " ] = None
config [ " trakt " ] [ " authorization " ] = {
" access_token " : authorization [ " access_token " ] ,
" token_type " : authorization [ " token_type " ] ,
" expires_in " : authorization [ " expires_in " ] ,
" refresh_token " : authorization [ " refresh_token " ] ,
" scope " : authorization [ " scope " ] ,
" created_at " : authorization [ " created_at " ]
}
logger . info ( f " Saving authorization information to { self . config_path } " )
yaml . round_trip_dump ( config , open ( self . config_path , " w " ) , indent = ind , block_seq_indent = bsi )
self . authorization = authorization
return True
return False
@retry ( stop_max_attempt_number = 6 , wait_fixed = 10000 , retry_on_exception = util . retry_if_not_failed )
def _request ( self , url , params = None , json = None ) :
headers = {
" Content-Type " : " application/json " ,
" Authorization " : f " Bearer { self . authorization [ ' access_token ' ] } " ,
" trakt-api-version " : " 2 " ,
" trakt-api-key " : self . client_id
}
output_json = [ ]
if params is None :
params = { }
pages = 1
current = 1
if self . config . trace_mode :
logger . debug ( f " URL: { base_url } { url } " )
if params :
logger . debug ( f " Params: { params } " )
if json :
logger . debug ( f " JSON: { json } " )
while current < = pages :
if pages > 1 :
params [ " page " ] = current
if json is not None :
response = self . config . post ( f " { base_url } { url } " , json = json , headers = headers )
else :
response = self . config . get ( f " { base_url } { url } " , headers = headers , params = params )
if pages == 1 and " X-Pagination-Page-Count " in response . headers :
pages = int ( response . headers [ " X-Pagination-Page-Count " ] )
if response . status_code > = 400 :
raise Failed ( f " ( { response . status_code } ) { response . reason } " )
json_data = response . json ( )
if self . config . trace_mode :
logger . debug ( f " Response: { json_data } " )
if isinstance ( json_data , dict ) :
return json_data
else :
output_json . extend ( json_data )
current + = 1
return output_json
def user_ratings ( self , is_movie ) :
media = " movie " if is_movie else " show "
id_type = " tmdb " if is_movie else " tvdb "
return { int ( i [ media ] [ " ids " ] [ id_type ] ) : i [ " rating " ] for i in self . _request ( f " /users/me/ratings/ { media } s " ) }
def convert ( self , external_id , from_source , to_source , media_type ) :
path = f " /search/ { from_source } / { external_id } "
params = { " type " : media_type } if from_source in [ " tmdb " , " tvdb " ] else None
lookup = self . _request ( path , params = params )
if lookup and media_type in lookup [ 0 ] and to_source in lookup [ 0 ] [ media_type ] [ " ids " ] :
return lookup [ 0 ] [ media_type ] [ " ids " ] [ to_source ]
raise Failed ( f " Trakt Error: No { to_source . upper ( ) . replace ( ' B ' , ' b ' ) } ID found for { from_source . upper ( ) . replace ( ' B ' , ' b ' ) } ID: { external_id } " )
def list_description ( self , data ) :
try :
return self . _request ( requests . utils . urlparse ( data ) . path ) [ " description " ]
except Failed :
raise Failed ( f " Trakt Error: List { data } not found " )
def _parse ( self , items , typeless = False , item_type = None , trakt_ids = False ) :
ids = [ ]
for item in items :
if typeless :
data = item
current_type = item_type
elif item_type :
data = item [ item_type ]
current_type = item_type
elif " type " in item and item [ " type " ] in id_translation :
data = item [ id_translation [ item [ " type " ] ] ]
current_type = item [ " type " ]
else :
continue
id_type , id_display = id_types [ current_type ]
if id_type in data [ " ids " ] and data [ " ids " ] [ id_type ] :
final_id = data [ " ids " ] [ id_type ]
if current_type == " episode " :
final_id = f " { final_id } _ { item [ current_type ] [ ' season ' ] } "
if current_type in [ " episode " , " season " ] :
final_id = f " { final_id } _ { item [ current_type ] [ ' number ' ] } "
if current_type in [ " person " , " list " ] :
final_id = ( final_id , data [ " name " ] )
final_type = f " { id_type } _ { current_type } " if current_type in [ " episode " , " season " , " person " ] else id_type
ids . append ( ( int ( item [ " id " ] ) , final_id , final_type ) if trakt_ids else ( final_id , final_type ) )
else :
name = data [ " name " ] if current_type in [ " person " , " list " ] else f " { data [ ' title ' ] } ( { data [ ' year ' ] } ) "
logger . error ( f " Trakt Error: No { id_display } found for { name } " )
return ids
def _build_item_json ( self , ids ) :
data = { }
for input_id , id_type in ids :
movies = id_type in [ " imdb " , " tmdb " ]
shows = id_type in [ " imdb " , " tvdb " , " tmdb_show " , " tvdb_season " , " tvdb_episode " ]
if not movies and not shows :
continue
type_set = str ( id_type ) . split ( " _ " )
id_set = str ( input_id ) . split ( " _ " )
item = { " ids " : { type_set [ 0 ] : id_set [ 0 ] if type_set [ 0 ] == " imdb " else int ( id_set [ 0 ] ) } }
if id_type in [ " tvdb_season " , " tvdb_episode " ] :
season_data = { " number " : int ( id_set [ 1 ] ) }
if id_type == " tvdb_episode " :
season_data [ " episodes " ] = [ { " number " : int ( id_set [ 2 ] ) } ]
item [ " seasons " ] = [ season_data ]
if movies :
if " movies " not in data :
data [ " movies " ] = [ ]
data [ " movies " ] . append ( item )
if shows :
if " shows " not in data :
data [ " shows " ] = [ ]
data [ " shows " ] . append ( item )
return data
def sync_list ( self , slug , ids ) :
current_ids = self . _list ( slug , urlparse = False )
def read_result ( data , obj_type , result_type , result_str = None ) :
result_str = result_str if result_str else result_type . capitalize ( )
if data [ result_type ] [ obj_type ] > 0 :
logger . info ( f " { data [ result_type ] [ obj_type ] } { obj_type . capitalize ( ) } { result_str } " )
def read_not_found ( data , result_str ) :
not_found = [ ]
for item in data [ " not_found " ] [ " movies " ] :
not_found . append ( ( item [ " ids " ] [ " tmdb " ] , " tmdb " ) )
for item in data [ " not_found " ] [ " shows " ] :
not_found . append ( ( item [ " ids " ] [ " tvdb " ] , " tvdb " ) )
for item in data [ " not_found " ] [ " seasons " ] :
not_found . append ( ( f " { item [ ' ids ' ] [ ' tvdb ' ] } _ { item [ ' seasons ' ] [ 0 ] [ ' number ' ] } " , " tvdb_season " ) )
for item in data [ " not_found " ] [ " episodes " ] :
not_found . append ( ( f " { item [ ' ids ' ] [ ' tvdb ' ] } _ { item [ ' seasons ' ] [ 0 ] [ ' number ' ] } _ { item [ ' seasons ' ] [ 0 ] [ ' episodes ' ] [ 0 ] [ ' number ' ] } " , " tvdb_episode " ) )
if not_found :
logger . error ( f " { len ( not_found ) } Items Unable to { result_str } : { not_found } " )
add_ids = [ id_set for id_set in ids if id_set not in current_ids ]
if add_ids :
logger . info ( " " )
results = self . _request ( f " /users/me/lists/ { slug } /items " , json = self . _build_item_json ( add_ids ) )
for object_type in [ " movies " , " shows " , " seasons " , " episodes " ] :
read_result ( results , object_type , " added " )
read_not_found ( results , " Add " )
time . sleep ( 1 )
remove_ids = [ id_set for id_set in current_ids if id_set not in ids ]
if remove_ids :
logger . info ( " " )
results = self . _request ( f " /users/me/lists/ { slug } /items/remove " , json = self . _build_item_json ( remove_ids ) )
for object_type in [ " movies " , " shows " , " seasons " , " episodes " ] :
read_result ( results , object_type , " deleted " , " Removed " )
read_not_found ( results , " Remove " )
time . sleep ( 1 )
trakt_ids = self . _list ( slug , urlparse = False , trakt_ids = True )
trakt_lookup = { f " { ty } _ { i_id } " : t_id for t_id , i_id , ty in trakt_ids }
rank_ids = [ trakt_lookup [ f " { ty } _ { i_id } " ] for i_id , ty in ids if f " { ty } _ { i_id } " in trakt_lookup ]
self . _request ( f " /users/me/lists/ { slug } /items/reorder " , json = { " rank " : rank_ids } )
logger . info ( " " )
logger . info ( " Trakt List Ordered Successfully " )
def all_user_lists ( self , user = " me " ) :
try :
items = self . _request ( f " /users/ { user } /lists " )
except Failed :
raise Failed ( f " Trakt Error: User { user } not found " )
if len ( items ) == 0 :
raise Failed ( f " Trakt Error: User { user } has no lists " )
return [ ( user , i [ " ids " ] [ " slug " ] , i [ " name " ] ) for i in items ]
def all_liked_lists ( self ) :
items = self . _request ( f " /users/likes/lists " )
if len ( items ) == 0 :
raise Failed ( f " Trakt Error: No Liked lists found " )
return { self . build_user_url ( i [ ' list ' ] [ ' user ' ] [ ' ids ' ] [ ' slug ' ] , i [ ' list ' ] [ ' ids ' ] [ ' slug ' ] ) : i [ " list " ] [ " name " ] for i in items }
def build_user_url ( self , user , name ) :
return f " { base_url . replace ( ' api. ' , ' ' ) } /users/ { user } /lists/ { name } "
def _list ( self , data , urlparse = True , trakt_ids = False ) :
try :
url = requests . utils . urlparse ( data ) . path if urlparse else f " /users/me/lists/ { data } "
items = self . _request ( f " { url } /items " )
except Failed :
raise Failed ( f " Trakt Error: List { data } not found " )
if len ( items ) == 0 :
raise Failed ( f " Trakt Error: List { data } is empty " )
return self . _parse ( items , trakt_ids = trakt_ids )
def _userlist ( self , list_type , user , is_movie , sort_by = None ) :
try :
url_end = " movies " if is_movie else " shows "
if sort_by :
url_end = f " { url_end } / { sort_by } "
items = self . _request ( f " /users/ { user } / { list_type } / { url_end } " )
except Failed :
raise Failed ( f " Trakt Error: User { user } not found " )
if len ( items ) == 0 :
raise Failed ( f " Trakt Error: { user } ' s { list_type . capitalize ( ) } is empty " )
return self . _parse ( items , item_type = " movie " if is_movie else " show " )
def _recommendations ( self , limit , is_movie ) :
media_type = " Movie " if is_movie else " Show "
try :
items = self . _request ( f " /recommendations/ { ' movies ' if is_movie else ' shows ' } " , params = { " limit " : limit } )
except Failed :
raise Failed ( f " Trakt Error: failed to fetch { media_type } Recommendations " )
if len ( items ) == 0 :
raise Failed ( f " Trakt Error: no { media_type } Recommendations were found " )
return self . _parse ( items , typeless = True , item_type = " movie " if is_movie else " show " )
def _charts ( self , chart_type , is_movie , params , time_period = None ) :
chart_url = f " { chart_type } / { time_period } " if time_period else chart_type
items = self . _request ( f " / { ' movies ' if is_movie else ' shows ' } / { chart_url } " , params = params )
return self . _parse ( items , typeless = chart_type == " popular " , item_type = " movie " if is_movie else " show " )
def get_people ( self , data ) :
return { str ( i [ 0 ] [ 0 ] ) : i [ 0 ] [ 1 ] for i in self . _list ( data ) if i [ 1 ] == " tmdb_person " }
def validate_list ( self , trakt_lists ) :
values = util . get_list ( trakt_lists , split = False )
trakt_values = [ ]
for value in values :
if isinstance ( value , dict ) :
raise Failed ( " Trakt Error: List cannot be a dictionary " )
try :
self . _list ( value )
trakt_values . append ( value )
except Failed as e :
logger . error ( e )
if len ( trakt_values ) == 0 :
raise Failed ( f " Trakt Error: No valid Trakt Lists in { values } " )
return trakt_values
def validate_chart ( self , err_type , method_name , data , is_movie ) :
valid_dicts = [ ]
for trakt_dict in util . get_list ( data , split = False ) :
if not isinstance ( trakt_dict , dict ) :
raise Failed ( f " { err_type } Error: { method_name } must be a dictionary " )
dict_methods = { dm . lower ( ) : dm for dm in trakt_dict }
try :
if method_name == " trakt_chart " :
final_dict = { }
final_dict [ " chart " ] = util . parse ( err_type , " chart " , trakt_dict , methods = dict_methods , parent = method_name , options = [ " recommended " , " watched " , " collected " , " trending " , " popular " ] )
final_dict [ " limit " ] = util . parse ( err_type , " limit " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " int " , default = 10 )
final_dict [ " time_period " ] = None
if final_dict [ " chart " ] in [ " recommended " , " watched " , " collected " ] and " time_period " in dict_methods :
final_dict [ " time_period " ] = util . parse ( err_type , " time_period " , trakt_dict , methods = dict_methods , parent = method_name , default = " weekly " , options = periods )
if " query " in dict_methods :
final_dict [ " query " ] = util . parse ( err_type , " query " , trakt_dict , methods = dict_methods , parent = method_name )
if " year " in dict_methods :
try :
if trakt_dict [ dict_methods [ " year " ] ] and len ( str ( trakt_dict [ dict_methods [ " year " ] ] ) ) == 4 :
final_dict [ " year " ] = util . parse ( err_type , " year " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " int " , minimum = 1000 , maximum = 3000 )
else :
final_dict [ " year " ] = util . parse ( err_type , " year " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " int " , minimum = 1000 , maximum = 3000 , range_split = " - " )
except Failed :
raise Failed ( f " { err_type } Error: trakt_chart year attribute must be either a 4 digit year or a range of two 4 digit year with a ' - ' i.e. 1950 or 1950-1959 " )
if " runtimes " in dict_methods :
final_dict [ " runtimes " ] = util . parse ( err_type , " runtimes " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " int " , range_split = " - " )
if " ratings " in dict_methods :
final_dict [ " ratings " ] = util . parse ( err_type , " ratings " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " int " , minimum = 0 , maximum = 100 , range_split = " - " )
if " genres " in dict_methods :
final_dict [ " genres " ] = util . parse ( err_type , " genres " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " commalist " , options = self . movie_genres if is_movie else self . show_genres )
if " languages " in dict_methods :
final_dict [ " languages " ] = util . parse ( err_type , " languages " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " commalist " , options = self . movie_languages if is_movie else self . show_languages )
if " countries " in dict_methods :
final_dict [ " countries " ] = util . parse ( err_type , " countries " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " commalist " , options = self . movie_countries if is_movie else self . show_countries )
if " certifications " in dict_methods :
final_dict [ " certifications " ] = util . parse ( err_type , " certifications " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " commalist " , options = self . movie_certifications if is_movie else self . show_certifications )
if " networks " in dict_methods and not is_movie :
final_dict [ " networks " ] = util . parse ( err_type , " networks " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " commalist " )
if " status " in dict_methods and not is_movie :
final_dict [ " status " ] = util . parse ( err_type , " status " , trakt_dict , methods = dict_methods , parent = method_name , datatype = " commalist " , options = status )
valid_dicts . append ( final_dict )
else :
userlist = util . parse ( err_type , " userlist " , trakt_dict , methods = dict_methods , parent = method_name , options = [ " recommended " , " watched " , " collected " , " watchlist " ] )
user = util . parse ( err_type , " user " , trakt_dict , methods = dict_methods , parent = method_name , default = " me " )
sort_by = None
if userlist in [ " recommended " , " watchlist " ] and " sort " in dict_methods :
sort_by = util . parse ( err_type , " sort_by " , trakt_dict , methods = dict_methods , parent = method_name , default = " rank " , options = [ " rank " , " added " , " released " , " title " ] )
self . _userlist ( " collection " if userlist == " collected " else userlist , user , is_movie , sort_by = sort_by )
valid_dicts . append ( { " userlist " : userlist , " user " : user , " sort_by " : sort_by } )
except Failed as e :
logger . error ( e )
if len ( valid_dicts ) == 0 :
raise Failed ( f " Trakt Error: No valid Trakt { method_name [ 6 : ] . capitalize ( ) } " )
return valid_dicts
def get_trakt_ids ( self , method , data , is_movie ) :
pretty = method . replace ( " _ " , " " ) . title ( )
media_type = " Movie " if is_movie else " Show "
if method == " trakt_list " :
logger . info ( f " Processing { pretty } : { data } " )
return self . _list ( data )
elif method == " trakt_recommendations " :
logger . info ( f " Processing { pretty } : { data } { media_type } { ' ' if data == 1 else ' s ' } " )
return self . _recommendations ( data , is_movie )
elif method == " trakt_chart " :
params = { " limit " : data [ " limit " ] }
chart_limit = f " { data [ ' limit ' ] } { data [ ' time_period ' ] . capitalize ( ) } " if data [ " time_period " ] else data [ " limit " ]
logger . info ( f " Processing { pretty } : { chart_limit } { data [ ' chart ' ] . capitalize ( ) } { media_type } { ' ' if data == 1 else ' s ' } " )
for attr in [ " query " , " year " , " runtimes " , " ratings " , " genres " , " languages " , " countries " , " certifications " , " networks " , " status " ] :
if attr in data :
logger . info ( f " { attr : >22 } : { ' , ' . join ( data [ attr ] ) if isinstance ( data [ attr ] , list ) else data [ attr ] } " )
values = [ status_translation [ v ] for v in data [ attr ] ] if attr == " status " else data [ attr ]
params [ attr ] = " , " . join ( values ) if isinstance ( values , list ) else values
return self . _charts ( data [ " chart " ] , is_movie , params , time_period = data [ " time_period " ] )
elif method == " trakt_userlist " :
logger . info ( f " Processing { pretty } { media_type } s from { data [ ' user ' ] } ' s { data [ ' userlist ' ] . capitalize ( ) } " )
return self . _userlist ( data [ " userlist " ] , data [ " user " ] , is_movie , sort_by = data [ " sort_by " ] )
elif method == " trakt_boxoffice " :
logger . info ( f " Processing { pretty } : { data } { media_type } { ' ' if data == 1 else ' s ' } " )
return self . _charts ( " boxoffice " , is_movie , { " limit " : data } )
else :
raise Failed ( f " Trakt Error: Method { method } not supported " )