Files
addon/specials/classicsearch.py
2021-11-26 18:55:51 +01:00

575 lines
20 KiB
Python

# -*- coding: utf-8 -*-
# -*- Channel New Search -*-
# -*- Created for Alfa-addon -*-
# -*- By the Alfa Develop Group -*-
from __future__ import division
from builtins import range
from core import support
from past.utils import old_div
#from builtins import str
import sys
PY3 = False
if sys.version_info[0] >= 3: PY3 = True; unicode = str; unichr = chr; long = int
import time, channelselector
if PY3:
from concurrent import futures
else:
from concurrent_py2 import futures
from core.item import Item
from core import tmdb, scrapertools, channeltools, filetools
from platformcode import logger, config, platformtools, unify
from core.support import typo, thumb
from specials.search import channel_selections, set_context, save_search
info_language = ["de", "en", "es", "fr", "it", "pt"] # from videolibrary.json
def_lang = info_language[config.getSetting("info_language", "videolibrary")]
def new_search(item):
logger.debug()
temp_search_file = config.getTempFile('temp-search')
if filetools.isfile(temp_search_file):
filetools.remove(temp_search_file)
itemlist = []
if config.getSetting('last_search'):
last_search = channeltools.getChannelSetting('Last_searched', 'search', '')
else:
last_search = ''
if item.search_text:
searched_text = item.search_text
else:
searched_text = platformtools.dialogInput(default=last_search, heading='')
save_search(searched_text)
if not searched_text:
return
channeltools.setChannelSetting('Last_searched', searched_text, 'search')
searched_text = searched_text.replace("+", " ")
if item.mode == 'person':
item.searched_text = searched_text
return actor_list(item)
if item.mode != 'all':
tmdb_info = tmdb.Tmdb(searched_text=searched_text, search_type=item.mode.replace('show', ''))
results = tmdb_info.results
for result in results:
result = tmdb_info.get_infoLabels(result, origen=result)
if item.mode == 'movie':
title = result['title']
else:
title = result['name']
item.mode = 'tvshow'
thumbnail = result.get('thumbnail', '')
fanart = result.get('fanart', '')
new_item = Item(channel=item.channel,
action='channel_search',
title=title,
text=searched_text,
thumbnail=thumbnail,
fanart=fanart,
mode=item.mode,
contentType=item.mode,
infoLabels=result)
if item.mode == 'movie':
new_item.contentTitle = result['title']
else:
new_item.contentSerieName = result['name']
itemlist.append(new_item)
if item.mode == 'all' or not itemlist:
return channel_search(Item(channel=item.channel,
title=searched_text,
text=searched_text,
mode='all',
infoLabels={}))
return itemlist
def channel_search(item):
logger.debug(item)
start = time.time()
searching = list()
searching_titles = list()
results = list()
valid = list()
ch_list = dict()
mode = item.mode
if item.infoLabels['tvshowtitle']:
item.text = item.infoLabels['tvshowtitle'].split(' - ')[0]
item.title = item.text
elif item.infoLabels['title']:
item.text = item.infoLabels['title'].split(' - ')[0]
item.title = item.text
temp_search_file = config.getTempFile('temp-search')
if filetools.isfile(temp_search_file):
itemlist = []
f = filetools.read(temp_search_file)
if f.startswith(item.text):
for it in f.split(','):
if it and it != item.text:
itemlist.append(Item().fromurl(it))
return itemlist
else:
filetools.remove(temp_search_file)
searched_id = item.infoLabels['tmdb_id']
channel_list, channel_titles = get_channels(item)
searching += channel_list
searching_titles += channel_titles
cnt = 0
progress = platformtools.dialogProgress(config.getLocalizedString(30993) % item.title, config.getLocalizedString(70744) % len(channel_list) + '\n' + ', '.join(searching_titles))
config.setSetting('tmdb_active', False)
search_action_list = []
module_dict = {}
for ch in channel_list:
try:
module = platformtools.channelImport(ch)
mainlist = getattr(module, 'mainlist')(Item(channel=ch, global_search=True))
module_dict[ch] = module
search_action_list.extend([elem for elem in mainlist if
elem.action == "search" and (mode == 'all' or elem.contentType in [mode, 'undefined'])])
if progress.iscanceled():
return []
except:
import traceback
logger.error('error importing/getting search items of ' + ch)
logger.error(traceback.format_exc())
total_search_actions = len(search_action_list)
with futures.ThreadPoolExecutor(max_workers=set_workers()) as executor:
c_results = []
for search_action in search_action_list:
c_results.append(executor.submit(get_channel_results, item, module_dict, search_action))
if progress.iscanceled():
break
for res in futures.as_completed(c_results):
search_action = res.result()[0]
channel = search_action.channel
if res.result()[1]:
if channel not in ch_list:
ch_list[channel] = []
ch_list[channel].extend(res.result()[1])
if res.result()[2]:
valid.extend(res.result()[2])
if progress.iscanceled():
break
search_action_list.remove(search_action)
# if no action of this channel remains
for it in search_action_list:
if it.channel == channel:
break
else:
cnt += 1
searching_titles.remove(searching_titles[searching.index(channel)])
searching.remove(channel)
progress.update(old_div(((total_search_actions - len(search_action_list)) * 100), total_search_actions), config.getLocalizedString(70744) % str(len(channel_list) - cnt) + '\n' + ', '.join(searching_titles))
progress.close()
cnt = 0
progress = platformtools.dialogProgress(config.getLocalizedString(30993) % item.title, config.getLocalizedString(60295) + '\n' + config.getLocalizedString(60293))
config.setSetting('tmdb_active', True)
# res_count = 0
for key, value in ch_list.items():
ch_name = channel_titles[channel_list.index(key)]
grouped = list()
cnt += 1
progress.update(old_div((cnt * 100), len(ch_list)), config.getLocalizedString(60295))
for it in value:
if it.channel == item.channel:
it.channel = key
if it in valid:
continue
if mode == 'all' or (it.contentType and mode == it.contentType):
if config.getSetting('result_mode') != 0:
if config.getLocalizedString(30992) not in it.title:
it.title += typo(ch_name,'_ [] color kod bold')
results.append(it)
else:
grouped.append(it)
elif (mode == 'movie' and it.contentTitle) or (mode == 'tvshow' and (it.contentSerieName or it.show)):
grouped.append(it)
else:
continue
if not grouped:
continue
# to_temp[key] = grouped
if config.getSetting('result_mode') == 0:
if not config.getSetting('unify'):
title = typo(ch_name,'bold') + typo(str(len(grouped)), '_ [] color kod bold')
else:
title = typo('%s %s' % (len(grouped), config.getLocalizedString(70695)), 'bold')
# res_count += len(grouped)
plot=''
for it in grouped:
plot += it.title +'\n'
ch_thumb = channeltools.get_channel_parameters(key)['thumbnail']
results.append(Item(channel=item.channel, title=title,
action='get_from_temp', thumbnail=ch_thumb, itemlist=[ris.tourl() for ris in grouped], plot=plot, page=1))
progress.close()
# "All Together" and movie mode -> search servers
if config.getSetting('result_mode') == 1 and mode == 'movie':
progress = platformtools.dialogProgress(config.getLocalizedString(30993) % item.title, config.getLocalizedString(60683))
valid_servers = []
with futures.ThreadPoolExecutor(max_workers=set_workers()) as executor:
c_results = [executor.submit(get_servers, v, module_dict) for v in valid]
completed = 0
for res in futures.as_completed(c_results):
if progress.iscanceled():
break
if res.result():
completed += 1
valid_servers.extend(res.result())
progress.update(old_div(completed * 100, len(valid)))
valid = valid_servers
progress.close()
# send_to_temp(to_temp)
results = sorted(results, key=lambda it: it.title)
results_statistic = config.getLocalizedString(59972) % (item.title, time.time() - start)
if mode == 'all':
results.insert(0, Item(title=typo(results_statistic, 'color kod bold'), thumbnail=thumb('search')))
else:
if not valid:
valid.append(Item(title=config.getLocalizedString(60347), thumbnail=thumb('nofolder')))
valid.insert(0, Item(title=typo(results_statistic, 'color kod bold'), thumbnail=thumb('search')))
results.insert(0, Item(title=typo(config.getLocalizedString(30025), 'color kod bold'), thumbnail=thumb('search')))
# logger.debug(results_statistic)
itlist = valid + results
writelist = item.text
for it in itlist:
writelist += ',' + it.tourl()
filetools.write(temp_search_file, writelist)
return itlist
def get_channel_results(item, module_dict, search_action):
ch = search_action.channel
results = list()
valid = list()
module = module_dict[ch]
searched_id = item.infoLabels['tmdb_id']
try:
results.extend(module.search(search_action, item.text))
if len(results) == 1:
if not results[0].action or config.getLocalizedString(70006).lower() in results[0].title.lower():
results.clear()
if item.mode != 'all':
for elem in results:
if not elem.infoLabels.get('year', ""):
elem.infoLabels['year'] = '-'
tmdb.set_infoLabels_item(elem)
if elem.infoLabels['tmdb_id'] == searched_id:
elem.from_channel = ch
if not config.getSetting('unify'):
elem.title += ' [%s]' % ch
valid.append(elem)
# if len(results) < 0 and len(results) < max_results and item.mode != 'all':
#
# if len(results) == 1:
# if not results[0].action or config.getLocalizedString(30992).lower() in results[0].title.lower():
# return [ch, []]
#
# results = get_info(results)
return [search_action, results, valid]
except:
return [search_action, results, valid]
def get_servers(item, module_dict):
item.global_search = True
ch = item.channel
results = list()
module = module_dict[ch]
try:
results = getattr(module, item.action)(item)
except:
import traceback
logger.error(traceback.format_exc())
return [r.clone(title=r.title + typo(item.channel, '_ [] color kod')) for r in results if r.action == 'play']
def get_info(itemlist):
logger.debug()
tmdb.set_infoLabels_itemlist(itemlist, True, forced=True)
return itemlist
def get_channels(item):
logger.debug()
channels_list = list()
title_list = list()
all_channels = channelselector.filterchannels('all')
for ch in all_channels:
channel = ch.channel
ch_param = channeltools.get_channel_parameters(channel)
if not ch_param.get("active", False):
continue
list_cat = ch_param.get("categories", [])
if not ch_param.get("include_in_global_search", False):
continue
if 'anime' in list_cat:
n = list_cat.index('anime')
list_cat[n] = 'tvshow'
if item.mode == 'all' or (item.mode in list_cat):
if config.getSetting("include_in_global_search", channel) and ch_param.get("active", False):
channels_list.append(channel)
title_list.append(ch_param.get('title', channel))
return channels_list, title_list
def settings(item):
return platformtools.showChannelSettings(caption=config.getLocalizedString(59993))
def set_workers():
workers = config.getSetting('thread_number') if config.getSetting('thread_number') > 0 else None
return workers
def genres_menu(item):
itemlist = []
mode = item.mode.replace('show', '')
genres = tmdb.get_dic_genres(mode)
for key, value in list(genres[mode].items()):
discovery = {'url': 'discover/%s' % mode, 'with_genres': key,
'language': def_lang, 'page': '1'}
itemlist.append(Item(channel=item.channel, title=value, page=1,
action='discover_list', discovery=discovery,
mode=item.mode))
from core import support
support.thumb(itemlist, mode='genre')
return sorted(itemlist, key=lambda it: it.title)
def years_menu(item):
import datetime
itemlist = []
mode = item.mode.replace('show', '')
par_year = 'primary_release_year'
thumb = support.thumb('movie_year')
if mode != 'movie':
par_year = 'first_air_date_year'
thumb = thumb('tvshow_year')
c_year = datetime.datetime.now().year + 1
l_year = c_year - 31
for year in range(l_year, c_year):
discovery = {'url': 'discover/%s' % mode, 'page': '1',
'%s' % par_year: '%s' % year,
'sort_by': 'popularity.desc', 'language': def_lang}
itemlist.append(Item(channel=item.channel, title=str(year), action='discover_list',
discovery=discovery, mode=item.mode, year_=str(year), thumbnail=thumb))
itemlist.reverse()
itemlist.append(Item(channel=item.channel, title=typo(config.getLocalizedString(70745),'color kod bold'), url='',
action="year_cus", mode=item.mode, par_year=par_year))
return itemlist
def year_cus(item):
mode = item.mode.replace('show', '')
heading = config.getLocalizedString(70042)
year = platformtools.dialogNumeric(0, heading, default="")
item.discovery = {'url': 'discover/%s' % mode, 'page': '1',
'%s' % item.par_year: '%s' % year,
'sort_by': 'popularity.desc', 'language': def_lang}
item.action = "discover_list"
if year and len(year) == 4:
return discover_list(item)
def actor_list(item):
itemlist = []
dict_ = {'url': 'search/person', 'language': def_lang, 'query': item.searched_text, 'page': item.page}
prof = {'Acting': 'Actor', 'Directing': 'Director', 'Production': 'Productor'}
plot = ''
item.search_type = 'person'
tmdb_inf = tmdb.discovery(item, dict_=dict_)
results = tmdb_inf.results
if not results:
return results
for elem in results:
name = elem.get('name', '')
if not name:
continue
rol = elem.get('known_for_department', '')
rol = prof.get(rol, rol)
# genero = elem.get('gender', 0)
# if genero == 1 and rol in prof:
# rol += 'a'
# rol = rol.replace('Actora', 'Actriz')
know_for = elem.get('known_for', '')
cast_id = elem.get('id', '')
if know_for:
t_k = know_for[0].get('title', '')
if t_k:
plot = '%s in %s' % (rol, t_k)
thumbnail = 'https://image.tmdb.org/t/p/original%s' % elem.get('profile_path', '')
title = name + typo(rol,'_ [] color kod')
discovery = {'url': 'person/%s/combined_credits' % cast_id, 'page': '1',
'sort_by': 'primary_release_date.desc', 'language': def_lang}
itemlist.append(Item(channel=item.channel, title=title, action='discover_list', cast_='cast',
discovery=discovery, thumbnail=thumbnail, plot=plot, page=1))
if len(results) >= config.getSetting('pagination', default=20):
next_ = item.page + 1
itemlist.append(Item(channel=item.channel, title=typo(config.getLocalizedString(90006),'bold color kod'), action='actor_list',
page=next_, thumbnail=thumbnail,
searched_text=item.searched_text))
return itemlist
def discover_list(item):
import datetime
itemlist = []
year = 0
tmdb_inf = tmdb.discovery(item, dict_=item.discovery, cast=item.cast_)
result = tmdb_inf.results
total_pages = tmdb_inf.total_pages
tvshow = False
for elem in result:
elem = tmdb_inf.get_infoLabels(elem, origen=elem)
if 'title' in elem:
title = unify.normalize(elem['title']).capitalize()
else:
title = unify.normalize(elem['name']).capitalize()
tvshow = True
elem['tmdb_id'] = elem['id']
mode = item.mode or elem['mediatype']
thumbnail = elem.get('thumbnail', '')
fanart = elem.get('fanart', '')
if item.cast_:
release = elem.get('release_date', '0000') or elem.get('first_air_date', '0000')
year = scrapertools.find_single_match(release, r'(\d{4})')
if not item.cast_ or (item.cast_ and (int(year) <= int(datetime.datetime.today().year))):
if config.getSetting('new_search'):
new_item = Item(channel='globalsearch', title=title, infoLabels=elem,
action='new_search', text=title,
thumbnail=thumbnail, fanart=fanart,
context='', mode = 'search/' + mode, contentType=mode,
release_date=year, folder = False)
else:
new_item = Item(channel='search', title=title, infoLabels=elem,
action='channel_search', text=title,
thumbnail=thumbnail, fanart=fanart,
context='', mode=mode, contentType=mode,
release_date=year)
if tvshow:
new_item.contentSerieName = title
else:
new_item.contentTitle = title
itemlist.append(new_item)
itemlist = set_context(itemlist)
if item.cast_:
itemlist.sort(key=lambda it: int(it.release_date), reverse=True)
return itemlist
elif config.getSetting('pagination', default=20):
if item.discovery:
page = item.discovery['page'] = int(item.discovery['page']) + 1
else:
page = int(item.page) + 1
support.nextPage(itemlist, item, page=page, total_pages=total_pages)
return itemlist
def get_from_temp(item):
logger.debug()
n = 30
nTotal = len(item.itemlist)
nextp = n * item.page
prevp = n * (item.page - 1)
results = [Item().fromurl(elem) for elem in item.itemlist[prevp:nextp]]
if nextp < nTotal:
results.append(Item(channel='search', title=typo(config.getLocalizedString(30992), 'bold color kod'),
action='get_from_temp', itemlist=item.itemlist, page=item.page + 1, nextPage=True))
tmdb.set_infoLabels_itemlist(results, True)
for elem in results:
if not elem.infoLabels.get('year', ""):
elem.infoLabels['year'] = '-'
tmdb.set_infoLabels_item(elem, True)
return results