@ -86,9 +86,9 @@ class CollectionBuilder:
else :
raise Failed ( " Collection Error: template sub-attribute optional is blank " )
for m in template :
if m not in self . data and m not in [ " default " , " optional " ] :
if template[ m ] :
for m ethod_name, attr_data in template :
if m ethod_name not in self . data and m ethod_name not in [ " default " , " optional " ] :
if attr_data :
def replace_txt ( txt ) :
txt = str ( txt )
for option in optional :
@ -108,33 +108,33 @@ class CollectionBuilder:
try : return int ( txt )
except ValueError : return txt
try :
if isinstance ( template[ m ] , dict ) :
a = { }
for sm in template[ m ] :
if isinstance ( template[ m ] [ sm ] , list ) :
if isinstance ( attr_data , dict ) :
discover_n ame = { }
for sm in attr_data :
if isinstance ( attr_data [ sm ] , list ) :
temp_list = [ ]
for li in template[ m ] [ sm ] :
for li in attr_data [ sm ] :
temp_list . append ( replace_txt ( li ) )
a[ sm ] = temp_list
discover_n ame [ sm ] = temp_list
else :
a[ sm ] = replace_txt ( template[ m ] [ sm ] )
elif isinstance ( template[ m ] , list ) :
a = [ ]
for li in template[ m ] :
discover_n ame [ sm ] = replace_txt ( attr_data [ sm ] )
elif isinstance ( attr_data , list ) :
discover_n ame = [ ]
for li in attr_data :
if isinstance ( li , dict ) :
temp_dict = { }
for sm in li :
temp_dict [ sm ] = replace_txt ( li [ sm ] )
a. append ( temp_dict )
discover_n ame . append ( temp_dict )
else :
a. append ( replace_txt ( li ) )
discover_n ame . append ( replace_txt ( li ) )
else :
a = replace_txt ( template [ m ] )
discover_name = replace_txt ( attr_data )
except Failed :
continue
self . data [ m ] = a
self . data [ m ethod_name ] = discover_n ame
else :
raise Failed ( f " Collection Error: template attribute { m } is blank " )
raise Failed ( f " Collection Error: template attribute { m ethod_name } is blank " )
skip_collection = True
if " schedule " not in methods :
@ -211,21 +211,21 @@ class CollectionBuilder:
else :
raise Failed ( " Collection Error: tmdb_person attribute is blank " )
for m in self . data :
if " tmdb " in m . lower ( ) and not config . TMDb : raise Failed ( f " Collection Error: { m } requires TMDb to be configured " )
elif " trakt " in m . lower ( ) and not config . Trakt : raise Failed ( f " Collection Error: { m } requires Trakt todo be configured " )
elif " imdb " in m . lower ( ) and not config . IMDb : raise Failed ( f " Collection Error: { m } requires TMDb or Trakt to be configured " )
elif " tautulli " in m . lower ( ) and not self . library . Tautulli : raise Failed ( f " Collection Error: { m } requires Tautulli to be configured " )
elif " mal " in m . lower ( ) and not config . MyAnimeList : raise Failed ( f " Collection Error: { m } requires MyAnimeList to be configured " )
elif self . data [ m ] is not None :
for m ethod_name, method_data in self . data . items ( ) :
if " tmdb " in m ethod_name . lower ( ) and not config . TMDb : raise Failed ( f " Collection Error: { m ethod_name } requires TMDb to be configured " )
elif " trakt " in m ethod_name . lower ( ) and not config . Trakt : raise Failed ( f " Collection Error: { m ethod_name } requires Trakt todo be configured " )
elif " imdb " in m ethod_name . lower ( ) and not config . IMDb : raise Failed ( f " Collection Error: { m ethod_name } requires TMDb or Trakt to be configured " )
elif " tautulli " in m ethod_name . lower ( ) and not self . library . Tautulli : raise Failed ( f " Collection Error: { m ethod_name } requires Tautulli to be configured " )
elif " mal " in m ethod_name . lower ( ) and not config . MyAnimeList : raise Failed ( f " Collection Error: { m ethod_name } requires MyAnimeList to be configured " )
elif method_data is not None :
logger . debug ( " " )
logger . debug ( f " Method: { m } " )
logger . debug ( f " Value: { self . data [ m ] } " )
if m . lower ( ) in util . method_alias :
method_name = util . method_alias [ m . lower ( ) ]
logger . warning ( f " Collection Warning: { m } attribute will run as { method_name } " )
logger . debug ( f " Method: { m ethod_name } " )
logger . debug ( f " Value: { method_data } " )
if m ethod_name . lower ( ) in util . method_alias :
method_name = util . method_alias [ m ethod_name . lower ( ) ]
logger . warning ( f " Collection Warning: { m ethod_name } attribute will run as { method_name } " )
else :
method_name = m . lower ( )
method_name = m ethod_name . lower ( )
if method_name in util . show_only_lists and self . library . is_movie :
raise Failed ( f " Collection Error: { method_name } attribute only works for show libraries " )
elif method_name in util . movie_only_lists and self . library . is_show :
@ -235,76 +235,76 @@ class CollectionBuilder:
elif method_name not in util . collectionless_lists and self . collectionless :
raise Failed ( f " Collection Error: { method_name } attribute does not work for Collectionless collection " )
elif method_name == " summary " :
self . summaries [ method_name ] = self . data [ m ]
self . summaries [ method_name ] = method_data
elif method_name == " tmdb_summary " :
self . summaries [ method_name ] = config . TMDb . get_movie_show_or_collection ( util . regex_first_int ( self . data [ m ] , " TMDb ID " ) , self . library . is_movie ) . overview
self . summaries [ method_name ] = config . TMDb . get_movie_show_or_collection ( util . regex_first_int ( method_data , " TMDb ID " ) , self . library . is_movie ) . overview
elif method_name == " tmdb_description " :
self . summaries [ method_name ] = config . TMDb . get_list ( util . regex_first_int ( self . data [ m ] , " TMDb List ID " ) ) . description
self . summaries [ method_name ] = config . TMDb . get_list ( util . regex_first_int ( method_data , " TMDb List ID " ) ) . description
elif method_name == " tmdb_biography " :
self . summaries [ method_name ] = config . TMDb . get_person ( util . regex_first_int ( self . data [ m ] , " TMDb Person ID " ) ) . biography
self . summaries [ method_name ] = config . TMDb . get_person ( util . regex_first_int ( method_data , " TMDb Person ID " ) ) . biography
elif method_name == " tvdb_summary " :
self . summaries [ method_name ] = config . TVDb . get_movie_or_show ( self . data [ m ] , self . library . Plex . language , self . library . is_movie ) . summary
self . summaries [ method_name ] = config . TVDb . get_movie_or_show ( method_data , self . library . Plex . language , self . library . is_movie ) . summary
elif method_name == " tvdb_description " :
self . summaries [ method_name ] = config . TVDb . get_list_description ( self . data [ m ] , self . library . Plex . language )
self . summaries [ method_name ] = config . TVDb . get_list_description ( method_data , self . library . Plex . language )
elif method_name == " trakt_description " :
self . summaries [ method_name ] = config . Trakt . standard_list ( config . Trakt . validate_trakt_list ( util . get_list ( self . data [ m ] ) ) [ 0 ] ) . description
self . summaries [ method_name ] = config . Trakt . standard_list ( config . Trakt . validate_trakt_list ( util . get_list ( method_data ) ) [ 0 ] ) . description
elif method_name == " letterboxd_description " :
self . summaries [ method_name ] = config . Letterboxd . get_list_description ( self . data [ m ] , self . library . Plex . language )
self . summaries [ method_name ] = config . Letterboxd . get_list_description ( method_data , self . library . Plex . language )
elif method_name == " collection_mode " :
if str ( self . data [ m ] ) . lower ( ) == " default " :
if str ( method_data ) . lower ( ) == " default " :
self . details [ method_name ] = " default "
elif str ( self . data [ m ] ) . lower ( ) == " hide " :
elif str ( method_data ) . lower ( ) == " hide " :
self . details [ method_name ] = " hide "
elif str ( self . data [ m ] ) . lower ( ) in [ " hide_items " , " hideitems " ] :
elif str ( method_data ) . lower ( ) in [ " hide_items " , " hideitems " ] :
self . details [ method_name ] = " hideItems "
elif str ( self . data [ m ] ) . lower ( ) in [ " show_items " , " showitems " ] :
elif str ( method_data ) . lower ( ) in [ " show_items " , " showitems " ] :
self . details [ method_name ] = " showItems "
else :
raise Failed ( f " Collection Error: { self . data [ m ] } collection_mode invalid \n \t default (Library default) \n \t hide (Hide Collection) \n \t hide_items (Hide Items in this Collection) \n \t show_items (Show this Collection and its Items) " )
raise Failed ( f " Collection Error: { method_data } collection_mode invalid \n \t default (Library default) \n \t hide (Hide Collection) \n \t hide_items (Hide Items in this Collection) \n \t show_items (Show this Collection and its Items) " )
elif method_name == " collection_order " :
if str ( self . data [ m ] ) . lower ( ) == " release " :
if str ( method_data ) . lower ( ) == " release " :
self . details [ method_name ] = " release "
elif str ( self . data [ m ] ) . lower ( ) == " alpha " :
elif str ( method_data ) . lower ( ) == " alpha " :
self . details [ method_name ] = " release "
else :
raise Failed ( f " Collection Error: { self . data [ m ] } collection_order invalid \n \t release (Order Collection by release dates) \n \t alpha (Order Collection Alphabetically) " )
raise Failed ( f " Collection Error: { method_data } collection_order invalid \n \t release (Order Collection by release dates) \n \t alpha (Order Collection Alphabetically) " )
elif method_name == " url_poster " :
self . posters [ method_name ] = self . data [ m ]
self . posters [ method_name ] = method_data
elif method_name == " tmdb_poster " :
self . posters [ method_name ] = f " { config . TMDb . image_url } { config . TMDb . get_movie_show_or_collection ( util . regex_first_int ( self . data [ m ] , ' TMDb ID ' ) , self . library . is_movie ) . poster_path } "
self . posters [ method_name ] = f " { config . TMDb . image_url } { config . TMDb . get_movie_show_or_collection ( util . regex_first_int ( method_data , ' TMDb ID ' ) , self . library . is_movie ) . poster_path } "
elif method_name == " tmdb_profile " :
self . posters [ method_name ] = f " { config . TMDb . image_url } { config . TMDb . get_person ( util . regex_first_int ( self . data [ m ] , ' TMDb Person ID ' ) ) . profile_path } "
self . posters [ method_name ] = f " { config . TMDb . image_url } { config . TMDb . get_person ( util . regex_first_int ( method_data , ' TMDb Person ID ' ) ) . profile_path } "
elif method_name == " tvdb_poster " :
self . posters [ method_name ] = f " { config . TVDb . get_movie_or_series ( self . data [ m ] , self . library . Plex . language , self . library . is_movie ) . poster_path } "
self . posters [ method_name ] = f " { config . TVDb . get_movie_or_series ( method_data , self . library . Plex . language , self . library . is_movie ) . poster_path } "
elif method_name == " file_poster " :
if os . path . exists ( self . data [ m ] ) :
self . posters [ method_name ] = os . path . abspath ( self . data [ m ] )
if os . path . exists ( method_data ) :
self . posters [ method_name ] = os . path . abspath ( method_data )
else :
raise Failed ( f " Collection Error: Poster Path Does Not Exist: { os . path . abspath ( self . data [ m ] ) } " )
raise Failed ( f " Collection Error: Poster Path Does Not Exist: { os . path . abspath ( method_data ) } " )
elif method_name == " url_background " :
self . backgrounds [ method_name ] = self . data [ m ]
self . backgrounds [ method_name ] = method_data
elif method_name == " tmdb_background " :
self . backgrounds [ method_name ] = f " { config . TMDb . image_url } { config . TMDb . get_movie_show_or_collection ( util . regex_first_int ( self . data [ m ] , ' TMDb ID ' ) , self . library . is_movie ) . poster_path } "
self . backgrounds [ method_name ] = f " { config . TMDb . image_url } { config . TMDb . get_movie_show_or_collection ( util . regex_first_int ( method_data , ' TMDb ID ' ) , self . library . is_movie ) . poster_path } "
elif method_name == " tvdb_background " :
self . posters [ method_name ] = f " { config . TVDb . get_movie_or_series ( self . data [ m ] , self . library . Plex . language , self . library . is_movie ) . background_path } "
self . posters [ method_name ] = f " { config . TVDb . get_movie_or_series ( method_data , self . library . Plex . language , self . library . is_movie ) . background_path } "
elif method_name == " file_background " :
if os . path . exists ( self . data [ m ] ) : self . backgrounds [ method_name ] = os . path . abspath ( self . data [ m ] )
else : raise Failed ( f " Collection Error: Background Path Does Not Exist: { os . path . abspath ( self . data [ m ] ) } " )
if os . path . exists ( method_data ) : self . backgrounds [ method_name ] = os . path . abspath ( method_data )
else : raise Failed ( f " Collection Error: Background Path Does Not Exist: { os . path . abspath ( method_data ) } " )
elif method_name == " label_sync_mode " :
if self . data [ m ] . lower ( ) in [ " append " , " sync " ] : self . details [ method_name ] = self . data [ m ] . lower ( )
if str ( method_data ) . lower ( ) in [ " append " , " sync " ] : self . details [ method_name ] = method_data . lower ( )
else : raise Failed ( " Collection Error: label_sync_mode attribute must be either ' append ' or ' sync ' " )
elif method_name == " sync_mode " :
if self . data [ m ] . lower ( ) in [ " append " , " sync " ] : self . details [ method_name ] = self . data [ m ] . lower ( )
if str ( method_data ) . lower ( ) in [ " append " , " sync " ] : self . details [ method_name ] = method_data . lower ( )
else : raise Failed ( " Collection Error: sync_mode attribute must be either ' append ' or ' sync ' " )
elif method_name in [ " arr_tag " , " label " ] :
self . details [ method_name ] = util . get_list ( self . data [ m ] )
self . details [ method_name ] = util . get_list ( method_data )
elif method_name in util . boolean_details :
if isinstance ( self . data [ m ] , bool ) : self . details [ method_name ] = self . data [ m ]
elif str ( self . data [ m ] ) . lower ( ) in [ " t " , " true " ] : self . details [ method_name ] = True
elif str ( self . data [ m ] ) . lower ( ) in [ " f " , " false " ] : self . details [ method_name ] = False
if isinstance ( method_data , bool ) : self . details [ method_name ] = method_data
elif str ( method_data ) . lower ( ) in [ " t " , " true " ] : self . details [ method_name ] = True
elif str ( method_data ) . lower ( ) in [ " f " , " false " ] : self . details [ method_name ] = False
else : raise Failed ( f " Collection Error: { method_name } attribute must be either true or false " )
elif method_name in util . all_details :
self . details [ method_name ] = self . data [ m ]
self . details [ method_name ] = method_data
elif method_name in [ " year " , " year.not " ] :
self . methods . append ( ( " plex_search " , [ [ ( method_name , util . get_year_list ( self . data [ m ] , method_name ) ) ] ] ) )
elif method_name in [ " decade " , " decade.not " ] :
@ -325,33 +325,33 @@ class CollectionBuilder:
elif method_name == " plex_all " :
self . methods . append ( ( method_name , [ " " ] ) )
elif method_name == " plex_collection " :
self . methods . append ( ( method_name , self . library . validate_collections ( self . data [ m ] if isinstance ( self . data [ m ] , list ) else [ self . data [ m ] ] ) ) )
self . methods . append ( ( method_name , self . library . validate_collections ( method_data if isinstance ( method_data , list ) else [ method_data ] ) ) )
elif method_name == " anidb_popular " :
list_count = util . regex_first_int ( self . data [ m ] , " List Size " , default = 40 )
list_count = util . regex_first_int ( method_data , " List Size " , default = 40 )
if 1 < = list_count < = 30 :
self . methods . append ( ( method_name , [ list_count ] ) )
else :
logger . warning ( " Collection Error: anidb_popular must be an integer between 1 and 30 defaulting to 30 " )
self . methods . append ( ( method_name , [ 30 ] ) )
elif method_name == " mal_id " :
self . methods . append ( ( method_name , util . get_int_list ( self . data [ m ] , " MyAnimeList ID " ) ) )
self . methods . append ( ( method_name , util . get_int_list ( method_data , " MyAnimeList ID " ) ) )
elif method_name in [ " anidb_id " , " anidb_relation " ] :
self . methods . append ( ( method_name , config . AniDB . validate_anidb_list ( util . get_int_list ( self . data [ m ] , " AniDB ID " ) , self . library . Plex . language ) ) )
self . methods . append ( ( method_name , config . AniDB . validate_anidb_list ( util . get_int_list ( method_data , " AniDB ID " ) , self . library . Plex . language ) ) )
elif method_name in [ " anilist_id " , " anilist_relations " , " anilist_studio " ] :
self . methods . append ( ( method_name , config . AniList . validate_anilist_ids ( util . get_int_list ( self . data [ m ] , " AniList ID " ) , studio = method_name == " anilist_studio " ) ) )
self . methods . append ( ( method_name , config . AniList . validate_anilist_ids ( util . get_int_list ( method_data , " AniList ID " ) , studio = method_name == " anilist_studio " ) ) )
elif method_name == " trakt_list " :
self . methods . append ( ( method_name , config . Trakt . validate_trakt_list ( util . get_list ( self . data [ m ] ) ) ) )
self . methods . append ( ( method_name , config . Trakt . validate_trakt_list ( util . get_list ( method_data ) ) ) )
elif method_name == " trakt_list_details " :
valid_list = config . Trakt . validate_trakt_list ( util . get_list ( self . data [ m ] ) )
valid_list = config . Trakt . validate_trakt_list ( util . get_list ( method_data ) )
item = config . Trakt . standard_list ( valid_list [ 0 ] )
if hasattr ( item , " description " ) and item . description :
self . summaries [ method_name ] = item . description
self . methods . append ( ( method_name [ : - 8 ] , valid_list ) )
elif method_name == " trakt_watchlist " :
self . methods . append ( ( method_name , config . Trakt . validate_trakt_watchlist ( util . get_list ( self . data [ m ] ) , self . library . is_movie ) ) )
self . methods . append ( ( method_name , config . Trakt . validate_trakt_watchlist ( util . get_list ( method_data ) , self . library . is_movie ) ) )
elif method_name == " imdb_list " :
new_list = [ ]
for imdb_list in util . get_list ( self . data [ m ] , split = False ) :
for imdb_list in util . get_list ( method_data , split = False ) :
if isinstance ( imdb_list , dict ) :
dict_methods = { dm . lower ( ) : dm for dm in imdb_list }
if " url " in dict_methods and imdb_list [ dict_methods [ " url " ] ] :
@ -365,13 +365,13 @@ class CollectionBuilder:
new_list . append ( { " url " : imdb_url , " limit " : list_count } )
self . methods . append ( ( method_name , new_list ) )
elif method_name == " letterboxd_list " :
self . methods . append ( ( method_name , util . get_list ( self . data [ m ] , split = False ) ) )
self . methods . append ( ( method_name , util . get_list ( method_data , split = False ) ) )
elif method_name == " letterboxd_list_details " :
values = util . get_list ( self . data [ m ] , split = False )
values = util . get_list ( method_data , split = False )
self . summaries [ method_name ] = config . Letterboxd . get_list_description ( values [ 0 ] , self . library . Plex . language )
self . methods . append ( ( method_name [ : - 8 ] , values ) )
elif method_name in util . dictionary_lists :
if isinstance ( self . data [ m ] , dict ) :
if isinstance ( method_data , dict ) :
def get_int ( parent , method , data_in , methods_in , default_in , minimum = 1 , maximum = None ) :
if method not in methods_in :
logger . warning ( f " Collection Warning: { parent } { methods_in [ method ] } attribute not found using { default_in } as default " )
@ -386,50 +386,50 @@ class CollectionBuilder:
logger . warning ( f " Collection Warning: { parent } { methods_in [ method ] } attribute { data_in [ methods_in [ method ] ] } invalid must an integer >= { minimum } using { default_in } as default " )
return default_in
if method_name == " filters " :
for f in self . data [ m ] :
if f . lower ( ) in util . method_alias or ( f . lower ( ) . endswith ( " .not " ) and f . lower ( ) [ : - 4 ] in util . method_alias ) :
filter_method = ( util . method_alias [ f . lower ( ) [ : - 4 ] ] + f . lower ( ) [ - 4 : ] ) if f . lower ( ) . endswith ( " .not " ) else util . method_alias [ f . lower ( ) ]
logger . warning ( f " Collection Warning: { f } filter will run as { filter_method } " )
for f ilter_name, filter_data in method_data . items ( ) :
if f ilter_name . lower ( ) in util . method_alias or ( f ilter_name . lower ( ) . endswith ( " .not " ) and f ilter_name . lower ( ) [ : - 4 ] in util . method_alias ) :
filter_method = ( util . method_alias [ f ilter_name . lower ( ) [ : - 4 ] ] + f ilter_name . lower ( ) [ - 4 : ] ) if f ilter_name . lower ( ) . endswith ( " .not " ) else util . method_alias [ f ilter_name . lower ( ) ]
logger . warning ( f " Collection Warning: { f ilter_name } filter will run as { filter_method } " )
else :
filter_method = f . lower ( )
filter_method = f ilter_name . lower ( )
if filter_method in util . movie_only_filters and self . library . is_show :
raise Failed ( f " Collection Error: { filter_method } filter only works for movie libraries " )
elif self . data [ m ] [ f ] is None :
elif filter_data is None :
raise Failed ( f " Collection Error: { filter_method } filter is blank " )
elif filter_method == " year " :
filter_data = util . get_year_list ( self . data [ m ] [ f ] , f " { filter_method } filter " )
valid_data = util . get_year_list ( filter_data , current_year , f " { filter_method } filter " )
elif filter_method in [ " max_age " , " duration.gte " , " duration.lte " , " tmdb_vote_count.gte " , " tmdb_vote_count.lte " ] :
filter_data = util . check_number ( self . data [ m ] [ f ] , f " { filter_method } filter " , minimum = 1 )
valid_data = util . check_number ( filter_data , f " { filter_method } filter " , minimum = 1 )
elif filter_method in [ " year.gte " , " year.lte " ] :
filter_data = util . check_number ( self . data [ m ] [ f ] , f " { filter_method } filter " , minimum = 1800 , maximum = current_year )
valid_data = util . check_year ( filter_data , current_year , f " { filter_method } filter " )
elif filter_method in [ " rating.gte " , " rating.lte " ] :
filter_data = util . check_number ( self . data [ m ] [ f ] , f " { filter_method } filter " , number_type = " float " , minimum = 0.1 , maximum = 10 )
valid_data = util . check_number ( filter_data , f " { filter_method } filter " , number_type = " float " , minimum = 0.1 , maximum = 10 )
elif filter_method in [ " originally_available.gte " , " originally_available.lte " ] :
filter_data = util . check_date ( self . data [ m ] [ f ] , f " { filter_method } filter " )
valid_data = util . check_date ( filter_data , f " { filter_method } filter " )
elif filter_method == " original_language " :
filter_data = util . get_list ( self . data [ m ] [ f ] , lower = True )
valid_data = util . get_list ( filter_data , lower = True )
elif filter_method == " collection " :
filter_data = self . data [ m ] [ f ] if isinstance ( self . data [ m ] [ f ] , list ) else [ self . data [ m ] [ f ] ]
valid_data = filter_data if isinstance ( filter_data , list ) else [ filter_data ]
elif filter_method in util . all_filters :
filter_data = util . get_list ( self . data [ m ] [ f ] )
valid_data = util . get_list ( filter_data )
else :
raise Failed ( f " Collection Error: { filter_method } filter not supported " )
self . filters . append ( ( filter_method , filter _data) )
self . filters . append ( ( filter_method , valid _data) )
elif method_name == " plex_collectionless " :
new_dictionary = { }
dict_methods = { dm . lower ( ) : dm for dm in self . data [ m ] }
dict_methods = { dm . lower ( ) : dm for dm in method_data }
prefix_list = [ ]
if " exclude_prefix " in dict_methods and self . data [ m ] [ dict_methods [ " exclude_prefix " ] ] :
if isinstance ( self . data [ m ] [ dict_methods [ " exclude_prefix " ] ] , list ) :
prefix_list . extend ( self . data [ m ] [ dict_methods [ " exclude_prefix " ] ] )
if " exclude_prefix " in dict_methods and method_data [ dict_methods [ " exclude_prefix " ] ] :
if isinstance ( method_data [ dict_methods [ " exclude_prefix " ] ] , list ) :
prefix_list . extend ( method_data [ dict_methods [ " exclude_prefix " ] ] )
else :
prefix_list . append ( str ( self . data [ m ] [ dict_methods [ " exclude_prefix " ] ] ) )
prefix_list . append ( str ( method_data [ dict_methods [ " exclude_prefix " ] ] ) )
exact_list = [ ]
if " exclude " in dict_methods and self . data [ m ] [ dict_methods [ " exclude " ] ] :
if isinstance ( self . data [ m ] [ dict_methods [ " exclude " ] ] , list ) :
exact_list . extend ( self . data [ m ] [ dict_methods [ " exclude " ] ] )
if " exclude " in dict_methods and method_data [ dict_methods [ " exclude " ] ] :
if isinstance ( method_data [ dict_methods [ " exclude " ] ] , list ) :
exact_list . extend ( method_data [ dict_methods [ " exclude " ] ] )
else :
exact_list . append ( str ( self . data [ m ] [ dict_methods [ " exclude " ] ] ) )
exact_list . append ( str ( method_data [ dict_methods [ " exclude " ] ] ) )
if len ( prefix_list ) == 0 and len ( exact_list ) == 0 :
raise Failed ( " Collection Error: you must have at least one exclusion " )
new_dictionary [ " exclude_prefix " ] = prefix_list
@ -464,61 +464,61 @@ class CollectionBuilder:
self . methods . append ( ( method_name , [ searches ] ) )
elif method_name == " tmdb_discover " :
new_dictionary = { " limit " : 100 }
for a in self . data [ m ] :
a_name = a . lower ( )
if self . data [ m ] [ a ] :
if ( self . library . is_movie and a_name in util . discover_movie ) or ( self . library . is_show and a_name in util . discover_tv ) :
if a_name == " language " :
if re . compile ( " ([a-z] {2} )-([A-Z] {2} ) " ) . match ( str ( self . data [ m ] [ a ] ) ) :
new_dictionary [ a_name] = str ( self . data [ m ] [ a ] )
for discover_name, discover_data in method_data :
discover_final = discover_name . lower ( )
if discover_data :
if ( self . library . is_movie and discover_final in util . discover_movie ) or ( self . library . is_show and discover_final in util . discover_tv ) :
if discover_final == " language " :
if re . compile ( " ([a-z] {2} )-([A-Z] {2} ) " ) . match ( str ( discover_data ) ) :
new_dictionary [ discover_final] = str ( discover_data )
else :
raise Failed ( f " Collection Error: { m } attribute { a_name } : { self . data [ m ] [ a ] } must match pattern ([a-z] {{ 2 }} )-([A-Z] {{ 2 }} ) e.g. en-US " )
elif a_name == " region " :
if re . compile ( " ^[A-Z] {2} $ " ) . match ( str ( self . data [ m ] [ a ] ) ) :
new_dictionary [ a_name] = str ( self . data [ m ] [ a ] )
raise Failed ( f " Collection Error: { m ethod_name} attribute { discover_final } : { discover_data } must match pattern ([a-z] {{ 2 }} )-([A-Z] {{ 2 }} ) e.g. en-US " )
elif discover_final == " region " :
if re . compile ( " ^[A-Z] {2} $ " ) . match ( str ( discover_data ) ) :
new_dictionary [ discover_final] = str ( discover_data )
else :
raise Failed ( f " Collection Error: { m } attribute { a_name } : { self . data [ m ] [ a ] } must match pattern ^[A-Z] {{ 2 }} $ e.g. US " )
elif a_name == " sort_by " :
if ( self . library . is_movie and self . data [ m ] [ a ] in util . discover_movie_sort ) or ( self . library . is_show and self . data [ m ] [ a ] in util . discover_tv_sort ) :
new_dictionary [ a_name] = self . data [ m ] [ a ]
raise Failed ( f " Collection Error: { m ethod_name} attribute { discover_final } : { discover_data } must match pattern ^[A-Z] {{ 2 }} $ e.g. US " )
elif discover_final == " sort_by " :
if ( self . library . is_movie and discover_data in util . discover_movie_sort ) or ( self . library . is_show and discover_data in util . discover_tv_sort ) :
new_dictionary [ discover_final] = discover_data
else :
raise Failed ( f " Collection Error: { m } attribute { a_name } : { self . data [ m ] [ a ] } is invalid " )
elif a_name == " certification_country " :
if " certification " in self . data [ m ] or " certification.lte " in self . data [ m ] or " certification.gte " in self . data [ m ] :
new_dictionary [ a_name] = self . data [ m ] [ a ]
raise Failed ( f " Collection Error: { m ethod_name} attribute { discover_final } : { discover_data } is invalid " )
elif discover_final == " certification_country " :
if " certification " in method_data or " certification.lte " in method_data or " certification.gte " in method_data :
new_dictionary [ discover_final] = discover_data
else :
raise Failed ( f " Collection Error: { m } attribute { a_name } : must be used with either certification, certification.lte, or certification.gte " )
elif a_name in [ " certification " , " certification.lte " , " certification.gte " ] :
if " certification_country " in self . data [ m ] :
new_dictionary [ a_name] = self . data [ m ] [ a ]
raise Failed ( f " Collection Error: { m ethod_name} attribute { discover_final } : must be used with either certification, certification.lte, or certification.gte " )
elif discover_final in [ " certification " , " certification.lte " , " certification.gte " ] :
if " certification_country " in method_data :
new_dictionary [ discover_final] = discover_data
else :
raise Failed ( f " Collection Error: { m } attribute { a_name } : must be used with certification_country " )
elif a_name in [ " include_adult " , " include_null_first_air_dates " , " screened_theatrically " ] :
if self . data [ m ] [ a ] is True :
new_dictionary [ a_name] = self . data [ m ] [ a ]
elif a_name in util . discover_dates :
new_dictionary [ a_name] = util . check_date ( self . data [ m ] [ a ] , f " { m } attribute { a_name } " , return_string = True )
elif a_name in [ " primary_release_year " , " year " , " first_air_date_year " ] :
new_dictionary [ a_name] = util . check_number ( self . data [ m ] [ a ] , f " { m } attribute { a_name } " , minimum = 1800 , maximum = current_year + 1 )
elif a_name in [ " vote_count.gte " , " vote_count.lte " , " vote_average.gte " , " vote_average.lte " , " with_runtime.gte " , " with_runtime.lte " ] :
new_dictionary [ a_name] = util . check_number ( self . data [ m ] [ a ] , f " { m } attribute { a_name } " , minimum = 1 )
elif a_name in [ " with_cast " , " with_crew " , " with_people " , " with_companies " , " with_networks " , " with_genres " , " without_genres " , " with_keywords " , " without_keywords " , " with_original_language " , " timezone " ] :
new_dictionary [ a_name] = self . data [ m ] [ a ]
raise Failed ( f " Collection Error: { m ethod_name} attribute { discover_final } : must be used with certification_country " )
elif discover_final in [ " include_adult " , " include_null_first_air_dates " , " screened_theatrically " ] :
if discover_data is True :
new_dictionary [ discover_final] = discover_data
elif discover_final in util . discover_dates :
new_dictionary [ discover_final] = util . check_date ( discover_data , f " { method_name } attribute { discover_final } " , return_string = True )
elif discover_final in [ " primary_release_year " , " year " , " first_air_date_year " ] :
new_dictionary [ discover_final] = util . check_number ( discover_data , f " { method_name } attribute { discover_final } " , minimum = 1800 , maximum = current_year + 1 )
elif discover_final in [ " vote_count.gte " , " vote_count.lte " , " vote_average.gte " , " vote_average.lte " , " with_runtime.gte " , " with_runtime.lte " ] :
new_dictionary [ discover_final] = util . check_number ( discover_data , f " { method_name } attribute { discover_final } " , minimum = 1 )
elif discover_final in [ " with_cast " , " with_crew " , " with_people " , " with_companies " , " with_networks " , " with_genres " , " without_genres " , " with_keywords " , " without_keywords " , " with_original_language " , " timezone " ] :
new_dictionary [ discover_final] = discover_data
else :
raise Failed ( f " Collection Error: { m } attribute { a_name } not supported " )
elif a_name == " limit " :
if isinstance ( self . data [ m ] [ a ] , int ) and self . data [ m ] [ a ] > 0 :
new_dictionary [ a_name] = self . data [ m ] [ a ]
raise Failed ( f " Collection Error: { m ethod_name} attribute { discover_final } not supported " )
elif discover_final == " limit " :
if isinstance ( discover_data , int ) and discover_data > 0 :
new_dictionary [ discover_final] = discover_data
else :
raise Failed ( f " Collection Error: { m } attribute { a_name } : must be a valid number greater then 0 " )
raise Failed ( f " Collection Error: { m ethod_name} attribute { discover_final } : must be a valid number greater then 0 " )
else :
raise Failed ( f " Collection Error: { m } attribute { a_name } not supported " )
raise Failed ( f " Collection Error: { m ethod_name} attribute { discover_final } not supported " )
else :
raise Failed ( f " Collection Error: { m } parameter { a_name } is blank " )
raise Failed ( f " Collection Error: { m ethod_name} parameter { discover_final } is blank " )
if len ( new_dictionary ) > 1 :
self . methods . append ( ( method_name , [ new_dictionary ] ) )
else :
raise Failed ( f " Collection Error: { m } had no valid fields " )
raise Failed ( f " Collection Error: { m ethod_name } had no valid fields " )
elif " tautulli " in method_name :
new_dictionary = { }
if method_name == " tautulli_popular " :
@ -527,22 +527,22 @@ class CollectionBuilder:
new_dictionary [ " list_type " ] = " watched "
else :
raise Failed ( f " Collection Error: { method_name } attribute not supported " )
dict_methods = { dm . lower ( ) : dm for dm in self . data [ m ] }
new_dictionary [ " list_days " ] = get_int ( method_name , " list_days " , self . data [ m ] , dict_methods , 30 )
new_dictionary [ " list_size " ] = get_int ( method_name , " list_size " , self . data [ m ] , dict_methods , 10 )
new_dictionary [ " list_buffer " ] = get_int ( method_name , " list_buffer " , self . data [ m ] , dict_methods , 20 )
dict_methods = { dm . lower ( ) : dm for dm in method_data }
new_dictionary [ " list_days " ] = get_int ( method_name , " list_days " , method_data , dict_methods , 30 )
new_dictionary [ " list_size " ] = get_int ( method_name , " list_size " , method_data , dict_methods , 10 )
new_dictionary [ " list_buffer " ] = get_int ( method_name , " list_buffer " , method_data , dict_methods , 20 )
self . methods . append ( ( method_name , [ new_dictionary ] ) )
elif method_name == " mal_season " :
new_dictionary = { " sort_by " : " anime_num_list_users " }
dict_methods = { dm . lower ( ) : dm for dm in self . data [ m ] }
dict_methods = { dm . lower ( ) : dm for dm in method_data }
if " sort_by " not in dict_methods :
logger . warning ( " Collection Warning: mal_season sort_by attribute not found using members as default " )
elif not self . data [ m ] [ dict_methods [ " sort_by " ] ] :
elif not method_data [ dict_methods [ " sort_by " ] ] :
logger . warning ( " Collection Warning: mal_season sort_by attribute is blank using members as default " )
elif self . data [ m ] [ dict_methods [ " sort_by " ] ] not in util . mal_season_sort :
logger . warning ( f " Collection Warning: mal_season sort_by attribute { self . data [ m ] [ dict_methods [ ' sort_by ' ] ] } invalid must be either ' members ' or ' score ' using members as default " )
elif method_data [ dict_methods [ " sort_by " ] ] not in util . mal_season_sort :
logger . warning ( f " Collection Warning: mal_season sort_by attribute { method_data [ dict_methods [ ' sort_by ' ] ] } invalid must be either ' members ' or ' score ' using members as default " )
else :
new_dictionary [ " sort_by " ] = util . mal_season_sort [ self . data [ m ] [ dict_methods [ " sort_by " ] ] ]
new_dictionary [ " sort_by " ] = util . mal_season_sort [ method_data [ dict_methods [ " sort_by " ] ] ]
if current_time . month in [ 1 , 2 , 3 ] : new_dictionary [ " season " ] = " winter "
elif current_time . month in [ 4 , 5 , 6 ] : new_dictionary [ " season " ] = " spring "
@ -551,49 +551,49 @@ class CollectionBuilder:
if " season " not in dict_methods :
logger . warning ( f " Collection Warning: mal_season season attribute not found using the current season: { new_dictionary [ ' season ' ] } as default " )
elif not self . data [ m ] [ dict_methods [ " season " ] ] :
elif not method_data [ dict_methods [ " season " ] ] :
logger . warning ( f " Collection Warning: mal_season season attribute is blank using the current season: { new_dictionary [ ' season ' ] } as default " )
elif self . data [ m ] [ dict_methods [ " season " ] ] not in util . pretty_seasons :
logger . warning ( f " Collection Warning: mal_season season attribute { self . data [ m ] [ dict_methods [ ' season ' ] ] } invalid must be either ' winter ' , ' spring ' , ' summer ' or ' fall ' using the current season: { new_dictionary [ ' season ' ] } as default " )
elif method_data [ dict_methods [ " season " ] ] not in util . pretty_seasons :
logger . warning ( f " Collection Warning: mal_season season attribute { method_data [ dict_methods [ ' season ' ] ] } invalid must be either ' winter ' , ' spring ' , ' summer ' or ' fall ' using the current season: { new_dictionary [ ' season ' ] } as default " )
else :
new_dictionary [ " season " ] = self . data [ m ] [ dict_methods [ " season " ] ]
new_dictionary [ " season " ] = method_data [ dict_methods [ " season " ] ]
new_dictionary [ " year " ] = get_int ( method_name , " year " , self . data [ m ] , dict_methods , current_time . year , minimum = 1917 , maximum = current_time . year + 1 )
new_dictionary [ " limit " ] = get_int ( method_name , " limit " , self . data [ m ] , dict_methods , 100 , maximum = 500 )
new_dictionary [ " year " ] = get_int ( method_name , " year " , method_data , dict_methods , current_time . year , minimum = 1917 , maximum = current_time . year + 1 )
new_dictionary [ " limit " ] = get_int ( method_name , " limit " , method_data , dict_methods , 100 , maximum = 500 )
self . methods . append ( ( method_name , [ new_dictionary ] ) )
elif method_name == " mal_userlist " :
new_dictionary = { " status " : " all " , " sort_by " : " list_score " }
dict_methods = { dm . lower ( ) : dm for dm in self . data [ m ] }
dict_methods = { dm . lower ( ) : dm for dm in method_data }
if " username " not in dict_methods :
raise Failed ( " Collection Error: mal_userlist username attribute is required " )
elif not self . data [ m ] [ dict_methods [ " username " ] ] :
elif not method_data [ dict_methods [ " username " ] ] :
raise Failed ( " Collection Error: mal_userlist username attribute is blank " )
else :
new_dictionary [ " username " ] = self . data [ m ] [ dict_methods [ " username " ] ]
new_dictionary [ " username " ] = method_data [ dict_methods [ " username " ] ]
if " status " not in dict_methods :
logger . warning ( " Collection Warning: mal_season status attribute not found using all as default " )
elif not self . data [ m ] [ dict_methods [ " status " ] ] :
elif not method_data [ dict_methods [ " status " ] ] :
logger . warning ( " Collection Warning: mal_season status attribute is blank using all as default " )
elif self . data [ m ] [ dict_methods [ " status " ] ] not in util . mal_userlist_status :
logger . warning ( f " Collection Warning: mal_season status attribute { self . data [ m ] [ dict_methods [ ' status ' ] ] } invalid must be either ' all ' , ' watching ' , ' completed ' , ' on_hold ' , ' dropped ' or ' plan_to_watch ' using all as default " )
elif method_data [ dict_methods [ " status " ] ] not in util . mal_userlist_status :
logger . warning ( f " Collection Warning: mal_season status attribute { method_data [ dict_methods [ ' status ' ] ] } invalid must be either ' all ' , ' watching ' , ' completed ' , ' on_hold ' , ' dropped ' or ' plan_to_watch ' using all as default " )
else :
new_dictionary [ " status " ] = util . mal_userlist_status [ self . data [ m ] [ dict_methods [ " status " ] ] ]
new_dictionary [ " status " ] = util . mal_userlist_status [ method_data [ dict_methods [ " status " ] ] ]
if " sort_by " not in dict_methods :
logger . warning ( " Collection Warning: mal_season sort_by attribute not found using score as default " )
elif not self . data [ m ] [ dict_methods [ " sort_by " ] ] :
elif not method_data [ dict_methods [ " sort_by " ] ] :
logger . warning ( " Collection Warning: mal_season sort_by attribute is blank using score as default " )
elif self . data [ m ] [ dict_methods [ " sort_by " ] ] not in util . mal_userlist_sort :
logger . warning ( f " Collection Warning: mal_season sort_by attribute { self . data [ m ] [ dict_methods [ ' sort_by ' ] ] } invalid must be either ' score ' , ' last_updated ' , ' title ' or ' start_date ' using score as default " )
elif method_data [ dict_methods [ " sort_by " ] ] not in util . mal_userlist_sort :
logger . warning ( f " Collection Warning: mal_season sort_by attribute { method_data [ dict_methods [ ' sort_by ' ] ] } invalid must be either ' score ' , ' last_updated ' , ' title ' or ' start_date ' using score as default " )
else :
new_dictionary [ " sort_by " ] = util . mal_userlist_sort [ self . data [ m ] [ dict_methods [ " sort_by " ] ] ]
new_dictionary [ " sort_by " ] = util . mal_userlist_sort [ method_data [ dict_methods [ " sort_by " ] ] ]
new_dictionary [ " limit " ] = get_int ( method_name , " limit " , self . data [ m ] , dict_methods , 100 , maximum = 1000 )
new_dictionary [ " limit " ] = get_int ( method_name , " limit " , method_data , dict_methods , 100 , maximum = 1000 )
self . methods . append ( ( method_name , [ new_dictionary ] ) )
elif " anilist " in method_name :
new_dictionary = { " sort_by " : " score " }
dict_methods = { dm . lower ( ) : dm for dm in self . data [ m ] }
dict_methods = { dm . lower ( ) : dm for dm in method_data }
if method_name == " anilist_season " :
if current_time . month in [ 12 , 1 , 2 ] : new_dictionary [ " season " ] = " winter "
elif current_time . month in [ 3 , 4 , 5 ] : new_dictionary [ " season " ] = " spring "
@ -602,51 +602,51 @@ class CollectionBuilder:
if " season " not in dict_methods :
logger . warning ( f " Collection Warning: anilist_season season attribute not found using the current season: { new_dictionary [ ' season ' ] } as default " )
elif not self . data [ m ] [ dict_methods [ " season " ] ] :
elif not method_data [ dict_methods [ " season " ] ] :
logger . warning ( f " Collection Warning: anilist_season season attribute is blank using the current season: { new_dictionary [ ' season ' ] } as default " )
elif self . data [ m ] [ dict_methods [ " season " ] ] not in util . pretty_seasons :
logger . warning ( f " Collection Warning: anilist_season season attribute { self . data [ m ] [ dict_methods [ ' season ' ] ] } invalid must be either ' winter ' , ' spring ' , ' summer ' or ' fall ' using the current season: { new_dictionary [ ' season ' ] } as default " )
elif method_data [ dict_methods [ " season " ] ] not in util . pretty_seasons :
logger . warning ( f " Collection Warning: anilist_season season attribute { method_data [ dict_methods [ ' season ' ] ] } invalid must be either ' winter ' , ' spring ' , ' summer ' or ' fall ' using the current season: { new_dictionary [ ' season ' ] } as default " )
else :
new_dictionary [ " season " ] = self . data [ m ] [ dict_methods [ " season " ] ]
new_dictionary [ " season " ] = method_data [ dict_methods [ " season " ] ]
new_dictionary [ " year " ] = get_int ( method_name , " year " , self . data [ m ] , dict_methods , current_time . year , minimum = 1917 , maximum = current_time . year + 1 )
new_dictionary [ " year " ] = get_int ( method_name , " year " , method_data , dict_methods , current_time . year , minimum = 1917 , maximum = current_time . year + 1 )
elif method_name == " anilist_genre " :
if " genre " not in dict_methods :
raise Failed ( f " Collection Warning: anilist_genre genre attribute not found " )
elif not self . data [ m ] [ dict_methods [ " genre " ] ] :
elif not method_data [ dict_methods [ " genre " ] ] :
raise Failed ( f " Collection Warning: anilist_genre genre attribute is blank " )
else :
new_dictionary [ " genre " ] = self . config . AniList . validate_genre ( self . data [ m ] [ dict_methods [ " genre " ] ] )
new_dictionary [ " genre " ] = self . config . AniList . validate_genre ( method_data [ dict_methods [ " genre " ] ] )
elif method_name == " anilist_tag " :
if " tag " not in dict_methods :
raise Failed ( f " Collection Warning: anilist_tag tag attribute not found " )
elif not self . data [ m ] [ dict_methods [ " tag " ] ] :
elif not method_data [ dict_methods [ " tag " ] ] :
raise Failed ( f " Collection Warning: anilist_tag tag attribute is blank " )
else :
new_dictionary [ " tag " ] = self . config . AniList . validate_tag ( self . data [ m ] [ dict_methods [ " tag " ] ] )
new_dictionary [ " tag " ] = self . config . AniList . validate_tag ( method_data [ dict_methods [ " tag " ] ] )
if " sort_by " not in dict_methods :
logger . warning ( f " Collection Warning: { method_name } sort_by attribute not found using score as default " )
elif not self . data [ m ] [ dict_methods [ " sort_by " ] ] :
elif not method_data [ dict_methods [ " sort_by " ] ] :
logger . warning ( f " Collection Warning: { method_name } sort_by attribute is blank using score as default " )
elif str ( self . data [ m ] [ dict_methods [ " sort_by " ] ] ) . lower ( ) not in [ " score " , " popular " ] :
logger . warning ( f " Collection Warning: { method_name } sort_by attribute { self . data [ m ] [ dict_methods [ ' sort_by ' ] ] } invalid must be either ' score ' or ' popular ' using score as default " )
elif str ( method_data [ dict_methods [ " sort_by " ] ] ) . lower ( ) not in [ " score " , " popular " ] :
logger . warning ( f " Collection Warning: { method_name } sort_by attribute { method_data [ dict_methods [ ' sort_by ' ] ] } invalid must be either ' score ' or ' popular ' using score as default " )
else :
new_dictionary [ " sort_by " ] = self . data [ m ] [ dict_methods [ " sort_by " ] ]
new_dictionary [ " sort_by " ] = method_data [ dict_methods [ " sort_by " ] ]
new_dictionary [ " limit " ] = get_int ( method_name , " limit " , self . data [ m ] , dict_methods , 0 , maximum = 500 )
new_dictionary [ " limit " ] = get_int ( method_name , " limit " , method_data , dict_methods , 0 , maximum = 500 )
self . methods . append ( ( method_name , [ new_dictionary ] ) )
else :
raise Failed ( f " Collection Error: { m } attribute is not a dictionary: { self . data [ m ] } " )
raise Failed ( f " Collection Error: { m ethod_name } attribute is not a dictionary: { method_data } " )
elif method_name in util . count_lists :
list_count = util . regex_first_int ( self . data [ m ] , " List Size " , default = 10 )
list_count = util . regex_first_int ( method_data , " List Size " , default = 10 )
if list_count < 1 :
logger . warning ( f " Collection Warning: { method_name } must be an integer greater then 0 defaulting to 10 " )
list_count = 10
self . methods . append ( ( method_name , [ list_count ] ) )
elif " tvdb " in method_name :
values = util . get_list ( self . data [ m ] )
values = util . get_list ( method_data )
if method_name [ - 8 : ] == " _details " :
if method_name == " tvdb_movie_details " :
item = config . TVDb . get_movie ( self . library . Plex . language , values [ 0 ] )
@ -670,7 +670,7 @@ class CollectionBuilder:
else :
self . methods . append ( ( method_name , values ) )
elif method_name in util . tmdb_lists :
values = config . TMDb . validate_tmdb_list ( util . get_int_list ( self . data [ m ] , f " TMDb { util . tmdb_type [ method_name ] } ID " ) , util . tmdb_type [ method_name ] )
values = config . TMDb . validate_tmdb_list ( util . get_int_list ( method_data , f " TMDb { util . tmdb_type [ method_name ] } ID " ) , util . tmdb_type [ method_name ] )
if method_name [ - 8 : ] == " _details " :
if method_name in [ " tmdb_collection_details " , " tmdb_movie_details " , " tmdb_show_details " ] :
item = config . TMDb . get_movie_show_or_collection ( values [ 0 ] , self . library . is_movie )
@ -694,13 +694,13 @@ class CollectionBuilder:
else :
self . methods . append ( ( method_name , values ) )
elif method_name in util . all_lists :
self . methods . append ( ( method_name , util . get_list ( self . data [ m ] ) ) )
self . methods . append ( ( method_name , util . get_list ( method_data ) ) )
elif method_name not in util . other_attributes :
raise Failed ( f " Collection Error: { method_name } attribute not supported " )
elif m in util . all_lists or m in util . method_alias or m in util . plex_searches :
raise Failed ( f " Collection Error: { m } attribute is blank " )
elif m ethod_name in util . all_lists or m ethod_name in util . method_alias or m ethod_name in util . plex_searches :
raise Failed ( f " Collection Error: { m ethod_name } attribute is blank " )
else :
logger . warning ( f " Collection Warning: { m } attribute is blank " )
logger . warning ( f " Collection Warning: { m ethod_name } attribute is blank " )
self . sync = self . library . sync_mode == " sync "
if " sync_mode " in methods :