-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathdefault.py
1557 lines (1438 loc) · 70.4 KB
/
default.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
from time import strftime, strptime, time, mktime, localtime, tzname
import os, sys, re, socket, urllib, unicodedata, threading
from traceback import print_exc
from datetime import datetime, date, timedelta
from dateutil import tz
from operator import itemgetter
import xbmc, xbmcgui, xbmcaddon, xbmcvfs
try:
import simplejson as json
except ImportError:
import json
# http://mail.python.org/pipermail/python-list/2009-June/540579.html
import _strptime
__addon__ = xbmcaddon.Addon()
__addonid__ = __addon__.getAddonInfo('id')
__addonname__ = __addon__.getAddonInfo('name')
__cwd__ = __addon__.getAddonInfo('path').decode('utf-8')
__author__ = __addon__.getAddonInfo('author')
__version__ = __addon__.getAddonInfo('version')
__language__ = __addon__.getLocalizedString
__datapath__ = os.path.join(xbmc.translatePath('special://masterprofile/addon_data/').decode('utf-8'), __addonid__)
__profilepath__ = os.path.join(xbmc.translatePath('special://profile/addon_data/').decode('utf-8'), __addonid__)
__resource__ = xbmc.translatePath(os.path.join(__cwd__, 'resources', 'lib').encode("utf-8")).decode("utf-8")
sys.path = [__resource__] + sys.path
from thetvdbapi import TheTVDB
from country_lookup import CountryLookup
from fanarttv import FanartTV
MAX_INFO_LOG_LEVEL = 1
MAX_DEBUG_LOG_LEVEL = 2
NEXTAIRED_DB = 'next.aired.db'
COUNTRY_DB = 'country.db'
OLD_FILES = [ 'nextaired.db', 'next_aired.db', 'canceled.db', 'cancelled.db' ]
LISTITEM_ART = [ 'poster', 'banner', 'clearlogo' ] # This order MUST match the settings.xml list!!
USEFUL_ART = LISTITEM_ART + [ 'characterart', 'clearart', 'fanart', 'landscape' ]
LEADING_ZERO_REGEX = re.compile(r"^0")
CLASSIFICATION_REGEX = re.compile(r"(?:^| \| )(Scripted|Mini-Series|Documentary|Animation|Game Show|Reality|Talk Show|Variety)( \| |$)")
STATUS = { '0' : __language__(32201),
'1' : __language__(32202),
'2' : __language__(32203),
'3' : __language__(32204),
'4' : __language__(32205),
'5' : __language__(32206),
'6' : __language__(32207),
'7' : __language__(32208),
'8' : __language__(32209),
'9' : __language__(32210),
'10' : __language__(32211),
'11' : __language__(32212),
'-1' : ''}
# Get localized date format
DATE_FORMAT = xbmc.getRegion('dateshort').lower()
if DATE_FORMAT[0] == 'd':
DATE_FORMAT = '%d-%m-%y'
elif DATE_FORMAT[0] == 'm':
DATE_FORMAT = '%m-%d-%y'
elif DATE_FORMAT[0] == 'y':
DATE_FORMAT = '%y-%m-%d'
NICE_DATE_FORMAT = xbmc.getRegion('datelong').lower().replace('%d%d', '%d').replace("'", "").decode('utf-8')
for xx, yy in (('%a', '%(wday)s'), ('%b', '%(month)s'), ('%d', '%(day)s'), ('%y', '%(year)s'), ('%m', '%(mm)s')):
NICE_DATE_FORMAT = NICE_DATE_FORMAT.replace(xx, yy)
NICE_DATE_FORMAT = re.sub(r"%[a-z]", '%(unk)s', NICE_DATE_FORMAT)
NICE_DATE_NO_YEAR = re.sub(r"(?<=\)s)[^%]*%\(year\)s[^%]*|^%\(year\)s[^%]*", ' ', NICE_DATE_FORMAT).strip()
NICE_SHORT_DATE = re.sub(r"%\(wday\)s[^%]*", '', NICE_DATE_NO_YEAR)
MAIN_DB_VER = 6
COUNTRY_DB_VER = 1
FAILURE_PAUSE = 5*60
INT_REGEX = re.compile(r"^([0-9]+)$")
if not xbmcvfs.exists(__datapath__):
xbmcvfs.mkdir(__datapath__)
if __profilepath__ != __datapath__ and not xbmcvfs.exists(__profilepath__):
xbmcvfs.mkdir(__profilepath__)
# If the user wants a json cache of the latest show + episode info, they can
# create an id-cache dir and we'll put ID_NUMBER.json files into it.
id_cache_dir = os.path.join(__datapath__, 'id-cache')
if not xbmcvfs.exists(id_cache_dir):
id_cache_dir = None
# if level <= 0, sends LOGERROR msg. For positive values, sends LOGNOTICE
# if level <= MAX_INFO_LOG_LEVEL, else LOGDEBUG. If level is omitted, we assume 10.
def log(txt, level=10):
if level > max(MAX_DEBUG_LOG_LEVEL, MAX_INFO_LOG_LEVEL):
return
if isinstance (txt,str):
txt = txt.decode("utf-8")
message = u'%s: %s' % (__addonid__, txt)
log_level = (xbmc.LOGERROR if level <= 0 else (xbmc.LOGNOTICE if level <= MAX_INFO_LOG_LEVEL else xbmc.LOGDEBUG))
xbmc.log(msg=message.encode("utf-8"), level=log_level)
def _unicode(text, encoding='utf-8'):
try: text = unicode(text, encoding)
except: pass
return text
def normalize(d, key = None, default = ""):
if key is None:
text = d
else:
text = d.get(key, default)
if not text:
return text
try:
# We avoid "ignore" here because that can cause the text to go completely empty.
text = unicodedata.normalize('NFKD', _unicode(text))
text = text.encode('ascii')
except:
pass
return text
STRIP_PUNCT_RE = re.compile("[.,:;?!*$^#|<>'\"]")
STRIP_EXTRA_SPACES_RE = re.compile(r"\s{2,}")
def lc_stripped_name(show_name):
show_name = STRIP_EXTRA_SPACES_RE.sub(' ', show_name.lower())
return STRIP_PUNCT_RE.sub('', show_name.strip())
def maybe_int(d, key, default = 0):
v = d.get(key, str(default))
return int(v) if INT_REGEX.match(v) else v
class NextAired:
def __init__(self):
self.WINDOW = xbmcgui.Window(10000)
self.set_today()
self.tznames = ','.join(map(str,tzname))
self.weekdays = []
for j in range(11, 18):
self.weekdays.append(xbmc.getLocalizedString(j))
self.wdays = []
for j in range(41, 48):
self.wdays.append(xbmc.getLocalizedString(j))
self.local_months = []
for j in range(51, 63):
self.local_months.append(xbmc.getLocalizedString(j))
self.ampm = xbmc.getCondVisibility('substring(System.Time,Am)') or xbmc.getCondVisibility('substring(System.Time,Pm)')
self.improve_dates = __addon__.getSetting("ImproveDates") == 'true'
self.ignore_specials = __addon__.getSetting("IgnoreSpecials") == 'true'
if __profilepath__ == __datapath__:
self.profile_name = ''
else:
m = re.search(r"([^/\\]+)[/\\]?$", xbmc.translatePath('special://profile/'))
self.profile_name = m.group(1)
log("### profile_name = %s" % self.profile_name, level=6)
# "last_success" is when we last successfully made it through an update pass without fetch errors.
# "last_update" is when we last successfully marked-up the shows to note which ones need an update.
# "last_failure" is when we last failed to fetch data, with failure_cnt counting consecutive failures.
self.last_success = self.last_update = self.last_failure = self.failure_cnt = 0
self._parse_argv()
self._footprints()
self.check_xbmc_version()
if self.SERVICE and self.xbmc_version < 12.3: # Let's try ignoring the service process on frodo
log("### ignoring service proc on XBMC version %s" % self.xbmc_version, level=1)
elif self.TVSHOWTITLE:
self.return_properties(self.TVSHOWTITLE)
elif self.UPDATESHOW:
self.update_show(self.UPDATESHOW)
elif self.BACKEND:
self.run_backend()
elif not self.SILENT:
self.show_gui()
elif self.STOP:
self.stop_background_updating()
else:
for old_file in OLD_FILES:
self.rm_file(old_file)
self.do_background_updating()
def _parse_argv(self):
global MAX_INFO_LOG_LEVEL
try:
self.params = dict(arg.split("=") for arg in sys.argv[1].split("&"))
except:
self.params = {}
want_infolog = self.params.get("infolog", None)
if want_infolog is not None:
MAX_INFO_LOG_LEVEL = int(want_infolog)
log("### params: %s" % self.params, level=2)
self.SERVICE = self.params.get("service", False)
self.SILENT = self.SERVICE or self.params.get("silent", False)
self.BACKEND = self.params.get("backend", False)
self.TVSHOWTITLE = normalize(self.params, "tvshowtitle", False)
self.UPDATESHOW = normalize(self.params, "updateshow", False)
self.FORCEUPDATE = self.params.get("force", False)
self.RESET = self.params.get("reset", False)
self.STOP = self.params.get("stop", False)
self.ONCE = self.params.get("once", False)
def _footprints(self):
def_level = 2 if self.TVSHOWTITLE else 1
style = 'background' if self.SILENT else 'GUI'
force = 'w/FORCEUPDATE ' if self.FORCEUPDATE else ''
reset = 'w/RESET ' if self.RESET else ''
log("### %s starting %s proc %s%s(%s)" % (__addonname__, style, force, reset, __version__), level=def_level)
if def_level == 1:
log("### dateformat: %s" % DATE_FORMAT, level=4)
log("### nice-date-format: %s" % NICE_DATE_FORMAT, level=4)
log("### nice-date-no-year: %s" % NICE_DATE_NO_YEAR, level=4)
log("### nice-short-date: %s" % NICE_SHORT_DATE, level=4)
def set_today(self):
self.now = time()
self.date = date.today()
self.datestr = str(self.date)
self.yesterday = self.date - timedelta(days=1)
self.yesterstr = str(self.yesterday)
self.tomorrow = self.date + timedelta(days=1)
# Returns elapsed seconds since last update failure.
def get_last_failure(self):
v = self.WINDOW.getProperty("NextAired.last_failure")
v = float(v) if v != "" else 0
if v and self.failure_cnt == 0:
self.failure_cnt += 1
self.last_failure = max(v, self.last_failure)
return self.now - self.last_failure
def set_last_failure(self):
self.last_failure = self.now
self.get_last_failure()
self.WINDOW.setProperty("NextAired.last_failure", str(self.last_failure))
def is_time_for_update(self, update_after_seconds):
self.now = time()
if self.FORCEUPDATE:
return True
if update_after_seconds == 0:
return False
if self.now - self.last_success < update_after_seconds:
return False
if self.get_last_failure() < FAILURE_PAUSE * min(self.failure_cnt, 24):
return False
return True
def do_background_updating(self):
# We greatly prefer the service.py version of background updating vs
# the skin's because we get notified to stop when the user logs out.
if not self.SERVICE:
xbmc.sleep(2000)
background_id = self.WINDOW.getProperty("NextAired.background_id")
if background_id != '':
bg_info = background_id.split(' ', 1)
# NOTE: the amount of elapsed time we allow here must be NO LARGER
# than the sleep time in handle_bg_version_change().
if len(bg_info) == 2 and time() - float(bg_info[1]) < 15:
self.close("exiting this duplicate background-proc (skin vs service)")
my_unique_id = "%s,%s %s" % (os.getpid(), threading.currentThread().ident, time())
self.WINDOW.setProperty("NextAired.background_id", my_unique_id)
while not xbmc.abortRequested:
bg_lock = self.WINDOW.getProperty("NextAired.bgnd_lock")
if bg_lock == "" or time() - float(bg_lock) > 10*60:
break
xbmc.sleep(1000)
profile_dir = xbmc.translatePath("special://profile/addon_data/")
next_chk = self.now
this_day = ''
while not xbmc.abortRequested:
self.now = time()
if self.now < next_chk:
# We can't sleep for very long at a time or a shutdown bogs down.
# To combat this, we do very little most of the times that we wake
# up, and the rest of the work after enough time has passed.
xbmc.sleep(1000)
continue
if self.now > next_chk + 15:
# We slept for a longer time than expected, so this probably means that
# the computer is waking up from suspend. Since the networking may not
# be ready to go just yet, we'll delay our upgrade checking a bit more.
next_chk = self.now + 60
continue
if self.WINDOW.getProperty("NextAired.background_id") != my_unique_id:
self.close("another background script was started -- stopping older background proc")
latest_version = xbmcaddon.Addon().getAddonInfo('version')
if latest_version != __version__:
self.handle_bg_version_change(latest_version)
if not self.SERVICE and xbmc.translatePath("special://profile/addon_data/") != profile_dir:
self.close("profile directory changed -- stopping background proc")
try:
update_every = int(__addon__.getSetting('update_every'))*60*60 # hours -> seconds
except:
update_every = 0
# Note that we run the update routine at least once a day to age the episode lists,
# even if we don't grab any new data (if it's not update time just yet).
self.set_today()
next_chk = self.now + 20
if self.datestr != this_day or self.is_time_for_update(update_every):
if self.update_data(update_every):
self.failure_cnt = 0
this_day = self.datestr
else:
self.failure_cnt += 1
next_chk = self.now + FAILURE_PAUSE * min(self.failure_cnt, 24)
self.nextlist = [] # Discard the in-memory data until the next update
if self.ONCE:
break
else:
xbmc.sleep(1000)
self.close("abort requested -- stopping background processing")
def stop_background_updating(self):
self.WINDOW.setProperty("NextAired.background_id", 'stop')
def load_data(self):
if self.RESET:
self.rm_file(NEXTAIRED_DB)
self.rm_file(COUNTRY_DB)
# Snag our TV-network -> Country mapping DB.
cl = self.get_list(COUNTRY_DB)
self.country_dict = (cl.pop(0) if cl else {})
self.country_last_update = (cl.pop() if cl else 0)
db_ver = (cl.pop(0) if cl else 0)
if db_ver != COUNTRY_DB_VER:
self.country_dict = {}
ep_list = self.get_list(NEXTAIRED_DB)
ep_list_len = len(ep_list)
show_dict = (ep_list.pop(0) if ep_list else {})
self.last_success = (ep_list.pop() if ep_list else None)
db_ver = (ep_list.pop(0) if ep_list else None)
self.last_update = (ep_list.pop() if ep_list else self.last_success)
self.old_tznames = (ep_list.pop(0) if ep_list else '')
old_profile_name = (ep_list.pop(0) if ep_list else '')
if db_ver is None or self.last_success is None:
if self.RESET:
log("### starting without prior data (DB RESET requested)", level=1)
elif ep_list_len:
log("### ignoring bogus %s file" % NEXTAIRED_DB, level=1)
else:
log("### no prior data found", level=1)
show_dict = {}
self.last_success = self.last_update = 0
elif db_ver < MAIN_DB_VER:
self.upgrade_data_format(show_dict, db_ver)
elif db_ver > MAIN_DB_VER:
self.close("ERROR: DB version is too new for this script (%d > %d) -- exiting" % (db_ver, MAIN_DB_VER))
self.RESET = False # Make sure we don't honor this multiple times.
if self.profile_name != '':
self.maybe_merge_profile_DB(show_dict)
if self.profile_name != old_profile_name:
self.FORCEUPDATE = True # We need a forced update when switching profiles.
return (show_dict, self.now - self.last_update)
def save_data(self, show_dict):
self.save_file(
[show_dict, MAIN_DB_VER, self.tznames, self.profile_name, self.last_update, self.last_success],
NEXTAIRED_DB)
def set_update_lock(self, DIALOG_PROGRESS = None):
if DIALOG_PROGRESS:
self.WINDOW.setProperty("NextAired.user_lock", str(self.now))
else:
self.WINDOW.setProperty("NextAired.bgnd_status", "0|0|...")
self.WINDOW.setProperty("NextAired.bgnd_lock", str(self.now))
def clear_update_lock(self, DIALOG_PROGRESS = None):
if DIALOG_PROGRESS:
DIALOG_PROGRESS.close()
self.WINDOW.clearProperty("NextAired.user_lock")
else:
self.WINDOW.clearProperty("NextAired.bgnd_lock")
xbmc.sleep(1000)
self.WINDOW.clearProperty("NextAired.bgnd_status")
def update_data(self, update_after_seconds, force_show = None):
self.nextlist = []
show_dict, elapsed_secs = self.load_data()
# This should prevent the background and user code from updating the DB at the same time.
if self.SILENT:
DIALOG_PROGRESS = None
# We double-check this here, just in case it changed.
if self.is_time_for_update(update_after_seconds):
self.set_update_lock()
locked_for_update = True
xbmc.sleep(2000) # try to avoid a race-condition
# Background updating: we will just skip our update if the user is doing an update.
user_lock = self.WINDOW.getProperty("NextAired.user_lock")
if user_lock != "":
if self.now - float(user_lock) <= 10*60:
self.clear_update_lock()
# We failed to get data, so this will cause us to check on the update in a bit.
# (No need to save it as a failure property -- this is just for us.)
self.last_failure = self.now
return False
# User's lock has sat around for too long, so just ignore it.
self.max_fetch_failures = 8
else:
locked_for_update = False
socket.setdefaulttimeout(60)
elif self.is_time_for_update(update_after_seconds): # We only lock if we're going to do some updating.
# User updating: we will wait for a background update to finish, then see if we have recent data.
DIALOG_PROGRESS = xbmcgui.DialogProgress()
DIALOG_PROGRESS.create(__language__(32101), __language__(32102))
self.max_fetch_failures = 8 if self.FORCEUPDATE else 4
# Create our user-lock file and check if the background updater is running.
self.set_update_lock(DIALOG_PROGRESS)
locked_for_update = True
newest_time = 0
prior_name = ''
while True:
bg_lock = self.WINDOW.getProperty("NextAired.bgnd_lock")
if bg_lock == "":
break
if newest_time == 0:
newest_time = float(bg_lock)
bg_status = self.WINDOW.getProperty("NextAired.bgnd_status")
bg_status = bg_status.split('|', 2)
if len(bg_status) == 3:
status_time, percent, show_name = (float(bg_status[0]), int(bg_status[1]), bg_status[2])
if show_name != prior_name:
DIALOG_PROGRESS.update(percent, __language__(32102), show_name)
prior_name = show_name
if DIALOG_PROGRESS.iscanceled():
self.clear_update_lock(DIALOG_PROGRESS)
xbmcgui.Dialog().ok(__language__(32103),__language__(32104))
locked_for_update = False
break
if status_time > newest_time:
newest_time = status_time
if time() - newest_time > 2*60:
# Background lock has sat around for too long, so just ignore it.
newest_time = 0
break
xbmc.sleep(500)
if newest_time:
# If we had to wait for the bgnd updater, re-read the data and unlock if they did an update.
show_dict, elapsed_secs = self.load_data()
if locked_for_update and not self.is_time_for_update(update_after_seconds):
self.clear_update_lock(DIALOG_PROGRESS)
locked_for_update = False
socket.setdefaulttimeout(10)
else:
locked_for_update = False
if locked_for_update:
log("### starting data update", level=1)
self.last_failure = 0
# If the local timezone changed, we will need to recompute the Airtime values.
if self.tznames != self.old_tznames:
for tid, show in show_dict.iteritems():
if 'show_changed' not in show:
show['show_changed'] = 1
# We want to recreate our country DB every week.
if len(self.country_dict) < 500 or self.now - self.country_last_update >= 7*24*60*60:
try:
log("### grabbing a new country mapping list", level=1)
if not self.SILENT:
DIALOG_PROGRESS.update(0, __language__(32102), "country.db")
self.country_dict = CountryLookup().get_country_dict()
self.save_file([self.country_dict, COUNTRY_DB_VER, self.now], COUNTRY_DB)
except:
pass
slang = __addon__.getSetting("SearchLang").split(' ')[0]
log('### search language = "%s"' % slang, level=2)
tvdb = TheTVDB('1D62F2F90030C444', slang, want_raw = True)
if force_show is None:
# This typically asks TheTVDB for an update-zip file and tweaks the show_dict to note needed updates.
tv_up = tvdb_updater(tvdb)
need_full_scan, got_update = tv_up.note_updates(show_dict, elapsed_secs)
if need_full_scan or got_update:
self.last_update = self.now
elif not got_update:
self.set_last_failure()
self.max_fetch_failures = 0
tv_up = None
else:
need_full_scan = False
art_rescan_after = 24*60*60 - 5*60
else:
tvdb = None # We don't use this unless we're locked for the update.
need_full_scan = False
# A max-fetch of 0 disables all updating.
self.max_fetch_failures = 0
art_rescan_after = 0
art_rescan_type = LISTITEM_ART[int(__addon__.getSetting("ThumbType"))]
title_dict = {}
for tid, show in show_dict.iteritems():
name = show['localname']
if (force_show is None or force_show == name) and self.profile_name in show['profiles']:
del show['profiles'][self.profile_name]
title_dict[name] = tid
if force_show is not None and force_show in title_dict:
show = show_dict[title_dict[force_show]]
if 'show_changed' not in show:
show['show_changed'] = 1
if 'eps_changed' not in show:
show['eps_changed'] = (1, 0)
TVlist = self.listing()
for tid in re.split(r"\D+", __addon__.getSetting("ExtraShows")):
if tid != '':
prior_data = show_dict.get(int(tid), None)
if prior_data:
name = prior_data['localname']
else:
name = '/%s/' % tid
# This fake data in the art hash ensures that we trust the tid value.
fake_art = {'ExtraShow': 'http://thetvdb.com/fake/%s-fake.jpg' % tid}
TVlist.append((name, name, fake_art, '', '', tid, ''))
omitShow = {}
for tid in re.split(r"\D+", __addon__.getSetting("OmitShows")):
if tid != '':
omitShow[int(tid)] = 1;
total_show = len(TVlist)
if total_show == 0:
if locked_for_update:
self.clear_update_lock(DIALOG_PROGRESS)
self.set_last_failure()
return False
count = 0
user_canceled = False
id_re = re.compile(r"\bthetvdb\.com/[^'" + '"' + r":]+/([0-9]+)-")
for show in TVlist:
count += 1
name = show[0]
if force_show is not None and name != force_show:
continue
art = show[2]
premiered_year = show[6][:4] if show[6] != '' else None
percent = int(float(count * 100) / total_show)
if self.SILENT:
self.WINDOW.setProperty("NextAired.bgnd_status", "%f|%d|%s" % (time(), percent, name))
elif locked_for_update and self.max_fetch_failures > 0:
DIALOG_PROGRESS.update(percent, __language__(32102), name)
if DIALOG_PROGRESS.iscanceled():
DIALOG_PROGRESS.close()
xbmcgui.Dialog().ok(__language__(32103),__language__(32104))
user_canceled = True
self.set_last_failure()
self.max_fetch_failures = 0
log("### TVlist #%s = %s" % (show[5], name), level=(3 if locked_for_update else 10))
current_show = {
"localname": name,
"path": show[1],
"art": {},
"dbid": show[3],
"thumbnail": show[4],
"profiles": { }
}
# Try to figure out what the tvdb number is by using the art URLs and the imdbnumber value
m2 = id_re.search(str(art).replace('%3a', ':').replace('%2f', '/'))
m2_num = int(m2.group(1)) if m2 else 0
m4 = id_re.search(show[4].replace('%3a', ':').replace('%2f', '/'))
m4_num = int(m4.group(1)) if m4 else 0
m5 = INT_REGEX.match(show[5])
m5_num = int(m5.group(1)) if m5 else 0
if m5_num and (m2_num == m5_num or m4_num == m5_num):
# Most shows will be in agreement on the id when the scraper is using thetvdb.
tid = m5_num
else:
old_id = title_dict.get(name, 0)
old_data = show_dict.get(m5_num, None) if m5_num else None
if old_id and old_id == m5_num:
tid = m5_num
elif old_data and current_show['path'] == old_data['path']:
# This handles a localname change where we knew what the ID was before -- keep that info.
tid = m5_num
elif old_id and force_show is None:
# This is an "iffy" ID. We'll keep using it unless the user asked for a fresh start.
tid = old_id
else:
if self.max_fetch_failures <= 0:
continue
tid = self.find_show_id(tvdb, name, m5_num, premiered_year)
if tid == 0:
continue
prior_data = show_dict.get(tid, None)
if prior_data:
if self.profile_name in prior_data['profiles']:
continue # How'd we get a duplicate?? Skip it...
prior_data['profiles'][self.profile_name] = 1
current_show['profiles'] = prior_data['profiles']
self.age_episodes(prior_data)
if self.max_fetch_failures > 0:
tid = self.check_show_info(tvdb, tid, current_show, prior_data)
else:
tid = -tid
if tid < 0:
if not prior_data:
continue
for item in prior_data:
if item not in current_show:
current_show[item] = prior_data[item]
tid = -tid
for art_type in USEFUL_ART:
xart = art.get(art_type, None)
fudged_flag = 'fudged.' + art_type
if not xart:
if prior_data and fudged_flag in prior_data['art']:
xart = prior_data['art'][art_type]
elif art_rescan_type != art_type:
continue
elif 'x_art' in current_show and art_type in current_show['x_art']:
xart = current_show['x_art'][art_type]
else:
scan_ndx = 'last_%s_scan' % art_type
last_scan = prior_data['art'].get(scan_ndx, 0) if prior_data else 0
if not art_rescan_after or self.now < last_scan + art_rescan_after:
if last_scan:
current_show['art'][scan_ndx] = last_scan
continue
try:
xart = FanartTV.find_artwork(tid, art_type)
except:
pass
if xart:
log("### found missing %s for %s" % (art_type, name), level=1)
else:
log("### still missing a %s for %s" % (art_type, name), level=2)
current_show['art'][scan_ndx] = self.now
continue
try:
img_re = re.compile(r"^image:")
if not img_re.match(xart):
xart = "image://%s/" % urllib.quote(xart, '')
except:
pass
current_show['art'][fudged_flag] = True
current_show['art'][art_type] = xart
if 'x_art' in current_show:
del current_show['x_art']
current_show['profiles'][self.profile_name] = 1
log("### %s" % current_show)
show_dict[tid] = current_show
# If we did a lot of work, make sure we save it prior to doing anything else.
# This ensures that a bug in the following code won't make us redo everything.
if need_full_scan and locked_for_update:
self.save_data(show_dict)
if show_dict:
log("### data available", level=5)
WantYesterday = __addon__.getSetting("WantYesterday") == 'true'
remove_list = []
for tid, show in show_dict.iteritems():
if self.profile_name not in show['profiles']:
if not show['profiles']:
remove_list.append(tid)
continue
if omitShow.get(tid, False):
continue
if show['ep_ndx'] or (WantYesterday and len(show['episodes']) > 1):
self.nextlist.append(show)
for tid in remove_list:
log('### Removing obsolete show %s' % show_dict[tid]['localname'], level=2)
del show_dict[tid]
self.nextlist.sort(key=lambda x: (x['episodes'][x['ep_ndx']]['aired'], x['Show Name']))
log("### next list: %s shows ### %s" % (len(self.nextlist), self.nextlist), level=7)
self.check_today_show()
self.push_data()
else:
log("### no current show data...", level=5)
if locked_for_update:
if not self.last_failure and force_show is None:
self.last_success = self.now
self.save_data(show_dict)
log("### data update finished", level=1)
self.clear_update_lock(DIALOG_PROGRESS)
if not self.SILENT and self.last_failure and not user_canceled:
xbmcgui.Dialog().ok(__language__(32105), __language__(32106))
self.FORCEUPDATE = False
return not self.last_failure
def age_episodes(self, show):
episodes = show['episodes']
ep_len = len(episodes)
ep_ndx = show['ep_ndx']
if ep_ndx == 0: # 0 indicates no future episodes, so point past the end of the list.
ep_ndx = ep_len
# Start by finding the spot (if any) in the list where future episodes start.
while ep_ndx < ep_len and episodes[ep_ndx]['aired'][:10] < self.datestr:
ep_ndx += 1
# Next we remove episodes older than yesterday, but keep one prior ep that is older than that.
while ep_ndx > 1 and episodes[1]['aired'][:10] < self.yesterstr:
episodes.pop(0)
ep_len -= 1
ep_ndx -= 1
# Make a note of the index of the first upcoming episode or 0 if there are none.
show['ep_ndx'] = (ep_ndx if ep_ndx < ep_len else 0)
def check_xbmc_version(self):
# retrieve current installed version
json_query = xbmc.executeJSONRPC('{"jsonrpc": "2.0", "method": "Application.GetProperties", "params": {"properties": ["version", "name"]}, "id": 1}')
json_query = unicode(json_query, 'utf-8', errors='ignore')
json_response = json.loads(json_query)
log("### %s" % json_response)
try:
ver = json_response['result']['version']
self.xbmc_version = float('%s.%s' % (ver['major'], ver['minor']))
except:
self.xbmc_version = 12
self.videodb = 'videodb://tvshows/titles/' if self.xbmc_version >= 13 else 'videodb://2/2/'
def handle_bg_version_change(self, latest_version):
log("### NextAired version changed from %s to %s -- starting a replacement background proc" % (__version__, latest_version), level=1)
# Delay a bit, just to be sure that it is ready to run. We need to also be sure that this
# is at least as long as the elapsed time we allow in our NextAired.background_id check.
for cnt in range(15):
if xbmc.abortRequested:
sys.exit()
xbmc.sleep(1000)
json_query = xbmc.executeJSONRPC('{"jsonrpc": "2.0", "method": "Addons.ExecuteAddon", "params": {"addonid": "script.tv.show.next.aired", "params": %s}, "id": 0}' % json.dumps(self.params))
json_query = unicode(json_query, 'utf-8', errors='ignore')
json_response = json.loads(json_query)
log("### %s" % json_response)
self.close("stopping this older background proc")
def listing(self):
failures = 0
# If the computer is waking up from a sleep, this call might fail for a little bit.
while not xbmc.abortRequested:
json_query = xbmc.executeJSONRPC('{"jsonrpc": "2.0", "method": "VideoLibrary.GetTVShows", "params": {"properties": ["title", "file", "thumbnail", "art", "imdbnumber", "premiered"], "sort": { "method": "title" } }, "id": 1}')
json_query = unicode(json_query, 'utf-8', errors='ignore')
json_response = json.loads(json_query)
log("### %s" % json_response)
if 'result' in json_response:
break
failures += 1
if failures >= 5:
break
xbmc.sleep(5000)
try:
tvshows = json_response['result']['tvshows']
except:
tvshows = []
TVlist = []
for item in tvshows:
tvshowname = normalize(item, 'title')
path = item['file']
art = item['art']
thumbnail = item['thumbnail']
dbid = self.videodb + str(item['tvshowid']) + '/'
TVlist.append((tvshowname, path, art, dbid, thumbnail, item['imdbnumber'], item['premiered']))
log("### list: %s" % TVlist)
return TVlist
@staticmethod
def find_show_id(tvdb, show_name, maybe_id, want_year = None):
log("### find_show_id(%s, %s, %s)" % (show_name, maybe_id, want_year), level=2)
year_re = re.compile(r" \((\d\d\d\d)\)")
cntry_re = re.compile(r" \(([a-z][a-z])\)$", re.IGNORECASE)
lc_name = lc_stripped_name(show_name)
want_names = [ lc_name ]
if want_year:
want_year = str(want_year)
name_has_year = year_re.search(show_name)
if name_has_year:
want_year = name_has_year.group(1)
show_name = year_re.sub('', show_name)
lc_name = year_re.sub('', lc_name)
want_names.append(lc_name) # Add (stripped-year) "Show"
elif want_year:
want_names.insert(0, "%s (%s)" % (lc_name, want_year)) # Add "Show (1999)"
cntry_match = cntry_re.search(lc_name)
if cntry_match:
alt_name = cntry_re.sub(' ' + cntry_match.group(1), lc_name)
want_names.append(alt_name) # Since we have "Show (XX)", add "Show XX"
if want_year:
want_names.insert(1, "%s (%s)" % (alt_name, want_year)) # Add "Show XX (1999)"
alt_name = cntry_re.sub(" (%s) (%s)" % (want_year, cntry_match.group(1)), lc_name)
if want_names[0] != alt_name:
want_names.insert(1, alt_name) # Add "Show (1999) (XX)"
else:
want_names.insert(1, "%s (%s)" % (lc_name, want_year)) # Add "Show (XX) (1999)"
search_list = [ show_name ]
if want_year:
search_list.append("%s (%s)" % (show_name, want_year))
log("### want_names: %s" % want_names, level=6)
all_results = [ ]
for search_for in search_list:
log("### search_for: %s" % search_for, level=6)
try:
show_list = tvdb.get_matching_shows(search_for, language='all', want_raw=True)
except Exception, e:
log('### ERROR returned by get_matching_shows(): %s' % e, level=0)
return 0
if show_list is None:
show_list = []
for attrs in show_list:
attrs['SeriesName'] = lc_stripped_name(normalize(attrs, 'SeriesName'))
log("### id: %s, FirstAired: %s, SeriesName: %s" % (attrs['id'], attrs.get('FirstAired', '????')[:4], attrs['SeriesName']), level=6)
if int(attrs['id']) == maybe_id:
log("### verified id of %s" % maybe_id, level=2)
return maybe_id
if len(show_list) == 0 and cntry_re.search(search_for):
search_list.append(cntry_re.sub('', search_for))
all_results = show_list + all_results
for want_name in want_names:
for attrs in all_results:
year = attrs.get('FirstAired', '')[:4]
if want_year and year != want_year:
continue
match_names = [ attrs['SeriesName'] ]
if 'AliasNames' in attrs:
for alias in normalize(attrs, 'AliasNames').split('|'):
match_names.append(lc_stripped_name(alias))
for j in range(len(match_names)):
mname = match_names[j]
if len(year) == 4 and not year_re.search(mname):
match_names.append("%s (%s)" % (mname, year))
m = cntry_re.search(mname)
if m:
match_names.append(cntry_re.sub(" (%s) (%s)" % (year, m.group(1)), mname))
log("### match_names: %s" % match_names, level=6)
if want_name in match_names:
log("### found id of %s" % attrs['id'], level=2)
return int(attrs['id'])
log("### no match found", level=2)
return 0
def check_show_info(self, tvdb, tid, current_show, prior_data):
name = current_show['localname']
log("### check if %s is up-to-date (%d)" % (name, tid), level=4)
# If the prior_data isn't in need of an update, use it unchanged.
if prior_data:
earliest_id, eps_last_updated = prior_data.get('eps_changed', (None, 0))
if earliest_id is None:
eps_last_updated = prior_data['eps_last_updated']
show_changed = prior_data.get('show_changed', 0)
if prior_data['Country'] == 'Unknown' and self.country_dict.get(prior_data['Network'], None):
log("### Forcing show-change for %s to fix unknown country" % name, level=2)
if not show_changed:
show_changed = 1
if show_changed:
if earliest_id is None:
earliest_id = 0
elif earliest_id is None:
log("### no changes needed", level=5)
return -tid
else:
show_changed = 0
earliest_id = 1
eps_last_updated = 0
if earliest_id != 0:
for cnt in range(2):
log("### getting series & episode info for #%d - %s" % (tid, name), level=1)
try:
result = tvdb.get_show_and_episodes(tid)
break
except Exception, e:
log('### ERROR returned by get_show_and_episodes(): %s' % e, level=0)
self.set_last_failure()
self.max_fetch_failures -= 1
result = None
if result:
show = result[0]
episodes = result[1]
else:
show = None
else: # earliest_id == 0 when only the series-info changed
for cnt in range(2):
log("### getting series info for #%d - %s" % (tid, name), level=1)
try:
show = tvdb.get_show(tid)
break
except Exception, e:
log('### ERROR returned by get_show(): %s' % e, level=0)
self.set_last_failure()
self.max_fetch_failures -= 1
show = None
episodes = None
if not show:
if prior_data:
log("### no result: continuing to use the old data", level=1)
else:
log("### no result and no prior data", level=1)
return -tid
if id_cache_dir is not None:
for name, var in (('show', show), ('eps', episodes)):
if var is None:
continue
cache_file = os.path.join(id_cache_dir, '%s-%s.json' % (tid, name))
try:
with open(cache_file, 'w') as fh:
fh.write(json.dumps(var, sort_keys=True, indent=2, separators=(', ', ': ')))
except:
pass
network = normalize(show, 'Network', 'Unknown')
country = self.country_dict.get(network, 'Unknown')
tzone = CountryLookup.get_country_timezone(country)
m = re.search(r"\b" + network + r"\s\(([^\)]+)\)", show.get('Overview', ""))
if m:
for c in m.group(1).split(' and '):
t = CountryLookup.get_country_timezone(c)
if t is not None:
country = c
tzone = t
break
if tzone is None:
tzone = 'UTC'
try:
tzinfo = tz.gettz(tzone)
except Exception, e:
log('### tz.gettz() failed: %s' % e, level=2)
tzinfo = None
if tzinfo is None:
log("### didn't get tzinfo for %s" % tzone, level=1)
tzinfo = tz.tzutc()
try:
airtime = TheTVDB.convert_time(show.get('Airs_Time', ""))
except:
airtime = None
if airtime is not None:
hh_mm = airtime.strftime('%H:%M')
dt = datetime.combine(self.date, airtime).replace(tzinfo=tzinfo).astimezone(tz.tzlocal())
early_aired = '1900-01-01T' + dt.strftime('%H:%M:%S%z')
else:
hh_mm = ''
airtime = TheTVDB.convert_time('00:00')
early_aired = '1900-01-01T00:00:00+0000'
current_show['Show Name'] = normalize(show, 'SeriesName')
if current_show['localname'][:1] == '/':
name = current_show['localname'] = current_show['Show Name']
first_aired = show.get('FirstAired', None)
if first_aired:
first_aired = TheTVDB.convert_date(first_aired)
current_show['Premiered'] = first_aired.year
current_show['Started'] = first_aired.isoformat()
else:
current_show['Premiered'] = current_show['Started'] = ""
current_show['Country'] = country
current_show['TZ'] = tzone
current_show['Status'] = normalize(show, 'Status')
current_show['Genres'] = normalize(show, 'Genre').strip('|').replace('|', ' | ')
current_show['Network'] = network
current_show['Airtime'] = hh_mm
current_show['Runtime'] = maybe_int(show, 'Runtime', '')
current_show['x_art'] = {}
for art_type in ('banner', 'fanart', 'poster'):
if art_type in show and show[art_type] != '':
current_show['x_art'][art_type] = 'http://thetvdb.com/banners/%s' % show[art_type]
can_re = re.compile(r"canceled|ended", re.IGNORECASE)
if can_re.search(current_show['Status']):
current_show['canceled'] = True
elif 'canceled' in current_show:
del current_show['canceled']
# Let's assume we need a "None" episode -- it will get cleaned if we don't.
# The early_aired value has an accurate tzlocal time in it just in case it is
# the only item in the list ('aired' is where localized Airtime comes from).
episode_list = [ {'name': None, 'aired': early_aired, 'sn': 0, 'en': 0} ]
if episodes is not None:
mincode_re = re.compile(r"-(\d+)m$")
minutes_re = re.compile(r"\((\d+)(?: +(?:minutes|mins)|m)\)", re.IGNORECASE)
hour_re = re.compile(r"(\d+)[- ]hour\b", re.IGNORECASE)
max_eps_utime = 0
if episodes:
for ep in episodes:
last_updated = maybe_int(ep, 'lastupdated')
if last_updated > max_eps_utime:
max_eps_utime = last_updated
first_aired = TheTVDB.convert_date(ep.get('FirstAired', ""))
if not first_aired:
continue