diff --git a/CHANGES.md b/CHANGES.md index b847ed4970fe481ebfea621bc04ee3b2cf48141d..00e17cb6c1ef661861aa477843d67e6a15778c33 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -1,19 +1,60 @@ +### 4.0.17 (2015-04-19) + +[full changelog](https://github.com/SiCKRAGETV/SickRage/compare/v4.0.16...v4.0.17) + +* libnotify: Use gir-notify instead of pynotify. +* Fix for xml_declaration unexpected keyword in python 2.6. +* Fix UnboundLocalError exception can be thrown at piList.append(curQueueItem). +* Fix Auto determine indexer when indexer tag not present in nfo. +* Fix editShow: Select the default indexer language instead of forcing en. +* Fix Catch emailnotify smtp error. +* Fix Change log timeout errors to WARNING in requests provider clients. +* Fix Ignore OSError when obtaining the size of a path in get_size. +* New Add RSS to EZTV provider. +* New SceneTime provider. +* Fix iptorrent: Add missing import. +* New Add Pause Button on displayShow page. +* Fix Episode Thumbnail url when show is in DVD Order. +* Fix Catch all errors from Kodi notify. +* Fix Add user agent to RARBG when login. +* New Use better logging for get_size exceptions. +* New Added qbittorrent support. +* Fix rarbg: remove urllib.quote. +* Update PNotify to latest version. +* Fix Plex Notification. +* New added the option to use Plex Token vs Username / Password. +* Fix PP Case Sensitive associated files check. +* Fix unknown quality being accepted as a valid quality in PP. +* Fix added notification of unsupported multiepisode torrent. +* Fix proper: saveSearch broken for download propers. +* New Add logline to show found associated files. +* New Add a log line containing connecting IP on login. +* Fix eztv provider. +* Fix Title of None for Errors submitted through UI. +* New Stay on Show Page after "Update Kodi". +* Fix NoneType object has no attribute getOverview. +* New Manage Rolling Download on unpause from submenu button. +* Fix provide a more detailed http code error. +* Fix Show subtitle service unavailable as info. +* New Column Selection for displayShow. +* Fix Update IPTorrents search URL. + ### 4.0.16 (2015-04-12) [full changelog](https://github.com/SiCKRAGETV/SickRage/compare/v4.0.15...v4.0.16) -* New Feature: Added option to add a filter row on main show page (enabled in general settings - interface) -* New Feature: added scheduling status page -* Fixed EZTV rss blank result -* Fixed RARBG tokend -* Fixed Home page filter: change to allow to use parsed data for active(yes/no) -* Fixed OldTPB: Check if the returned results ar proper|repack -* Don't display paused show in backlogOverview -* Trakt Sync by Episode not by Show -* Added gzip setting in config.ini -* Added TRAKTROLLING to filter in viewlog -* Redone Scheduler -* Replace fuzzy images on Add Show ( Add Trending) +* New Feature: Added option to add a filter row on main show page (enabled in general settings - interface) +* New Feature: added scheduling status page +* Fixed EZTV rss blank result +* Fixed RARBG tokend +* Fixed Home page filter: change to allow to use parsed data for active(yes/no) +* Fixed OldTPB: Check if the returned results ar proper|repack +* Don't display paused show in backlogOverview +* Trakt Sync by Episode not by Show +* Added gzip setting in config.ini +* Added TRAKTROLLING to filter in viewlog +* Redone Scheduler +* Replace fuzzy images on Add Show ( Add Trending) ### 4.0.15 (2015-04-05) diff --git a/gui/slick/css/dark.css b/gui/slick/css/dark.css index 95467041f27778b1f1ba2ff1e62d3ea9a5217582..6a320fef293220ca1dda1ca6b9e6114a7366c03b 100644 --- a/gui/slick/css/dark.css +++ b/gui/slick/css/dark.css @@ -1192,6 +1192,29 @@ span.snatched b { opacity: 0.4; } +.displayShowTable { + table-layout: auto; + width: 100%; + border-collapse: collapse; + border-spacing: 0; + text-align: center; + border: none; + empty-cells: show; + color: #000 !important; +} + +.displayShowTable.display_show { + clear:both +} + +.displayShowTable th.row-seasonheader { + border: none !important; + background-color: #222 !important; + color: #fff !important; + padding-top: 15px !important; + text-align: left !important; +} + .sickbeardTable { table-layout: auto; width: 100%; diff --git a/gui/slick/css/lib/pnotify.custom.min.css b/gui/slick/css/lib/pnotify.custom.min.css index 1d0fa42eca7d01a07cc39d456075f7ce0c76c997..8f2d3eb74d4cf473f30609b7d6711947e16b47d1 100644 Binary files a/gui/slick/css/lib/pnotify.custom.min.css and b/gui/slick/css/lib/pnotify.custom.min.css differ diff --git a/gui/slick/css/light.css b/gui/slick/css/light.css index 8df3fe1d64aacf8d1fc65bcdca2ee36d32c58ce7..1c49d9714af641a0a414e4b1eb8d2fc6a20c2197 100644 --- a/gui/slick/css/light.css +++ b/gui/slick/css/light.css @@ -1167,6 +1167,29 @@ span.snatched b { opacity: 0.4; } +.displayShowTable { + table-layout: auto; + width: 100%; + border-collapse: collapse; + border-spacing: 0; + text-align: center; + border: none; + empty-cells: show; + color: #000 !important; +} + +.displayShowTable.display_show { + clear:both +} + +.displayShowTable th.row-seasonheader { + border: none !important; + background-color: #fff !important; + color: #000 !important; + padding-top: 15px !important; + text-align: left !important; +} + .sickbeardTable { table-layout: auto; width: 100%; diff --git a/gui/slick/css/style.css b/gui/slick/css/style.css index f4d250dc5c8ffd73d8350ed3549768d7ded9d381..4216d0c4ac1585481bdbde9cac540c47e4bf9113 100644 --- a/gui/slick/css/style.css +++ b/gui/slick/css/style.css @@ -1193,6 +1193,29 @@ span.snatched b { opacity: 0.4; } +.displayShowTable { + table-layout: auto; + width: 100%; + border-collapse: collapse; + border-spacing: 0; + text-align: center; + border: none; + empty-cells: show; + color: #000 !important; +} + +.displayShowTable.display_show { + clear:both +} + +.displayShowTable th.row-seasonheader { + border: none !important; + background-color: #222 !important; + color: #fff !important; + padding-top: 15px !important; + text-align: left !important; +} + .sickbeardTable { table-layout: auto; width: 100%; diff --git a/gui/slick/images/providers/scenetime.png b/gui/slick/images/providers/scenetime.png new file mode 100644 index 0000000000000000000000000000000000000000..32e729817493ca9411fc78e698a8b9c37a081780 Binary files /dev/null and b/gui/slick/images/providers/scenetime.png differ diff --git a/gui/slick/interfaces/default/config_general.tmpl b/gui/slick/interfaces/default/config_general.tmpl index e4651ce30aa262b1cb2560b3fc16242296476a06..3948f456e21749e879ee65fb39d716018da2ba6f 100644 --- a/gui/slick/interfaces/default/config_general.tmpl +++ b/gui/slick/interfaces/default/config_general.tmpl @@ -299,20 +299,11 @@ <label for="filter_row"> <span class="component-title">Filter Row</span> <span class="component-desc"> - <input type="checkbox" name="filter_row" id="display_filesize" #if $sickbeard.FILTER_ROW == True then 'checked="checked"' else ''#/> + <input type="checkbox" name="filter_row" id="filter_row" #if $sickbeard.FILTER_ROW == True then 'checked="checked"' else ''#/> <p>Add a filter row to the show display on the home page</p> </span> </label> </div> - <div class="field-pair"> - <label for="display_filesize"> - <span class="component-title">Display Filesizes</span> - <span class="component-desc"> - <input type="checkbox" name="display_filesize" id="display_filesize" #if $sickbeard.DISPLAY_FILESIZE == True then 'checked="checked"' else ''#/> - <p>display filesizes for downloaded episodes on show page</p> - </span> - </label> - </div> <div class="field-pair"> <label for="coming_eps_missed_range"> <span class="component-title">Missed episodes range</span> @@ -774,4 +765,4 @@ //--> </script> -#include $os.path.join($sickbeard.PROG_DIR, 'gui/slick/interfaces/default/inc_bottom.tmpl') +#include $os.path.join($sickbeard.PROG_DIR, 'gui/slick/interfaces/default/inc_bottom.tmpl') \ No newline at end of file diff --git a/gui/slick/interfaces/default/config_notifications.tmpl b/gui/slick/interfaces/default/config_notifications.tmpl index 3fb21751c7305d6e4c29eb30956b2c06d3784230..5b82eb6f80418f4d54a018a137acb373086d87f2 100644 --- a/gui/slick/interfaces/default/config_notifications.tmpl +++ b/gui/slick/interfaces/default/config_notifications.tmpl @@ -175,6 +175,73 @@ </div> <div id="content_use_plex"> + <div class="field-pair"> + <label for="plex_server_token"> + <span class="component-title">Plex Media Server Auth Token</span> + <input type="text" name="plex_server_token" id="plex_server_token" value="$sickbeard.PLEX_SERVER_TOKEN" class="form-control input-sm input250" /> + </label> + <label> + <span class="component-title"> </span> + <span class="component-desc">Auth Token used by plex</span> + </label> + <label> + <span class="component-title"> </span> + <span class="component-desc">(<a href="<%= anon_url('https://support.plex.tv/hc/en-us/articles/204059436-Finding-your-account-token-X-Plex-Token') %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;"><u>Finding your account token</u></a>)</span> + </label> + </div> + <div class="component-group" style="padding: 0; min-height: 130px"> + <div class="field-pair"> + <label for="plex_username"> + <span class="component-title">Server/client username</span> + <span class="component-desc"> + <input type="text" name="plex_username" id="plex_username" value="$sickbeard.PLEX_USERNAME" class="form-control input-sm input250" /> + <p>blank = no authentication</p> + </span> + </label> + </div> + <div class="field-pair"> + <label for="plex_password"> + <span class="component-title">Server/client password</span> + <span class="component-desc"> + <input type="password" name="plex_password" id="plex_password" value="#echo '*' * len($sickbeard.PLEX_PASSWORD)#" class="form-control input-sm input250" /> + <p>blank = no authentication</p> + </span> + </label> + </div> + </div> + + <div class="component-group" style="padding: 0; min-height: 50px"> + <div class="field-pair"> + <label for="plex_update_library"> + <span class="component-title">Update server library</span> + <span class="component-desc"> + <input type="checkbox" class="enabler" name="plex_update_library" id="plex_update_library" #if $sickbeard.PLEX_UPDATE_LIBRARY then 'checked="checked" ' else ''#/> + <p>update Plex Media Server library when a download finishes</p> + </span> + </label> + </div> + <div id="content_plex_update_library"> + <div class="field-pair"> + <label for="plex_server_host"> + <span class="component-title">Plex Media Server IP:Port</span> + <span class="component-desc"> + <input type="text" name="plex_server_host" id="plex_server_host" value="<%= re.sub(r'\b,\b', ', ', sickbeard.PLEX_SERVER_HOST) %>" class="form-control input-sm input350" /> + <div class="clear-left"> + <p>one or more hosts running Plex Media Server<br />(eg. 192.168.1.1:32400, 192.168.1.2:32400)</p> + </div> + </span> + </label> + </div> + + <div class="field-pair"> + <div class="testNotification" id="testPMS-result">Click below to test Plex server(s)</div> + <input class="btn" type="button" value="Test Plex Server" id="testPMS" /> + <input type="submit" class="config_submitter btn" value="Save Changes" /> + <div class="clear-left"> </div> + </div> + </div> + </div> + <div class="field-pair"> <label for="plex_notify_onsnatch"> <span class="component-title">Notify on snatch</span> @@ -202,78 +269,25 @@ </span> </label> </div> - <div class="field-pair"> - <label for="plex_update_library"> - <span class="component-title">Update library</span> - <span class="component-desc"> - <input type="checkbox" name="plex_update_library" id="plex_update_library" #if $sickbeard.PLEX_UPDATE_LIBRARY then "checked=\"checked\"" else ""# /> - <p>update Plex Media Server library when a download finishes ?</p> - </span> - </label> - </div> - <div class="field-pair"> - <label for="plex_server_host"> - <span class="component-title">Plex Media Server IP:Port</span> - <input type="text" name="plex_server_host" id="plex_server_host" value="$sickbeard.PLEX_SERVER_HOST" class="form-control input-sm input250" /> - </label> - <label> - <span class="component-title"> </span> - <span class="component-desc">host running Plex Media Server (eg. 192.168.1.100:32400)</span> - </label> - </div> - <div class="field-pair"> - <label for="plex_server_token"> - <span class="component-title">Plex Media Server Auth Token</span> - <input type="text" name="plex_server_token" id="plex_server_token" value="$sickbeard.PLEX_SERVER_TOKEN" class="form-control input-sm input250" /> - </label> - <label> - <span class="component-title"> </span> - <span class="component-desc">Auth Token used by plex</span> - </label> - <label> - <span class="component-title"> </span> - <span class="component-desc">(<a href="<%= anon_url('https://support.plex.tv/hc/en-us/articles/204059436-Finding-your-account-token-X-Plex-Token') %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;"><u>Finding your account token</u></a>)</span> - </label> - </div> <div class="field-pair"> <label for="plex_host"> <span class="component-title">Plex Client IP:Port</span> - <input type="text" name="plex_host" id="plex_host" value="$sickbeard.PLEX_HOST" class="form-control input-sm input350" /> - </label> - <label> - <span class="component-title"> </span> - <span class="component-desc">host running Plex Client (eg. 192.168.1.100:3000)</span> - </label> - <label> - <span class="component-title"> </span> - <span class="component-desc">(multiple host strings must be separated by commas)</span> - </label> - </div> - <div class="field-pair"> - <label for="plex_username"> - <span class="component-title">Plex Client username</span> - <input type="text" name="plex_username" id="plex_username" value="$sickbeard.PLEX_USERNAME" class="form-control input-sm input250" /> - </label> - <label> - <span class="component-title"> </span> - <span class="component-desc">username for your Plex client API (blank for none)</span> + <span class="component-desc"> + <input type="text" name="plex_host" id="plex_host" value="$sickbeard.PLEX_HOST" class="form-control input-sm input350" /> + <div class="clear-left"> + <p>one or more hosts running Plex client<br />(eg. 192.168.1.100:3000, 192.168.1.101:3000)</p> + </div> + </span> </label> </div> + <div class="field-pair"> - <label for="plex_password"> - <span class="component-title">Plex Client password</span> - <input type="password" name="plex_password" id="plex_password" value="$sickbeard.PLEX_PASSWORD" class="form-control input-sm input250" /> - </label> - <label> - <span class="component-title"> </span> - <span class="component-desc">password for your Plex client API (blank for none)</span> - </label> + <div class="testNotification" id="testPMC-result">Click below to test Plex client(s)</div> + <input class="btn" type="button" value="Test Plex Client" id="testPMC" /> + <input type="submit" class="config_submitter btn" value="Save Changes" /> + <div class=clear-left><p>Note: some Plex clients <b class="boldest">do not</b> support notifications e.g. Plexapp for Samsung TVs</p></div> </div> - <div class="testNotification" id="testPLEX-result">Click below to test.</div> - <input class="btn" type="button" value="Test Plex Client" id="testPLEX" /> - <input type="submit" class="config_submitter btn" value="Save Changes" /> </div><!-- /content_use_plex --> - </fieldset> </div><!-- /plex component-group --> diff --git a/gui/slick/interfaces/default/config_search.tmpl b/gui/slick/interfaces/default/config_search.tmpl index 2c745ab0d6baa96d22c8737f44480376da7a329b..4b2a8648aac37e51ad2db537b06272dd66dd83a4 100755 --- a/gui/slick/interfaces/default/config_search.tmpl +++ b/gui/slick/interfaces/default/config_search.tmpl @@ -448,8 +448,8 @@ <span class="component-title">Send .torrent files to:</span> <span class="component-desc"> <select name="torrent_method" id="torrent_method" class="form-control input-sm"> -#set $torrent_method_text = {'blackhole': "Black hole", 'utorrent': "uTorrent", 'transmission': "Transmission", 'deluge': "Deluge", 'download_station': "Synology DS", 'rtorrent': "rTorrent"} -#for $curAction in ('blackhole', 'utorrent', 'transmission', 'deluge', 'download_station', 'rtorrent'): +#set $torrent_method_text = {'blackhole': "Black hole", 'utorrent': "uTorrent", 'transmission': "Transmission", 'deluge': "Deluge", 'download_station': "Synology DS", 'rtorrent': "rTorrent", 'qbittorrent': "qbittorrent"} +#for $curAction in ('blackhole', 'utorrent', 'transmission', 'deluge', 'download_station', 'rtorrent', 'qbittorrent'): #set $selected = $html_selected if $sickbeard.TORRENT_METHOD == $curAction else '' <option value="$curAction"$selected>$torrent_method_text[$curAction]</option> #end for diff --git a/gui/slick/interfaces/default/displayShow.tmpl b/gui/slick/interfaces/default/displayShow.tmpl index 3ed05330b56b85fe151a1061e3bc978fb5767836..358389403416a46f0658d96a6e7e579b9c9443f9 100644 --- a/gui/slick/interfaces/default/displayShow.tmpl +++ b/gui/slick/interfaces/default/displayShow.tmpl @@ -7,6 +7,7 @@ #import os.path, os #import datetime #import urllib +#import ntpath #set global $title=$show.name ##set global $header = '<a></a>' % @@ -47,487 +48,533 @@ }); }); #end raw - - \$.fn.generateStars = function() { - return this.each(function(i,e){\$(e).html(\$('<span/>').width(\$(e).text()*12));}); - }; + + \$.fn.generateStars = function() { + return this.each(function(i,e){\$(e).html(\$('<span/>').width(\$(e).text()*12));}); + }; + + \$('.imdbstars').generateStars(); + + + + #if $show.is_anime: + \$("#animeTable").tablesorter({ + #else: + \$("#showTable").tablesorter({ + #end if + widgets: ['saveSort', 'stickyHeaders', 'columnSelector'], + widgetOptions : { + columnSelector_saveColumns: true, + columnSelector_layout : '<br/><label><input type="checkbox">{name}</label>', + columnSelector_mediaquery: false, + columnSelector_cssChecked : 'checked' + }, + }); - \$('.imdbstars').generateStars(); - + + + \$('#popover') + .popover({ + placement: 'bottom', + html: true, // required if content has HTML + content: '<div id="popover-target"></div>' + }) + // bootstrap popover event triggered when the popover opens + .on('shown.bs.popover', function () { + #if $show.is_anime: + \$.tablesorter.columnSelector.attachTo( \$('#animeTable'), '#popover-target'); + #else: + \$.tablesorter.columnSelector.attachTo( \$('#showTable'), '#popover-target'); + #end if + }); }); //--> </script> - <div class="pull-left form-inline"> - Change Show: - <div class="navShow"><img id="prevShow" src="$sbRoot/images/prev.png" alt="<<" title="Prev Show" /></div> - <select id="pickShow" class="form-control form-control-inline input-sm"> - #for $curShowList in $sortedShowLists: - #set $curShowType = $curShowList[0] - #set $curShowList = $curShowList[1] - - #if len($sortedShowLists) > 1: - <optgroup label="$curShowType"> - #end if - #for $curShow in $curShowList: - <option value="$curShow.indexerid" #if $curShow == $show then "selected=\"selected\"" else ""#>$curShow.name</option> - #end for - #if len($sortedShowLists) > 1: - </optgroup> - #end if - #end for - </select> - <div class="navShow"><img id="nextShow" src="$sbRoot/images/next.png" alt=">>" title="Next Show" /></div> - </div> - - <div class="clearfix"></div> - - <div id="showtitle" data-showname="$show.name"> - <h1 class="title" id="scene_exception_$show.indexerid">$show.name</h1> - </div> - - - #if $seasonResults: - ##There is a special/season_0?## - #if int($seasonResults[-1]["season"]) == 0: - #set $season_special = 1 - #else: - #set $season_special = 0 - #end if - - #if not $sickbeard.DISPLAY_SHOW_SPECIALS and $season_special: - $seasonResults.pop(-1) - #end if - - <span class="h2footer displayspecials pull-right"> - #if $season_special: - Display Specials: - #if sickbeard.DISPLAY_SHOW_SPECIALS: - <a class="inner" href="$sbRoot/toggleDisplayShowSpecials/?show=$show.indexerid">Hide</a> - #else: - <a class="inner" href="$sbRoot/toggleDisplayShowSpecials/?show=$show.indexerid">Show</a> - #end if - #end if - </span> - - <div class="h2footer pull-right"> - <span> - #if (len($seasonResults) > 14): - <select id="seasonJump" class="form-control input-sm" style="position: relative; top: -4px;"> - <option value="jump">Jump to Season</option> - #for $seasonNum in $seasonResults: - <option value="#season-$seasonNum["season"]">#if int($seasonNum["season"]) == 0 then "Specials" else "Season " + str($seasonNum["season"])#</option> - #end for - </select> - #else: - Season: - #for $seasonNum in $seasonResults: - #if int($seasonNum["season"]) == 0: - <a href="#season-$seasonNum["season"]">Specials</a> - #else: - <a href="#season-$seasonNum["season"]">${str($seasonNum["season"])}</a> - #end if - #if $seasonNum != $seasonResults[-1]: - <span class="separator">|</span> - #end if - #end for - #end if - </span> - - #end if - </div> - - <div class="clearfix"></div> + <div class="pull-left form-inline"> + Change Show: + <div class="navShow"><img id="prevShow" src="$sbRoot/images/prev.png" alt="<<" title="Prev Show" /></div> + <select id="pickShow" class="form-control form-control-inline input-sm"> + #for $curShowList in $sortedShowLists: + #set $curShowType = $curShowList[0] + #set $curShowList = $curShowList[1] + + #if len($sortedShowLists) > 1: + <optgroup label="$curShowType"> + #end if + #for $curShow in $curShowList: + <option value="$curShow.indexerid" #if $curShow == $show then "selected=\"selected\"" else ""#>$curShow.name</option> + #end for + #if len($sortedShowLists) > 1: + </optgroup> + #end if + #end for + </select> + <div class="navShow"><img id="nextShow" src="$sbRoot/images/next.png" alt=">>" title="Next Show" /></div> + </div> + + <div class="clearfix"></div> + + <div id="showtitle" data-showname="$show.name"> + <h1 class="title" id="scene_exception_$show.indexerid">$show.name</h1> + </div> + + + #if $seasonResults: + ##There is a special/season_0?## + #if int($seasonResults[-1]["season"]) == 0: + #set $season_special = 1 + #else: + #set $season_special = 0 + #end if + + #if not $sickbeard.DISPLAY_SHOW_SPECIALS and $season_special: + $seasonResults.pop(-1) + #end if + + <span class="h2footer displayspecials pull-right"> + #if $season_special: + Display Specials: + #if sickbeard.DISPLAY_SHOW_SPECIALS: + <a class="inner" href="$sbRoot/toggleDisplayShowSpecials/?show=$show.indexerid">Hide</a> + #else: + <a class="inner" href="$sbRoot/toggleDisplayShowSpecials/?show=$show.indexerid">Show</a> + #end if + #end if + </span> + + <div class="h2footer pull-right"> + <span> + #if (len($seasonResults) > 14): + <select id="seasonJump" class="form-control input-sm" style="position: relative; top: -4px;"> + <option value="jump">Jump to Season</option> + #for $seasonNum in $seasonResults: + <option value="#season-$seasonNum["season"]">#if int($seasonNum["season"]) == 0 then "Specials" else "Season " + str($seasonNum["season"])#</option> + #end for + </select> + #else: + Season: + #for $seasonNum in $seasonResults: + #if int($seasonNum["season"]) == 0: + <a href="#season-$seasonNum["season"]">Specials</a> + #else: + <a href="#season-$seasonNum["season"]">${str($seasonNum["season"])}</a> + #end if + #if $seasonNum != $seasonResults[-1]: + <span class="separator">|</span> + #end if + #end for + #end if + </span> + + #end if + </div> + + <div class="clearfix"></div> #if $show_message: - <div class="alert alert-info"> - $show_message - </div> + <div class="alert alert-info"> + $show_message + </div> #end if - - <div id="container"> - <div id="posterCol"> - <a href="$sbRoot/showPoster/?show=$show.indexerid&which=poster" rel="dialog" title="View Poster for $show.name"><img src="$sbRoot/showPoster/?show=$show.indexerid&which=poster_thumb" class="tvshowImg" alt=""/></a> - </div> - - <div id="showCol"> - - <div id="showinfo"> + + <div id="container"> + <div id="posterCol"> + <a href="$sbRoot/showPoster/?show=$show.indexerid&which=poster" rel="dialog" title="View Poster for $show.name"><img src="$sbRoot/showPoster/?show=$show.indexerid&which=poster_thumb" class="tvshowImg" alt=""/></a> + </div> + + <div id="showCol"> + + <div id="showinfo"> #if 'rating' in $show.imdb_info: #set $rating_tip = str($show.imdb_info['rating']) + " / 10" + " Stars" + "<br />" + str($show.imdb_info['votes']) + " Votes" - <span class="imdbstars" qtip-content="$rating_tip">$show.imdb_info['rating']</span> + <span class="imdbstars" qtip-content="$rating_tip">$show.imdb_info['rating']</span> #end if - + #set $_show = $show #if not $show.imdbid - <span>($show.startyear) - $show.runtime minutes - </span> + <span>($show.startyear) - $show.runtime minutes - </span> #else #if 'country_codes' in $show.imdb_info: #for $country in $show.imdb_info['country_codes'].split('|') - <img src="$sbRoot/images/blank.png" class="country-flag flag-${$country}" width="16" height="11" style="margin-left: 3px; vertical-align:middle;" /> + <img src="$sbRoot/images/blank.png" class="country-flag flag-${$country}" width="16" height="11" style="margin-left: 3px; vertical-align:middle;" /> #end for #end if #if 'year' in $show.imdb_info: - <span>($show.imdb_info['year']) - $show.imdb_info['runtimes'] minutes - </span> + <span>($show.imdb_info['year']) - $show.imdb_info['runtimes'] minutes - </span> #end if - <a href="<%= anon_url('http://www.imdb.com/title/', _show.imdbid) %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;" title="http://www.imdb.com/title/$show.imdbid"><img alt="[imdb]" height="16" width="16" src="$sbRoot/images/imdb.png" style="margin-top: -1px; vertical-align:middle;"/></a> + <a href="<%= anon_url('http://www.imdb.com/title/', _show.imdbid) %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;" title="http://www.imdb.com/title/$show.imdbid"><img alt="[imdb]" height="16" width="16" src="$sbRoot/images/imdb.png" style="margin-top: -1px; vertical-align:middle;"/></a> #end if - <a href="<%= anon_url(sickbeard.indexerApi(_show.indexer).config['show_url'], _show.indexerid) %>" onclick="window.open(this.href, '_blank'); return false;" title="$sickbeard.indexerApi($show.indexer).config["show_url"]$show.indexerid"><img alt="$sickbeard.indexerApi($show.indexer).name" height="16" width="16" src="$sbRoot/images/$sickbeard.indexerApi($show.indexer).config["icon"] "style="margin-top: -1px; vertical-align:middle;"/></a> + <a href="<%= anon_url(sickbeard.indexerApi(_show.indexer).config['show_url'], _show.indexerid) %>" onclick="window.open(this.href, '_blank'); return false;" title="$sickbeard.indexerApi($show.indexer).config["show_url"]$show.indexerid"><img alt="$sickbeard.indexerApi($show.indexer).name" height="16" width="16" src="$sbRoot/images/$sickbeard.indexerApi($show.indexer).config["icon"] "style="margin-top: -1px; vertical-align:middle;"/></a> #if $xem_numbering or $xem_absolute_numbering: - <a href="<%= anon_url('http://thexem.de/search?q=', _show.name) %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;" title="http://thexem.de/search?q-$show.name"><img alt="[xem]" height="16" width="16" src="$sbRoot/images/xem.png" style="margin-top: -1px; vertical-align:middle;"/></a> + <a href="<%= anon_url('http://thexem.de/search?q=', _show.name) %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;" title="http://thexem.de/search?q-$show.name"><img alt="[xem]" height="16" width="16" src="$sbRoot/images/xem.png" style="margin-top: -1px; vertical-align:middle;"/></a> #end if - </div> - - <div id="tags"> - <ul class="tags"> - #if not $show.imdbid - #if $show.genre: - #for $genre in $show.genre[1:-1].split('|') - <a href="<%= anon_url('http://trakt.tv/shows/popular/', genre.lower()) %>" target="_blank" title="View other popular $genre shows on trakt.tv."><li>$genre</li></a> - #end for - #end if - #end if - #if 'year' in $show.imdb_info: - #for $imdbgenre in $show.imdb_info['genres'].replace('Sci-Fi','Science-Fiction').split('|') - <a href="<%= anon_url('http://trakt.tv/shows/popular/', imdbgenre.lower()) %>" target="_blank" title="View other popular $imdbgenre shows on trakt.tv."><li>$imdbgenre</li></a> - #end for - #end if - </ul> - </div> - - <div id="summary"> - <table class="summaryTable pull-left"> - #set $anyQualities, $bestQualities = $Quality.splitQuality(int($show.quality)) - <tr><td class="showLegend">Quality: </td><td> - #if $show.quality in $qualityPresets: - <span class="quality $qualityPresetStrings[$show.quality]">$qualityPresetStrings[$show.quality]</span> - #else: - #if $anyQualities: - <i>Initial:</i> <%=", ".join([Quality.qualityStrings[x] for x in sorted(anyQualities)])%> #if $bestQualities then " </br> " else ""# - #end if - #if $bestQualities: - <i>Replace with:</i> <%=", ".join([Quality.qualityStrings[x] for x in sorted(bestQualities)])%> - #end if - #end if - - #if $show.network and $show.airs: - <tr><td class="showLegend">Originally Airs: </td><td>$show.airs #if not $network_timezones.test_timeformat($show.airs) then " <font color='#FF0000'><b>(invalid Timeformat)</b></font> " else ""# on $show.network</td></tr> - #else if $show.network: - <tr><td class="showLegend">Originally Airs: </td><td>$show.network</td></tr> - #else if $show.airs: - <tr><td class="showLegend">Originally Airs: </td><td>>$show.airs #if not $network_timezones.test_timeformat($show.airs) then " <font color='#FF0000'><b>(invalid Timeformat)</b></font> " else ""#</td></tr> - #end if - <tr><td class="showLegend">Show Status: </td><td>$show.status</td></tr> - <tr><td class="showLegend">Default EP Status: </td><td>$statusStrings[$show.default_ep_status]</td></tr> - #if $showLoc[1]: - <tr><td class="showLegend">Location: </td><td>$showLoc[0]</td></tr> - #else: - <tr><td class="showLegend"><span style="color: red;">Location: </span></td><td><span style="color: red;">$showLoc[0]</span> (dir is missing)</td></tr> - #end if - <tr><td class="showLegend">Scene Name:</td><td>#if $show.exceptions then $exceptions_string else $show.name#</td></tr> - - #if $show.rls_require_words: - <tr><td class="showLegend">Required Words: </td><td>#echo $show.rls_require_words#</td></tr> - #end if - #if $show.rls_ignore_words: - <tr><td class="showLegend">Ignored Words: </td><td>#echo $show.rls_ignore_words#</td></tr> - #end if - #if $bwl and $bwl.get_white_keywords_for("release_group"): - <tr><td class="showLegend">Wanted Group#if len($bwl.get_white_keywords_for("release_group"))>1 then "s" else ""#:</td> - <td>#echo ', '.join($bwl.get_white_keywords_for("release_group"))#</td> - </tr> - #end if - #if $bwl and $bwl.get_black_keywords_for("release_group"): - <tr><td class="showLegend">Unwanted Group#if len($bwl.get_black_keywords_for("release_group"))>1 then "s" else ""#:</td> - <td>#echo ', '.join($bwl.get_black_keywords_for("release_group"))#</td> - </tr> - #end if - - <tr><td class="showLegend">Size:</td><td>$sickbeard.helpers.pretty_filesize(sickbeard.helpers.get_size($showLoc[0]))</td></tr> - - </table> - - <table style="width:180px; float: right; vertical-align: middle; height: 100%;"> - <tr><td class="showLegend">Info Language:</td><td><img src="$sbRoot/images/flags/${show.lang}.png" width="16" height="11" alt="$show.lang" title="$show.lang" /></td></tr> - #if $sickbeard.USE_SUBTITLES - <tr><td class="showLegend">Subtitles: </td><td><img src="$sbRoot/images/#if int($show.subtitles) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> - #end if - <tr><td class="showLegend">Flat Folders: </td><td><img src="$sbRoot/images/#if $show.flatten_folders == 1 or $sickbeard.NAMING_FORCE_FOLDERS then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> + </div> + + <div id="tags"> + <ul class="tags"> + #if not $show.imdbid + #if $show.genre: + #for $genre in $show.genre[1:-1].split('|') + <a href="<%= anon_url('http://trakt.tv/shows/popular/', genre.lower()) %>" target="_blank" title="View other popular $genre shows on trakt.tv."><li>$genre</li></a> + #end for + #end if + #end if + #if 'year' in $show.imdb_info: + #for $imdbgenre in $show.imdb_info['genres'].replace('Sci-Fi','Science-Fiction').split('|') + <a href="<%= anon_url('http://trakt.tv/shows/popular/', imdbgenre.lower()) %>" target="_blank" title="View other popular $imdbgenre shows on trakt.tv."><li>$imdbgenre</li></a> + #end for + #end if + </ul> + </div> + + <div id="summary"> + <table class="summaryTable pull-left"> + #set $anyQualities, $bestQualities = $Quality.splitQuality(int($show.quality)) + <tr><td class="showLegend">Quality: </td><td> + #if $show.quality in $qualityPresets: + <span class="quality $qualityPresetStrings[$show.quality]">$qualityPresetStrings[$show.quality]</span> + #else: + #if $anyQualities: + <i>Initial:</i> <%=", ".join([Quality.qualityStrings[x] for x in sorted(anyQualities)])%> #if $bestQualities then " </br> " else ""# + #end if + #if $bestQualities: + <i>Replace with:</i> <%=", ".join([Quality.qualityStrings[x] for x in sorted(bestQualities)])%> + #end if + #end if + + #if $show.network and $show.airs: + <tr><td class="showLegend">Originally Airs: </td><td>$show.airs #if not $network_timezones.test_timeformat($show.airs) then " <font color='#FF0000'><b>(invalid Timeformat)</b></font> " else ""# on $show.network</td></tr> + #else if $show.network: + <tr><td class="showLegend">Originally Airs: </td><td>$show.network</td></tr> + #else if $show.airs: + <tr><td class="showLegend">Originally Airs: </td><td>>$show.airs #if not $network_timezones.test_timeformat($show.airs) then " <font color='#FF0000'><b>(invalid Timeformat)</b></font> " else ""#</td></tr> + #end if + <tr><td class="showLegend">Show Status: </td><td>$show.status</td></tr> + <tr><td class="showLegend">Default EP Status: </td><td>$statusStrings[$show.default_ep_status]</td></tr> + #if $showLoc[1]: + <tr><td class="showLegend">Location: </td><td>$showLoc[0]</td></tr> + #else: + <tr><td class="showLegend"><span style="color: red;">Location: </span></td><td><span style="color: red;">$showLoc[0]</span> (dir is missing)</td></tr> + #end if + <tr><td class="showLegend">Scene Name:</td><td>#if $show.exceptions then $exceptions_string else $show.name#</td></tr> + + #if $show.rls_require_words: + <tr><td class="showLegend">Required Words: </td><td>#echo $show.rls_require_words#</td></tr> + #end if + #if $show.rls_ignore_words: + <tr><td class="showLegend">Ignored Words: </td><td>#echo $show.rls_ignore_words#</td></tr> + #end if + #if $bwl and $bwl.get_white_keywords_for("release_group"): + <tr><td class="showLegend">Wanted Group#if len($bwl.get_white_keywords_for("release_group"))>1 then "s" else ""#:</td> + <td>#echo ', '.join($bwl.get_white_keywords_for("release_group"))#</td> + </tr> + #end if + #if $bwl and $bwl.get_black_keywords_for("release_group"): + <tr><td class="showLegend">Unwanted Group#if len($bwl.get_black_keywords_for("release_group"))>1 then "s" else ""#:</td> + <td>#echo ', '.join($bwl.get_black_keywords_for("release_group"))#</td> + </tr> + #end if + + <tr><td class="showLegend">Size:</td><td>$sickbeard.helpers.pretty_filesize(sickbeard.helpers.get_size($showLoc[0]))</td></tr> + + </table> + + <table style="width:180px; float: right; vertical-align: middle; height: 100%;"> + <tr><td class="showLegend">Info Language:</td><td><img src="$sbRoot/images/flags/${show.lang}.png" width="16" height="11" alt="$show.lang" title="$show.lang" /></td></tr> + #if $sickbeard.USE_SUBTITLES + <tr><td class="showLegend">Subtitles: </td><td><img src="$sbRoot/images/#if int($show.subtitles) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> + #end if + <tr><td class="showLegend">Flat Folders: </td><td><img src="$sbRoot/images/#if $show.flatten_folders == 1 or $sickbeard.NAMING_FORCE_FOLDERS then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> <tr><td class="showLegend">Paused: </td><td><img src="$sbRoot/images/#if int($show.paused) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> - <tr><td class="showLegend">Air-by-Date: </td><td><img src="$sbRoot/images/#if int($show.air_by_date) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> - <tr><td class="showLegend">Sports: </td><td><img src="$sbRoot/images/#if int($show.is_sports) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> - <tr><td class="showLegend">Anime: </td><td><img src="$sbRoot/images/#if int($show.is_anime) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> - <tr><td class="showLegend">DVD Order: </td><td><img src="$sbRoot/images/#if int($show.dvdorder) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> - <tr><td class="showLegend">Scene Numbering: </td><td><img src="$sbRoot/images/#if int($show.scene) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> - #if $anyQualities + $bestQualities - <tr><td class="showLegend">Archive First Match: </td><td><img src="$sbRoot/images/#if int($show.archive_firstmatch) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> - #end if - </table> - </div> - </div> - </div> - - <div class="clearfix"></div> - - <div class="pull-left" > - Change selected episodes to:</br> - <select id="statusSelect" class="form-control form-control-inline input-sm"> - #for $curStatus in [$WANTED, $SKIPPED, $ARCHIVED, $IGNORED, $FAILED] + sorted($Quality.DOWNLOADED): - #if $curStatus == $DOWNLOADED: - #continue - #end if - <option value="$curStatus">$statusStrings[$curStatus]</option> - #end for - </select> - <input type="hidden" id="showID" value="$show.indexerid" /> - <input type="hidden" id="indexer" value="$show.indexer" /> - <input class="btn btn-inline" type="button" id="changeStatus" value="Go" /> - </div> - - </br> - - <div class="pull-right clearfix" id="checkboxControls"> - <div style="padding-bottom: 5px;"> - <label for="wanted"><span class="wanted"><input type="checkbox" id="wanted" checked="checked" /> Wanted: <b>$epCounts[$Overview.WANTED]</b></span></label> - <label for="qual"><span class="qual"><input type="checkbox" id="qual" checked="checked" /> Low Quality: <b>$epCounts[$Overview.QUAL]</b></span></label> - <label for="good"><span class="good"><input type="checkbox" id="good" checked="checked" /> Downloaded: <b>$epCounts[$Overview.GOOD]</b></span></label> - <label for="skipped"><span class="skipped"><input type="checkbox" id="skipped" checked="checked" /> Skipped: <b>$epCounts[$Overview.SKIPPED]</b></span></label> - <label for="snatched"><span class="snatched"><input type="checkbox" id="snatched" checked="checked" /> Snatched: <b>$epCounts[$Overview.SNATCHED]</b></span></label> - </div> - - <div class="pull-right" > - <button class="btn btn-xs seriesCheck">Select Filtered Episodes</button> - <button class="btn btn-xs clearAll">Clear All</button> - </div> - </div> -<br /> - -<table class="sickbeardTable display_show" cellspacing="0" border="0" cellpadding="0"> + <tr><td class="showLegend">Air-by-Date: </td><td><img src="$sbRoot/images/#if int($show.air_by_date) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> + <tr><td class="showLegend">Sports: </td><td><img src="$sbRoot/images/#if int($show.is_sports) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> + <tr><td class="showLegend">Anime: </td><td><img src="$sbRoot/images/#if int($show.is_anime) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> + <tr><td class="showLegend">DVD Order: </td><td><img src="$sbRoot/images/#if int($show.dvdorder) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> + <tr><td class="showLegend">Scene Numbering: </td><td><img src="$sbRoot/images/#if int($show.scene) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> + #if $anyQualities + $bestQualities + <tr><td class="showLegend">Archive First Match: </td><td><img src="$sbRoot/images/#if int($show.archive_firstmatch) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr> + #end if + </table> + </div> + </div> + </div> + + <div class="clearfix"></div> + + <div class="pull-left" > + Change selected episodes to:</br> + <select id="statusSelect" class="form-control form-control-inline input-sm"> + #for $curStatus in [$WANTED, $SKIPPED, $ARCHIVED, $IGNORED, $FAILED] + sorted($Quality.DOWNLOADED): + #if $curStatus == $DOWNLOADED: + #continue + #end if + <option value="$curStatus">$statusStrings[$curStatus]</option> + #end for + </select> + <input type="hidden" id="showID" value="$show.indexerid" /> + <input type="hidden" id="indexer" value="$show.indexer" /> + <input class="btn btn-inline" type="button" id="changeStatus" value="Go" /> + </div> + </br> + + <div class="pull-right clearfix" id="checkboxControls"> + <div style="padding-bottom: 5px;"> + <label for="wanted"><span class="wanted"><input type="checkbox" id="wanted" checked="checked" /> Wanted: <b>$epCounts[$Overview.WANTED]</b></span></label> + <label for="qual"><span class="qual"><input type="checkbox" id="qual" checked="checked" /> Low Quality: <b>$epCounts[$Overview.QUAL]</b></span></label> + <label for="good"><span class="good"><input type="checkbox" id="good" checked="checked" /> Downloaded: <b>$epCounts[$Overview.GOOD]</b></span></label> + <label for="skipped"><span class="skipped"><input type="checkbox" id="skipped" checked="checked" /> Skipped: <b>$epCounts[$Overview.SKIPPED]</b></span></label> + <label for="snatched"><span class="snatched"><input type="checkbox" id="snatched" checked="checked" /> Snatched: <b>$epCounts[$Overview.SNATCHED]</b></span></label> + </div> + + <button id="popover" type="button" class="btn btn-xs">Select Columns</button> + <div class="pull-right" > + <button class="btn btn-xs seriesCheck">Select Filtered Episodes</button> + <button class="btn btn-xs clearAll">Clear All</button> + </div> + </div> +<br /> +<br /> +<br /> + +<table #if not $show.is_anime then "id=\"showTable\"" else "id=\"animeTable\""# class="displayShowTable display_show" cellspacing="0" border="0" cellpadding="0"> #set $curSeason = -1 #set $odd = 0 + #for $epResult in $sqlResults: + #set $epStr = str($epResult["season"]) + "x" + str($epResult["episode"]) + #if not $epStr in $epCats: + #continue + #end if + #if not $sickbeard.DISPLAY_SHOW_SPECIALS and int($epResult["season"]) == 0: + #continue + #end if + #set $scene = False + #set $scene_anime = False + #if not $show.air_by_date and not $show.is_sports and not $show.is_anime and $show.is_scene: + #set $scene = True + #elif not $show.air_by_date and not $show.is_sports and $show.is_anime and $show.is_scene: + #set $scene_anime = True + #end if + #set ($dfltSeas, $dfltEpis, $dfltAbsolute) = (0, 0, 0) + #if (epResult["season"], epResult["episode"]) in $xem_numbering: + #set ($dfltSeas, $dfltEpis) = $xem_numbering[(epResult["season"], epResult["episode"])] + #end if + #if epResult["absolute_number"] in $xem_absolute_numbering: + #set $dfltAbsolute = $xem_absolute_numbering[epResult["absolute_number"]] + #end if + + #if epResult["absolute_number"] in $scene_absolute_numbering: + #set $scAbsolute = $scene_absolute_numbering[epResult["absolute_number"]] + #set $dfltAbsNumbering = False + #else + #set $scAbsolute = $dfltAbsolute + #set $dfltAbsNumbering = True + #end if - #for $epResult in $sqlResults: - #set $epStr = str($epResult["season"]) + "x" + str($epResult["episode"]) - #if not $epStr in $epCats: - #continue - #end if - - #if not $sickbeard.DISPLAY_SHOW_SPECIALS and int($epResult["season"]) == 0: - #continue - #end if - - #set $scene = False - #set $scene_anime = False - #if not $show.air_by_date and not $show.is_sports and not $show.is_anime and $show.is_scene: - #set $scene = True - #elif not $show.air_by_date and not $show.is_sports and $show.is_anime and $show.is_scene: - #set $scene_anime = True - #end if - - #set ($dfltSeas, $dfltEpis, $dfltAbsolute) = (0, 0, 0) - - #if (epResult["season"], epResult["episode"]) in $xem_numbering: - #set ($dfltSeas, $dfltEpis) = $xem_numbering[(epResult["season"], epResult["episode"])] - #end if - - #if epResult["absolute_number"] in $xem_absolute_numbering: - #set $dfltAbsolute = $xem_absolute_numbering[epResult["absolute_number"]] - #end if - - #if epResult["absolute_number"] in $scene_absolute_numbering: - #set $scAbsolute = $scene_absolute_numbering[epResult["absolute_number"]] - #set $dfltAbsNumbering = False - #else - #set $scAbsolute = $dfltAbsolute - #set $dfltAbsNumbering = True - #end if - - #if (epResult["season"], epResult["episode"]) in $scene_numbering: - #set ($scSeas, $scEpis) = $scene_numbering[(epResult["season"], epResult["episode"])] - #set $dfltEpNumbering = False - #else - #set ($scSeas, $scEpis) = ($dfltSeas, $dfltEpis) - #set $dfltEpNumbering = True - #end if - - #if int($epResult["season"]) != $curSeason: - <tr> - <th class="row-seasonheader" colspan="13" style="width: auto;"><h3><a name="season-$epResult["season"]"></a>#if int($epResult["season"]) == 0 then "Specials" else "Season " + str($epResult["season"])#</h3></th> - </tr> - - <tr id="season-$epResult["season"]-cols" class="seasoncols"> - <th class="col-checkbox"><input type="checkbox" class="seasonCheck" id="$epResult["season"]" /></th> - <th class="col-metadata">NFO</th> - <th class="col-metadata">TBN</th> - <th class="col-ep">Episode</th> - #if $show.is_anime: - <th class="col-ep">Absolute</th> - #end if - #if $scene: - <th class="col-ep">Scene</th> - #end if - #if $scene_anime: - <th class="col-ep">Scene Absolute</th> - #end if - <th class="col-name" - #if ($sickbeard.DISPLAY_FILESIZE == True): - style="min-width: 190px" - #end if - >Name</th> - #if ($sickbeard.DISPLAY_FILESIZE == True): - <th class="col-ep">Size</th> - #end if - <th class="col-airdate">Airdate</th> - #if $sickbeard.DOWNLOAD_URL - <th class="col-ep">Download</th> - #end if - #if $sickbeard.USE_SUBTITLES and $show.subtitles: - <th class="col-subtitles">Subtitles</th> - #end if - <th class="col-status">Status</th> - <th class="col-search">Search</th> - </tr> - #set $curSeason = int($epResult["season"]) - #end if - - #set $epLoc = $epResult["location"] - - <tr class="$Overview.overviewStrings[$epCats[$epStr]] season-$curSeason seasonstyle"> - - <td class="col-checkbox"> - - #if int($epResult["status"]) != $UNAIRED - <input type="checkbox" class="epCheck" id="<%=str(epResult["season"])+'x'+str(epResult["episode"])%>" name="<%=str(epResult["season"]) +"x"+str(epResult["episode"]) %>" /> - #end if - </td> - - <td align="center"><img src="$sbRoot/images/#if $epResult["hasnfo"] == 1 then "nfo.gif\" alt=\"Y" else "nfo-no.gif\" alt=\"N"#" width="23" height="11" /></td> - - <td align="center"><img src="$sbRoot/images/#if $epResult["hastbn"] == 1 then "tbn.gif\" alt=\"Y" else "tbn-no.gif\" alt=\"N"#" width="23" height="11" /></td> - - <td align="center"> - #if $epLoc and $show._location and $epLoc.lower().startswith($show._location.lower()): - #set $epLoc = $epLoc[len($show._location)+1:] - #elif $epLoc and (not $epLoc.lower().startswith($show._location.lower()) or not $show._location): - #set $epLoc = $epLoc - #end if - - #if $epLoc != "" and $epLoc != None: - <span title="$epLoc" class="addQTip">$epResult["episode"]</span> - #else - $epResult["episode"] - #end if - </td> - - #if $show.is_anime: - <td align="center">$epResult["absolute_number"]</td> - #end if - - #if $scene: - <td align="center"> - <input type="text" placeholder="<%=str(dfltSeas) + 'x' + str(dfltEpis)%>" size="6" maxlength="8" - class="sceneSeasonXEpisode form-control input-scene" data-for-season="$epResult["season"]" data-for-episode="$epResult["episode"]" - id="sceneSeasonXEpisode_$show.indexerid<%="_"+str(epResult["season"])+"_"+str(epResult["episode"])%>" - title="Change the value here if scene numbering differs from the indexer episode numbering" - #if $dfltEpNumbering: - value="" - #else - value="<%=str(scSeas) + 'x' + str(scEpis)%>" - #end if - style="padding: 0; text-align: center; max-width: 60px;" /> - </td> - #elif $scene_anime: - <td align="center"> - <input type="text" placeholder="<%=str(dfltAbsolute)%>" size="6" maxlength="8" - class="sceneAbsolute form-control input-scene" data-for-absolute="$epResult["absolute_number"]" - id="sceneAbsolute_$show.indexerid<%="_"+str(epResult["absolute_number"])%>" - title="Change the value here if scene absolute numbering differs from the indexer absolute numbering" - #if $dfltAbsNumbering: - value="" - #else - value="<%=str(scAbsolute)%>" - #end if - style="padding: 0; text-align: center; max-width: 60px;" /> - </td> - #end if - - <td class="col-name"> - #if $epResult["description"] != "" and $epResult["description"] != None: - <img src="$sbRoot/images/info32.png" width="16" height="16" class="plotInfo" alt="" id="plot_info_$show.indexerid<%="_" + str(epResult["season"]) + "_" + str(epResult["episode"])%>" /> - #else: - <img src="$sbRoot/images/info32.png" width="16" height="16" class="plotInfoNone" alt="" /> - #end if - $epResult["name"] - </td> - - #if ($sickbeard.DISPLAY_FILESIZE == True): - <td class="col-ep"> - #if $epResult["file_size"]: - #set $file_size = $sickbeard.helpers.pretty_filesize($epResult["file_size"]) - $file_size - #end if - </td> - #end if - <td class="col-airdate"> - <span class="${fuzzydate}">#if int($epResult['airdate']) == 1 then 'never' else $sbdatetime.sbdatetime.sbfdate($sbdatetime.sbdatetime.convert_to_setting($network_timezones.parse_date_time($epResult['airdate'],$show.airs,$show.network)))#</span> - </td> - - #if $sickbeard.DOWNLOAD_URL and $epResult['location'] - <td> - #set $filename = $epResult['location'] - #for $rootDir in $sickbeard.ROOT_DIRS.split('|') - #if $rootDir.startswith('/') - #set $filename = $filename.replace($rootDir, "") - #end if - #end for - #set $filename = $sickbeard.DOWNLOAD_URL + $urllib.quote($filename.encode('utf8')) - <center><a href="$filename">Download</a></center> - </td> - #elif $sickbeard.DOWNLOAD_URL - <td></td> - #end if - - #if $sickbeard.USE_SUBTITLES and $show.subtitles: - <td class="col-subtitles" align="center"> - #if $epResult["subtitles"]: - #for $sub_lang in subliminal.language.language_list([x.strip() for x in $epResult["subtitles"].split(',') if x != ""]): - #if sub_lang.alpha2 != "" - <img src="$sbRoot/images/flags/${sub_lang.alpha2}.png" width="16" height="11" alt="${sub_lang}" /> + #if (epResult["season"], epResult["episode"]) in $scene_numbering: + #set ($scSeas, $scEpis) = $scene_numbering[(epResult["season"], epResult["episode"])] + #set $dfltEpNumbering = False + #else + #set ($scSeas, $scEpis) = ($dfltSeas, $dfltEpis) + #set $dfltEpNumbering = True + #end if + + #set $epLoc = $epResult["location"] + + #if int($epResult["season"]) != $curSeason: + #if $curSeason == -1: + <thead> + <tr class="seasoncols" style="display:none;"> + <th data-sorter="false" data-priority="critical" class="col-checkbox"><input type="checkbox" class="seasonCheck"/></th> + <th data-sorter="false" class="col-metadata">NFO</th> + <th data-sorter="false" class="col-metadata">TBN</th> + <th data-sorter="false" class="col-ep">Episode</th> + <th data-sorter="false" #if not $show.is_anime then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Absolute</th> + <th data-sorter="false" #if not $scene then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Scene</th> + <th data-sorter="false" #if not $scene_anime then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Scene Absolute</th> + <th data-sorter="false" class="col-name">Name</th> + <th data-sorter="false" class="col-name columnSelector-false">File Name</th> + <th data-sorter="false" class="col-ep columnSelector-false">Size</th> + <th data-sorter="false" class="col-airdate">Airdate</th> + <th data-sorter="false" #if not $sickbeard.DOWNLOAD_URL then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Download</th> + <th data-sorter="false" #if not $sickbeard.USE_SUBTITLES then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Subtitles</th> + <th data-sorter="false" class="col-status">Status</th> + <th data-sorter="false" class="col-search">Search</th> + </tr> + </thead> + <tbody class="tablesorter-no-sort"> + <tr> + <th class="row-seasonheader displayShowTable" colspan="13" style="width: auto;"><h3><a name="season-$epResult["season"]"></a>#if int($epResult["season"]) == 0 then "Specials" else "Season " + str($epResult["season"])#</h3></th> + </tr> + <tr id="season-$epResult["season"]-cols" class="seasoncols"> + <th class="col-checkbox"><input type="checkbox" class="seasonCheck" id="$epResult["season"]" /></th> + <th class="col-metadata">NFO</th> + <th class="col-metadata">TBN</th> + <th class="col-ep">Episode</th> + <th class="col-ep">Absolute</th> + <th class="col-ep">Scene</th> + <th class="col-ep">Scene Absolute</th> + <th class="col-name">Name</th> + <th class="col-name">File Name</th> + <th class="col-ep">Size</th> + <th class="col-airdate">Airdate</th> + <th class="col-ep">Download</th> + <th class="col-ep">Subtitles</th> + <th class="col-status">Status</th> + <th class="col-search">Search</th> + </tr> + #else: + </tbody> + <tbody class="tablesorter-no-sort"> + <tr> + <th class="row-seasonheader displayShowTable" colspan="13" style="width: auto;"><h3><a name="season-$epResult["season"]"></a>#if int($epResult["season"]) == 0 then "Specials" else "Season " + str($epResult["season"])#</h3></th> + </tr> + <tr id="season-$epResult["season"]-cols" class="seasoncols"> + <th class="col-checkbox"><input type="checkbox" class="seasonCheck" id="$epResult["season"]" /></th> + <th class="col-metadata">NFO</th> + <th class="col-metadata">TBN</th> + <th class="col-ep">Episode</th> + <th class="col-ep">Absolute</th> + <th class="col-ep">Scene</th> + <th class="col-ep">Scene Absolute</th> + <th class="col-name">Name</th> + <th class="col-name">File Name</th> + <th class="col-ep">Size</th> + <th class="col-airdate">Airdate</th> + <th class="col-ep">Download</th> + <th class="col-ep">Subtitles</th> + <th class="col-status">Status</th> + <th class="col-search">Search</th> + </tr> + #end if + </tbody> + <tbody> + #set $curSeason = int($epResult["season"]) + #end if + #set $epLoc = $epResult["location"] + <tr class="$Overview.overviewStrings[$epCats[$epStr]] season-$curSeason seasonstyle"> + <td class="col-checkbox"> + #if int($epResult["status"]) != $UNAIRED + <input type="checkbox" class="epCheck" id="<%=str(epResult["season"])+'x'+str(epResult["episode"])%>" name="<%=str(epResult["season"]) +"x"+str(epResult["episode"]) %>" /> + #end if + </td> + <td align="center"><img src="$sbRoot/images/#if $epResult["hasnfo"] == 1 then "nfo.gif\" alt=\"Y" else "nfo-no.gif\" alt=\"N"#" width="23" height="11" /></td> + <td align="center"><img src="$sbRoot/images/#if $epResult["hastbn"] == 1 then "tbn.gif\" alt=\"Y" else "tbn-no.gif\" alt=\"N"#" width="23" height="11" /></td> + <td align="center"> + #if $epLoc and $show._location and $epLoc.lower().startswith($show._location.lower()): + #set $epLoc = $epLoc[len($show._location)+1:] + #elif $epLoc and (not $epLoc.lower().startswith($show._location.lower()) or not $show._location): + #set $epLoc = $epLoc + #end if + + #if $epLoc != "" and $epLoc != None: + <span title="$epLoc" class="addQTip">$epResult["episode"]</span> #else - <img src="$sbRoot/images/flags/unknown.png" width="16" height="11" alt="Unknown" /> + $epResult["episode"] #end if - #end for - #end if - </td> - #end if - - #set $curStatus, $curQuality = $Quality.splitCompositeStatus(int($epResult["status"])) - #if $curQuality != Quality.NONE: - <td class="col-status">$statusStrings[$curStatus] <span class="quality $Quality.qualityStrings[$curQuality].replace("720p","HD720p").replace("1080p","HD1080p").replace("RawHD TV", "RawHD").replace("HD TV", "HD720p")">$Quality.qualityStrings[$curQuality]</span></td> - #else: - <td class="col-status">$statusStrings[$curStatus]</td> - #end if - - <td class="col-search"> - #if int($epResult["season"]) != 0: - #if ( int($epResult["status"]) in $Quality.SNATCHED or int($epResult["status"]) in $Quality.DOWNLOADED ) and $sickbeard.USE_FAILED_DOWNLOADS: - <a class="epRetry" id="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" name="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" href="retryEpisode?show=$show.indexerid&season=$epResult["season"]&episode=$epResult["episode"]"><img src="$sbRoot/images/search16.png" height="16" alt="retry" title="Retry Download" /></a> - #else: - <a class="epSearch" id="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" name="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" href="searchEpisode?show=$show.indexerid&season=$epResult["season"]&episode=$epResult["episode"]"><img src="$sbRoot/images/search16.png" width="16" height="16" alt="search" title="Manual Search" /></a> - #end if - #end if - - #if $sickbeard.USE_SUBTITLES and $show.subtitles and len(set(str($epResult["subtitles"]).split(',')).intersection(set($subtitles.wantedLanguages()))) < len($subtitles.wantedLanguages()) and $epResult["location"] - <a class="epSubtitlesSearch" href="searchEpisodeSubtitles?show=$show.indexerid&season=$epResult["season"]&episode=$epResult["episode"]"><img src="$sbRoot/images/closed_captioning.png" height="16" alt="search subtitles" title="Search Subtitles" /></a> - #end if - </td> - </tr> - - #end for - + </td> + <td align="center">$epResult["absolute_number"]</td> + <td align="center"> + <input type="text" placeholder="<%=str(dfltSeas) + 'x' + str(dfltEpis)%>" size="6" maxlength="8" + class="sceneSeasonXEpisode form-control input-scene" data-for-season="$epResult["season"]" data-for-episode="$epResult["episode"]" + id="sceneSeasonXEpisode_$show.indexerid<%="_"+str(epResult["season"])+"_"+str(epResult["episode"])%>" + title="Change the value here if scene numbering differs from the indexer episode numbering" + #if $dfltEpNumbering: + value="" + #else + value="<%=str(scSeas) + 'x' + str(scEpis)%>" + #end if + style="padding: 0; text-align: center; max-width: 60px;" /> + </td> + <td align="center"> + <input type="text" placeholder="<%=str(dfltAbsolute)%>" size="6" maxlength="8" + class="sceneAbsolute form-control input-scene" data-for-absolute="$epResult["absolute_number"]" + id="sceneAbsolute_$show.indexerid<%="_"+str(epResult["absolute_number"])%>" + title="Change the value here if scene absolute numbering differs from the indexer absolute numbering" + #if $dfltAbsNumbering: + value="" + #else + value="<%=str(scAbsolute)%>" + #end if + style="padding: 0; text-align: center; max-width: 60px;" /> + </td> + <td class="col-name"> + #if $epResult["description"] != "" and $epResult["description"] != None: + <img src="$sbRoot/images/info32.png" width="16" height="16" class="plotInfo" alt="" id="plot_info_$show.indexerid<%="_" + str(epResult["season"]) + "_" + str(epResult["episode"])%>" /> + #else: + <img src="$sbRoot/images/info32.png" width="16" height="16" class="plotInfoNone" alt="" /> + #end if + $epResult["name"] + </td> + <td class="col-name]"> + #if $epResult['location'] + #set $filename = $epResult['location'] + #for $rootDir in $sickbeard.ROOT_DIRS.split('|') + #if $rootDir.startswith('/') + #set $filename = ntpath.basename($filename) + #end if + #end for + $filename + #end if + </td> + <td class="col-ep"> + #if $epResult["file_size"]: + #set $file_size = $sickbeard.helpers.pretty_filesize($epResult["file_size"]) + $file_size + #end if + </td> + <td class="col-airdate"> + <span class="${fuzzydate}">#if int($epResult['airdate']) == 1 then 'never' else $sbdatetime.sbdatetime.sbfdate($sbdatetime.sbdatetime.convert_to_setting($network_timezones.parse_date_time($epResult['airdate'],$show.airs,$show.network)))#</span> + </td> + <td> + #if $sickbeard.DOWNLOAD_URL and $epResult['location'] + #if $epResult['location'] + #set $filename = $epResult['location'] + #for $rootDir in $sickbeard.ROOT_DIRS.split('|') + #if $rootDir.startswith('/') + #set $filename = $filename.replace($rootDir, "") + #end if + #end for + #set $filename = $sickbeard.DOWNLOAD_URL + $urllib.quote($filename.encode('utf8')) + <center><a href="$filename">Download</a></center> + #end if + #end if + </td> + <td class="col-subtitles" align="center"> + #if $epResult["subtitles"]: + #for $sub_lang in subliminal.language.language_list([x.strip() for x in $epResult["subtitles"].split(',') if x != ""]): + #if sub_lang.alpha2 != "" + <img src="$sbRoot/images/flags/${sub_lang.alpha2}.png" width="16" height="11" alt="${sub_lang}" /> + #else + <img src="$sbRoot/images/flags/unknown.png" width="16" height="11" alt="Unknown" /> + #end if + #end for + #end if + </td> + #set $curStatus, $curQuality = $Quality.splitCompositeStatus(int($epResult["status"])) + #if $curQuality != Quality.NONE: + <td class="col-status">$statusStrings[$curStatus] <span class="quality $Quality.qualityStrings[$curQuality].replace("720p","HD720p").replace("1080p","HD1080p").replace("RawHD TV", "RawHD").replace("HD TV", "HD720p")">$Quality.qualityStrings[$curQuality]</span></td> + #else: + <td class="col-status">$statusStrings[$curStatus]</td> + #end if + <td class="col-search"> + #if int($epResult["season"]) != 0: + #if ( int($epResult["status"]) in $Quality.SNATCHED or int($epResult["status"]) in $Quality.DOWNLOADED ) and $sickbeard.USE_FAILED_DOWNLOADS: + <a class="epRetry" id="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" name="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" href="retryEpisode?show=$show.indexerid&season=$epResult["season"]&episode=$epResult["episode"]"><img src="$sbRoot/images/search16.png" height="16" alt="retry" title="Retry Download" /></a> + #else: + <a class="epSearch" id="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" name="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" href="searchEpisode?show=$show.indexerid&season=$epResult["season"]&episode=$epResult["episode"]"><img src="$sbRoot/images/search16.png" width="16" height="16" alt="search" title="Manual Search" /></a> + #end if + #end if + #if $sickbeard.USE_SUBTITLES and $show.subtitles and len(set(str($epResult["subtitles"]).split(',')).intersection(set($subtitles.wantedLanguages()))) < len($subtitles.wantedLanguages()) and $epResult["location"] + <a class="epSubtitlesSearch" href="searchEpisodeSubtitles?show=$show.indexerid&season=$epResult["season"]&episode=$epResult["episode"]"><img src="$sbRoot/images/closed_captioning.png" height="16" alt="search subtitles" title="Search Subtitles" /></a> + #end if + </td> + </tr> + #end for + </tbody> </table> - + <!--Begin - Bootstrap Modal--> <div id="manualSearchModalFailed" class="modal fade"> diff --git a/gui/slick/interfaces/default/editShow.tmpl b/gui/slick/interfaces/default/editShow.tmpl index aafd072732e54d27d7e9909c90c3b68af3dd2403..ff40985259a683ceebc192deca44800e4d3e1b2d 100644 --- a/gui/slick/interfaces/default/editShow.tmpl +++ b/gui/slick/interfaces/default/editShow.tmpl @@ -78,7 +78,7 @@ <b>Info Language:</b><br /> (this will only affect the language of the retrieved metadata file contents and episode filenames)<br /> -<select name="indexerLang" id="indexerLangSelect" class="form-control form-control-inline input-sm bfh-languages" data-language="en" data-available="#echo ','.join($sickbeard.indexerApi().config['valid_languages'])#"></select><br /> +<select name="indexerLang" id="indexerLangSelect" class="form-control form-control-inline input-sm bfh-languages" data-language="#echo $sickbeard.INDEXER_DEFAULT_LANGUAGE#" data-available="#echo ','.join($sickbeard.indexerApi().config['valid_languages'])#"></select><br /> <br /> <b>Flatten files (no folders): </b> <input type="checkbox" name="flatten_folders" #if $show.flatten_folders == 1 and not $sickbeard.NAMING_FORCE_FOLDERS then "checked=\"checked\"" else ""# #if $sickbeard.NAMING_FORCE_FOLDERS then "disabled=\"disabled\"" else ""#/><br /> diff --git a/gui/slick/interfaces/default/inc_top.tmpl b/gui/slick/interfaces/default/inc_top.tmpl index 2bc4e16b18687a751426fdf66885b47f573b24da..f46ad2d63d7d2eb66fd1eacb81262034c3cb4cef 100644 --- a/gui/slick/interfaces/default/inc_top.tmpl +++ b/gui/slick/interfaces/default/inc_top.tmpl @@ -119,6 +119,9 @@ \$("#SubMenu a:contains('Notification')").addClass('btn').html('<span class="ui-icon ui-icon-note pull-left"></span> Notifications'); \$("#SubMenu a:contains('Update show in KODI')").addClass('btn').html('<span class="submenu-icon-kodi pull-left"></span> Update show in KODI'); \$("#SubMenu a[href$='/home/updateKODI/']").addClass('btn').html('<span class="submenu-icon-kodi pull-left"></span> Update KODI'); + \$("#SubMenu a:contains('Pause')").addClass('btn').html('<span class="ui-icon ui-icon-pause pull-left"></span> Pause'); + \$("#SubMenu a:contains('Resume')").addClass('btn').html('<span class="ui-icon ui-icon-play pull-left"></span> Resume'); + } \$(document).ready(function() { diff --git a/gui/slick/js/ajaxNotifications.js b/gui/slick/js/ajaxNotifications.js index 92b2fc22b853b4d2c880e66d7743424e30d410af..f1b38d7cdfab4198322c67ad140574a0332e5ab2 100644 --- a/gui/slick/js/ajaxNotifications.js +++ b/gui/slick/js/ajaxNotifications.js @@ -1,31 +1,42 @@ -var message_url = sbRoot + '/ui/get_messages'; +var message_url = sbRoot + '/ui/get_messages', + test = !1; +PNotify.prototype.options.addclass = 'stack-bottomright'; +PNotify.prototype.options.buttons.closer_hover = !1; +PNotify.prototype.options.delay = 5000; +PNotify.prototype.options.desktop = {desktop: !0, icon: ''}; +PNotify.prototype.options.hide = !0; +PNotify.prototype.options.history = !1; +PNotify.prototype.options.shadow = !1; +PNotify.prototype.options.stack = {dir1: 'up', dir2: 'left', firstpos1: 25, firstpos2: 25}; PNotify.prototype.options.styling = 'jqueryui'; -PNotify.prototype.options.buttons.closer_hover = false; -PNotify.prototype.options.delay = 4000; PNotify.prototype.options.width = '340px'; -PNotify.prototype.options.shadow = false; -PNotify.prototype.options.addclass = 'stack-bottomright'; -PNotify.prototype.options.stack = {'dir1': 'up', 'dir2': 'left', 'firstpos1': 25, 'firstpos2': 25}; +PNotify.desktop.permission(); + +function displayPNotify(type, title, message) { + var notification = new PNotify({ + type: type, title: title, + text: message.replace(/<br[\s\/]*(?:\s[^>]*)?>/ig, "\n") + .replace(/<[\/]?b(?:\s[^>]*)?>/ig, '*') + .replace(/<i(?:\s[^>]*)?>/ig, '[').replace(/<[\/]i>/ig, ']') + .replace(/<(?:[\/]?ul|\/li)(?:\s[^>]*)?>/ig, '').replace(/<li(?:\s[^>]*)?>/ig, "\n" + '* ') + }); +} function check_notifications() { - if(document.visibilityState == 'visible') { - $.getJSON(message_url, function(data){ - $.each(data, function(name,data){ - new PNotify({ - type: data.type, - hide: data.type == 'notice', - title: data.title, - text: data.message, - history: false - }); - }); - }); - } - - setTimeout(check_notifications, 3000) + if ('visible' == document.visibilityState) { + $.getJSON(message_url, function (data) { + $.each(data, function (name, data) { + displayPNotify(data.type, data.title, data.message) + }); + }); + } + setTimeout(check_notifications, 3000) } $(document).ready(function(){ - check_notifications(); + check_notifications(); + if (test) { + displayPNotify('notice', 'test', 'test<br/><i class="test-class">hello <b>world</b></i><ul><li>item 1</li><li>item 2</li></ul>'); + } }); \ No newline at end of file diff --git a/gui/slick/js/configNotifications.js b/gui/slick/js/configNotifications.js index 2d88287e9bd77a970b7a28648c6e81ef214ac399..f0b52ea97cb035e6755c8f8a574122d57dc43961 100644 --- a/gui/slick/js/configNotifications.js +++ b/gui/slick/js/configNotifications.js @@ -56,24 +56,44 @@ $(document).ready(function(){ }); }); - $('#testPLEX').click(function () { - var plex_host = $.trim($('#plex_host').val()); - var plex_username = $.trim($('#plex_username').val()); - var plex_password = $.trim($('#plex_password').val()); - if (!plex_host) { - $('#testPLEX-result').html('Please fill out the necessary fields above.'); + $('#testPMC').click(function () { + var plex_host = $.trim($('#plex_host').val()); + var plex_username = $.trim($('#plex_username').val()); + var plex_password = $.trim($('#plex_password').val()); + if (!plex_host) { + $('#testPMC-result').html('Please fill out the necessary fields above.'); $('#plex_host').addClass('warning'); - return; - } - $('#plex_host').removeClass('warning'); + return; + } + $('#plex_host').removeClass('warning'); $(this).prop('disabled', true); - $('#testPLEX-result').html(loading); - $.get(sbRoot + '/home/testPLEX', {'host': plex_host, 'username': plex_username, 'password': plex_password}) - .done(function (data) { - $('#testPLEX-result').html(data); - $('#testPLEX').prop('disabled', false); - }); - }); + $('#testPMC-result').html(loading); + $.get(sbRoot + '/home/testPMC', {'host': plex_host, 'username': plex_username, 'password': plex_password}) + .done(function (data) { + $('#testPMC-result').html(data); + $('#testPMC').prop('disabled', false); + }); + }); + + $('#testPMS').click(function () { + var plex_server_host = $.trim($('#plex_server_host').val()); + var plex_username = $.trim($('#plex_username').val()); + var plex_password = $.trim($('#plex_password').val()); + var plex_server_token = $.trim($('#plex_server_token').val()); + if (!plex_server_host) { + $('#testPMS-result').html('Please fill out the necessary fields above.'); + $('#plex_server_host').addClass('warning'); + return; + } + $('#plex_server_host').removeClass('warning'); + $(this).prop('disabled', true); + $('#testPMS-result').html(loading); + $.get(sbRoot + '/home/testPMS', {'host': plex_server_host, 'username': plex_username, 'password': plex_password, 'plex_server_token': plex_server_token}) + .done(function (data) { + $('#testPMS-result').html(data); + $('#testPMS').prop('disabled', false); + }); + }); $('#testBoxcar').click(function() { var boxcar_username = $.trim($('#boxcar_username').val()); diff --git a/gui/slick/js/configSearch.js b/gui/slick/js/configSearch.js index f586a488c832b21043c42adbc7921a3f95083776..1840bd68d60d86c72343173fdca9a51afde7f8bb 100644 --- a/gui/slick/js/configSearch.js +++ b/gui/slick/js/configSearch.js @@ -150,6 +150,12 @@ $(document).ready(function(){ $(torrent_verify_rtorrent).show(); $(torrent_auth_type_option).show(); //$('#directory_title').text(client + directory); + } else if ('qbittorrent' == selectedProvider){ + client = 'qbittorrent'; + $(torrent_path_option).hide(); + $(torrent_label_option).hide(); + $(torrent_label_anime_option).hide(); + $('#host_desc_torrent').text('URL to your qbittorrent client (e.g. http://localhost:8080)'); } $('#host_title').text(client + host); $('#username_title').text(client + username); diff --git a/gui/slick/js/displayShow.js b/gui/slick/js/displayShow.js index 4825c2cbcc45420f9fd67a2aa8199a504fd0c04f..fb13af78ae2b75e31d81470b27313683f018dd9f 100644 --- a/gui/slick/js/displayShow.js +++ b/gui/slick/js/displayShow.js @@ -259,5 +259,5 @@ $(document).ready(function () { sceneAbsolute = m[1]; } setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute); - }); + }); }); diff --git a/gui/slick/js/lib/pnotify.custom.min.js b/gui/slick/js/lib/pnotify.custom.min.js index 704c85ae41ad58fb3d4bd2142a226ec6683d8b94..f927d92712aa970d3baeabde56c84448bb888a33 100644 Binary files a/gui/slick/js/lib/pnotify.custom.min.js and b/gui/slick/js/lib/pnotify.custom.min.js differ diff --git a/sickbeard/__init__.py b/sickbeard/__init__.py index 2a9b9e0900c2db4036ad9e7cc7d570173352e8a3..f20e8ab68c229664d0557e75c5ddc01f0e539ef9 100755 --- a/sickbeard/__init__.py +++ b/sickbeard/__init__.py @@ -38,7 +38,7 @@ from sickbeard import providers, metadata, config, webserveInit from sickbeard.providers.generic import GenericProvider from providers import ezrss, btn, newznab, womble, thepiratebay, oldpiratebay, torrentleech, kat, iptorrents, \ omgwtfnzbs, scc, hdtorrents, torrentday, hdbits, hounddawgs, nextgen, speedcd, nyaatorrents, animenzb, torrentbytes, animezb, \ - freshontv, morethantv, bitsoup, t411, tokyotoshokan, shazbat, rarbg, alpharatio, tntvillage, binsearch, eztv + freshontv, morethantv, bitsoup, t411, tokyotoshokan, shazbat, rarbg, alpharatio, tntvillage, binsearch, eztv, scenetime from sickbeard.config import CheckSection, check_setting_int, check_setting_str, check_setting_float, ConfigMigrator, \ naming_ep_type from sickbeard import searchBacklog, showUpdater, versionChecker, properFinder, autoPostProcesser, \ @@ -482,7 +482,6 @@ COMING_EPS_LAYOUT = None COMING_EPS_DISPLAY_PAUSED = False COMING_EPS_SORT = None COMING_EPS_MISSED_RANGE = None -DISPLAY_FILESIZE = False FUZZY_DATING = False TRIM_ZERO = False DATE_PRESET = None @@ -802,7 +801,7 @@ def initialize(consoleLogging=True): NZB_METHOD = 'blackhole' TORRENT_METHOD = check_setting_str(CFG, 'General', 'torrent_method', 'blackhole') - if TORRENT_METHOD not in ('blackhole', 'utorrent', 'transmission', 'deluge', 'download_station', 'rtorrent'): + if TORRENT_METHOD not in ('blackhole', 'utorrent', 'transmission', 'deluge', 'download_station', 'rtorrent', 'qbittorrent'): TORRENT_METHOD = 'blackhole' DOWNLOAD_PROPERS = bool(check_setting_int(CFG, 'General', 'download_propers', 1)) @@ -1123,7 +1122,6 @@ def initialize(consoleLogging=True): COMING_EPS_DISPLAY_PAUSED = bool(check_setting_int(CFG, 'GUI', 'coming_eps_display_paused', 0)) COMING_EPS_SORT = check_setting_str(CFG, 'GUI', 'coming_eps_sort', 'date') COMING_EPS_MISSED_RANGE = check_setting_int(CFG, 'GUI', 'coming_eps_missed_range', 7) - DISPLAY_FILESIZE = bool(check_setting_int(CFG, 'GUI', 'display_filesize', 0)) FUZZY_DATING = bool(check_setting_int(CFG, 'GUI', 'fuzzy_dating', 0)) TRIM_ZERO = bool(check_setting_int(CFG, 'GUI', 'trim_zero', 0)) DATE_PRESET = check_setting_str(CFG, 'GUI', 'date_preset', '%x') @@ -2040,7 +2038,6 @@ def save_config(): new_config['GUI']['coming_eps_display_paused'] = int(COMING_EPS_DISPLAY_PAUSED) new_config['GUI']['coming_eps_sort'] = COMING_EPS_SORT new_config['GUI']['coming_eps_missed_range'] = int(COMING_EPS_MISSED_RANGE) - new_config['GUI']['display_filesize'] = int(DISPLAY_FILESIZE) new_config['GUI']['fuzzy_dating'] = int(FUZZY_DATING) new_config['GUI']['trim_zero'] = int(TRIM_ZERO) new_config['GUI']['date_preset'] = DATE_PRESET diff --git a/sickbeard/classes.py b/sickbeard/classes.py index 92bbc23e11cb067beee83b1757eaf503fa10c8f1..ec8fce996f5191dcae3f9ae1936893d66a37dcf8 100644 --- a/sickbeard/classes.py +++ b/sickbeard/classes.py @@ -276,6 +276,6 @@ class UIError(): """ def __init__(self, message): - self.title = sys.exc_info()[-2] + self.title = sys.exc_info()[-2] or message self.message = message self.time = datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S') \ No newline at end of file diff --git a/sickbeard/clients/__init__.py b/sickbeard/clients/__init__.py index 84646ac32955d2beb27b92ecf9ab13d609be43fc..432efc797127ae729371bfd42f03f4b63d00f4f8 100644 --- a/sickbeard/clients/__init__.py +++ b/sickbeard/clients/__init__.py @@ -20,7 +20,8 @@ __all__ = ['utorrent', 'transmission', 'deluge', 'download_station', - 'rtorrent' + 'rtorrent', + 'qbittorrent' ] import sickbeard @@ -29,13 +30,28 @@ from os import sys # Mapping error status codes to official W3C names http_error_code = { + 100: 'Continue', + 101: 'Switching Protocols', + 102: 'Processing', + 200: 'OK', + 201: 'Created', + 202: 'Accepted', + 203: 'Non-Authoritative Information', + 204: 'No Content', + 205: 'Reset Content', + 206: 'Partial Content', + 207: 'Multi-Status', + 208: 'Already Reported', + 226: 'IM Used', 300: 'Multiple Choices', 301: 'Moved Permanently', 302: 'Found', 303: 'See Other', 304: 'Not Modified', 305: 'Use Proxy', + 306: 'Switch Proxy', 307: 'Temporary Redirect', + 308: 'Permanent Redirect', 400: 'Bad Request', 401: 'Unauthorized', 402: 'Payment Required', @@ -54,13 +70,45 @@ http_error_code = { 415: 'Unsupported Media Type', 416: 'Requested Range Not Satisfiable', 417: 'Expectation Failed', + 418: 'Im a teapot', + 419: 'Authentication Timeout', + 420: 'Enhance Your Calm', + 422: 'Unprocessable Entity', + 423: 'Locked', + 424: 'Failed Dependency', + 426: 'Upgrade Required', + 428: 'Precondition Required', + 429: 'Too Many Requests', + 431: 'Request Header Fields Too Large', + 440: 'Login Timeout', + 444: 'No Response', + 449: 'Retry With', + 450: 'Blocked by Windows Parental Controls', + 451: 'Redirect', + 451: 'Unavailable For Legal Reasons', + 494: 'Request Header Too Large', + 495: 'Cert Error', + 496: 'No Cert', + 497: 'HTTP to HTTPS', + 498: 'Token expired/invalid', + 499: 'Client Closed Request', + 499: 'Token required', 500: 'Internal Server Error', 501: 'Not Implemented', 502: 'Bad Gateway', 503: 'Service Unavailable', 504: 'Gateway Timeout', 505: 'HTTP Version Not Supported', - 524: 'Request to host timedout waiting for reply back' + 506: 'Variant Also Negotiates', + 507: 'Insufficient Storage', + 508: 'Loop Detected', + 509: 'Bandwidth Limit Exceeded', + 510: 'Not Extended', + 511: 'Network Authentication Required', + 522: 'Cloudfare Connection timed out', + 524: 'Request to host timedout waiting for reply back', + 598: 'Network read timeout error', + 599: 'Network connect timeout error ' } default_host = {'utorrent': 'http://localhost:8000', @@ -68,6 +116,7 @@ default_host = {'utorrent': 'http://localhost:8000', 'deluge': 'http://localhost:8112', 'download_station': 'http://localhost:5000', 'rtorrent': 'scgi://localhost:5000', + 'qbittorrent': 'http://localhost:8080' } @@ -82,4 +131,4 @@ def getClientIstance(name): module = getClientModule(name) className = module.api.__class__.__name__ - return getattr(module, className) \ No newline at end of file + return getattr(module, className) diff --git a/sickbeard/clients/generic.py b/sickbeard/clients/generic.py index 7940ab17da62cc70cf64a1271facf9acc9b40d5b..3dfc58dde954a368363ced1c354d0eae3e2d755d 100644 --- a/sickbeard/clients/generic.py +++ b/sickbeard/clients/generic.py @@ -61,7 +61,7 @@ class GenericClient(object): logger.log(self.name + u': Invalid HTTP Request ' + str(e), logger.ERROR) return False except requests.exceptions.Timeout, e: - logger.log(self.name + u': Connection Timeout ' + str(e), logger.ERROR) + logger.log(self.name + u': Connection Timeout ' + str(e), logger.WARNING) return False except Exception, e: logger.log(self.name + u': Unknown exception raised when send torrent to ' + self.name + ': ' + str(e), diff --git a/sickbeard/clients/qbittorrent.py b/sickbeard/clients/qbittorrent.py new file mode 100644 index 0000000000000000000000000000000000000000..5923c274cc75b298107b5a839f074d978a0447aa --- /dev/null +++ b/sickbeard/clients/qbittorrent.py @@ -0,0 +1,73 @@ +# Author: Mr_Orange <mr_orange@hotmail.it> +# URL: http://code.google.com/p/sickbeard/ +# +# This file is part of SickRage. +# +# SickRage is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# SickRage is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with SickRage. If not, see <http://www.gnu.org/licenses/>. + +import sickbeard +from sickbeard import logger +from sickbeard.clients.generic import GenericClient +from lib import requests +from lib.requests.auth import HTTPDigestAuth + +class qbittorrentAPI(GenericClient): + def __init__(self, host=None, username=None, password=None): + + super(qbittorrentAPI, self).__init__('qbittorrent', host, username, password) + + self.url = self.host + self.session.auth = HTTPDigestAuth(self.username, self.password); + + def _get_auth(self): + + try: + self.response = self.session.get(self.host, verify=False) + self.auth = self.response.content + except: + return None + + return self.auth if not self.response.status_code == 404 else None + + def _add_torrent_uri(self, result): + + self.url = self.host+'command/download' + data = {'urls': result.url} + return self._request(method='post', data=data) + + def _add_torrent_file(self, result): + + self.url = self.host+'command/upload' + files = {'torrents': (result.name + '.torrent', result.content)} + return self._request(method='post', files=files) + + def _set_torrent_priority(self, result): + + self.url = self.host+'command/decreasePrio ' + if result.priority == 1: + self.url = self.host+'command/increasePrio' + + data = {'hashes': result.hash} + return self._request(method='post', data=data) + + def _set_torrent_pause(self, result): + + self.url = self.host+'command/resume' + if sickbeard.TORRENT_PAUSED: + self.url = self.host+'command/pause' + + data = {'hash': result.hash} + return self._request(method='post', data=data) + +api = qbittorrentAPI() \ No newline at end of file diff --git a/sickbeard/config.py b/sickbeard/config.py index 94413e32fcfe7ced4e738f164806859fd4961723..9a25153026e0aa6043c53cf136c9640ac9d9d325 100644 --- a/sickbeard/config.py +++ b/sickbeard/config.py @@ -230,7 +230,7 @@ def change_DOWNLOAD_PROPERS(download_propers): else: sickbeard.properFinderScheduler.enable = False sickbeard.traktCheckerScheduler.silent = True - logger.log(u"Waiting for the PROPERFINDER thread to exit", logger.INFO) + logger.log(u"Stopping PROPERFINDER thread", logger.INFO) def change_USE_TRAKT(use_trakt): use_trakt = checkbox_to_value(use_trakt) diff --git a/sickbeard/encodingKludge.py b/sickbeard/encodingKludge.py index ffef1d2b348d73e7b5f1ad007567646bce6a9838..fadb28d0ff178d05a0f080d2936eeab2f9fc4a7c 100644 --- a/sickbeard/encodingKludge.py +++ b/sickbeard/encodingKludge.py @@ -20,6 +20,21 @@ import os import chardet import sickbeard +def fixStupidEncodings(x, silent=False): + if type(x) == str: + try: + return x.decode(sickbeard.SYS_ENCODING) + except UnicodeDecodeError: + logger.log(u"Unable to decode value: " + repr(x), logger.ERROR) + return None + elif type(x) == unicode: + return x + else: + logger.log( + u"Unknown value passed in, ignoring it: " + str(type(x)) + " (" + repr(x) + ":" + repr(type(x)) + ")", + logger.DEBUG if silent else logger.ERROR) + return None + def _toUnicode(x): try: if not isinstance(x, unicode): diff --git a/sickbeard/helpers.py b/sickbeard/helpers.py index e3451266735199e79a47db046c9c5364637c2908..1d0d041d9e4fe6701834ba2b11dcc5b52de07fae 100644 --- a/sickbeard/helpers.py +++ b/sickbeard/helpers.py @@ -1037,6 +1037,9 @@ def validateShow(show, season=None, episode=None): if indexer_lang and not indexer_lang == sickbeard.INDEXER_DEFAULT_LANGUAGE: lINDEXER_API_PARMS['language'] = indexer_lang + if show.dvdorder != 0: + lINDEXER_API_PARMS['dvdorder'] = True + t = sickbeard.indexerApi(show.indexer).indexer(**lINDEXER_API_PARMS) if season is None and episode is None: return t @@ -1247,6 +1250,16 @@ def _getTempDir(): return os.path.join(tempfile.gettempdir(), "sickrage-%s" % (uid)) +def codeDescription(status_code): + """ + Returns the description of the URL error code + """ + if status_code in clients.http_error_code: + return clients.http_error_code[status_code] + else: + logger.log(u"Unknown error code. Please submit an issue", logger.WARNING) + return 'unknown' + def getURL(url, post_data=None, params=None, headers={}, timeout=30, session=None, json=False, proxyGlypeProxySSLwarning=None): """ Returns a byte-string retrieved from the url provider. @@ -1283,7 +1296,7 @@ def getURL(url, post_data=None, params=None, headers={}, timeout=30, session=Non if not resp.ok: logger.log(u"Requested url " + url + " returned status code is " + str( - resp.status_code) + ': ' + clients.http_error_code[resp.status_code], logger.DEBUG) + resp.status_code) + ': ' + codeDescription(resp.status_code), logger.DEBUG) return if proxyGlypeProxySSLwarning is not None: @@ -1292,7 +1305,7 @@ def getURL(url, post_data=None, params=None, headers={}, timeout=30, session=Non if not resp.ok: logger.log(u"GlypeProxySSLwarning: Requested url " + url + " returned status code is " + str( - resp.status_code) + ': ' + clients.http_error_code[resp.status_code], logger.DEBUG) + resp.status_code) + ': ' + codeDescription(resp.status_code), logger.DEBUG) return except requests.exceptions.HTTPError, e: @@ -1334,9 +1347,10 @@ def download_file(url, filename, session=None): try: resp = session.get(url) + if not resp.ok: logger.log(u"Requested url " + url + " returned status code is " + str( - resp.status_code) + ': ' + clients.http_error_code[resp.status_code], logger.DEBUG) + resp.status_code) + ': ' + codeDescription(resp.status_code), logger.DEBUG) return False with open(filename, 'wb') as fp: @@ -1409,7 +1423,11 @@ def get_size(start_path='.'): for dirpath, dirnames, filenames in ek.ek(os.walk, start_path): for f in filenames: fp = ek.ek(os.path.join, dirpath, f) - total_size += ek.ek(os.path.getsize, fp) + try: + total_size += ek.ek(os.path.getsize, fp) + except OSError as e: + logger.log('Unable to get size for file {filePath}. Error msg is: {errorMsg}'.format(filePath=fp, errorMsg=str(e)), logger.ERROR) + logger.log(traceback.format_exc(), logger.DEBUG) return total_size def generateApiKey(): @@ -1537,4 +1555,4 @@ def pretty_time_delta(seconds): elif minutes > 0: return '%s%02dm%02ds' % (sign_string, minutes, seconds) else: - return '%s%02ds' % (sign_string, seconds) \ No newline at end of file + return '%s%02ds' % (sign_string, seconds) diff --git a/sickbeard/metadata/generic.py b/sickbeard/metadata/generic.py index bd14875d003131092c3a2e624490ddc5851aa351..098b770424f5aacceb5e96a38d26e3942fee850e 100644 --- a/sickbeard/metadata/generic.py +++ b/sickbeard/metadata/generic.py @@ -951,11 +951,6 @@ class GenericMetadata(): name = showXML.findtext('title') - try: - indexer = int(showXML.findtext('indexer')) - except: - indexer = None - if showXML.findtext('tvdbid') != None: indexer_id = int(showXML.findtext('tvdbid')) elif showXML.findtext('id') != None: @@ -968,6 +963,18 @@ class GenericMetadata(): logger.log(u"Invalid Indexer ID (" + str(indexer_id) + "), not using metadata file", logger.WARNING) return empty_return + indexer = None + if showXML.findtext('indexer') != None: + indexer = int(showXML.findtext('indexer')) + elif showXML.find('episodeguide/url') != None: + epg_url = showXML.findtext('episodeguide/url').lower() + if str(indexer_id) in epg_url: + if 'thetvdb.com' in epg_url: + indexer = 1 + elif 'tvrage' in epg_url: + indexer = 2 + + except Exception, e: logger.log( u"There was an error parsing your existing metadata file: '" + metadata_path + "' error: " + ex(e), diff --git a/sickbeard/metadata/kodi_12plus.py b/sickbeard/metadata/kodi_12plus.py index 1dba0d1730678ff7d099e39ee961a235ac39b91d..02d5b8735086229133f65fd4d269a9913b309cf2 100644 --- a/sickbeard/metadata/kodi_12plus.py +++ b/sickbeard/metadata/kodi_12plus.py @@ -153,11 +153,8 @@ class KODI_12PlusMetadata(generic.GenericMetadata): episodeguide = etree.SubElement(tv_node, "episodeguide") episodeguideurl = etree.SubElement(episodeguide, "url") - episodeguideurl2 = etree.SubElement(tv_node, "episodeguideurl") if getattr(myShow, 'id', None) is not None: - showurl = sickbeard.indexerApi(show_obj.indexer).config['base_url'] + str(myShow["id"]) + '/all/en.zip' - episodeguideurl.text = showurl - episodeguideurl2.text = showurl + episodeguideurl.text = sickbeard.indexerApi(show_obj.indexer).config['base_url'] + str(myShow["id"]) + '/all/en.zip' mpaa = etree.SubElement(tv_node, "mpaa") if getattr(myShow, 'contentrating', None) is not None: diff --git a/sickbeard/metadata/mede8er.py b/sickbeard/metadata/mede8er.py index 0fe31761a6395f2d933b4807ef5de940d3940778..4db891493c761c5f7d000d188d776f564807e4ab 100644 --- a/sickbeard/metadata/mede8er.py +++ b/sickbeard/metadata/mede8er.py @@ -390,7 +390,7 @@ class Mede8erMetadata(mediabrowser.MediaBrowserMetadata): nfo_file = ek.ek(open, nfo_file_path, 'w') - data.write(nfo_file, encoding="utf-8", xml_declaration=True) + data.write(nfo_file, encoding="UTF-8") nfo_file.close() helpers.chmodAsParent(nfo_file_path) except IOError, e: @@ -435,7 +435,7 @@ class Mede8erMetadata(mediabrowser.MediaBrowserMetadata): nfo_file = ek.ek(open, nfo_file_path, 'w') - data.write(nfo_file, encoding="utf-8", xml_declaration = True) + data.write(nfo_file, encoding="UTF-8") nfo_file.close() helpers.chmodAsParent(nfo_file_path) except IOError, e: diff --git a/sickbeard/notifiers/emailnotify.py b/sickbeard/notifiers/emailnotify.py index 02255ac4da3584884880032ba7ad2f6590516193..9b21d29bd8f2390873a646cbdb36f111105aa814 100644 --- a/sickbeard/notifiers/emailnotify.py +++ b/sickbeard/notifiers/emailnotify.py @@ -190,7 +190,13 @@ class EmailNotifier: def _sendmail(self, host, port, smtp_from, use_tls, user, pwd, to, msg, smtpDebug=False): logger.log('HOST: %s; PORT: %s; FROM: %s, TLS: %s, USER: %s, PWD: %s, TO: %s' % ( host, port, smtp_from, use_tls, user, pwd, to), logger.DEBUG) - srv = smtplib.SMTP(host, int(port)) + try: + srv = smtplib.SMTP(host, int(port)) + except Exception as e: + logger.log(u"Exception generated while sending e-mail: " + str(e), logger.ERROR) + logger.log(traceback.format_exc(), logger.DEBUG) + return False + if smtpDebug: srv.set_debuglevel(1) try: diff --git a/sickbeard/notifiers/kodi.py b/sickbeard/notifiers/kodi.py index 7b4bd3c6e991d6a92ef28be450fea5e815b427e6..f7e3b91689452f78eda1c28309c7026d444c2734 100644 --- a/sickbeard/notifiers/kodi.py +++ b/sickbeard/notifiers/kodi.py @@ -247,8 +247,8 @@ class KODINotifier: logger.log(u"KODI HTTP response: " + result.replace('\n', ''), logger.DEBUG) return result - except (urllib2.URLError, IOError), e: - logger.log(u"Warning: Couldn't contact KODI HTTP at " + ek.ss(url) + " " + ex(e), + except Exception as e: + logger.log(u"Warning: Couldn't contact KODI HTTP at " + ek.ss(url) + " " + str(e), logger.WARNING) return False diff --git a/sickbeard/notifiers/libnotify.py b/sickbeard/notifiers/libnotify.py index 9984bdce0901f84b6197c1f61371ce1237102a10..d4dc7953598ddf0851d7db3b3ecc36f84fc475c1 100644 --- a/sickbeard/notifiers/libnotify.py +++ b/sickbeard/notifiers/libnotify.py @@ -29,10 +29,11 @@ def diagnose(): user-readable message indicating possible issues. ''' try: - import pynotify #@UnusedImport + from gi.repository import Notify #@UnusedImport except ImportError: - return (u"<p>Error: pynotify isn't installed. On Ubuntu/Debian, install the " - u"<a href=\"apt:python-notify\">python-notify</a> package.") + return (u"<p>Error: gir-notify isn't installed. On Ubuntu/Debian, install the " + u"<a href=\"apt:gir1.2-notify-0.7\">gir1.2-notify-0.7</a> or " + u"<a href=\"apt:gir1.0-notify-0.4\">gir1.0-notify-0.4</a> package.") if 'DISPLAY' not in os.environ and 'DBUS_SESSION_BUS_ADDRESS' not in os.environ: return (u"<p>Error: Environment variables DISPLAY and DBUS_SESSION_BUS_ADDRESS " u"aren't set. libnotify will only work when you run SickRage " @@ -58,26 +59,26 @@ def diagnose(): class LibnotifyNotifier: def __init__(self): - self.pynotify = None + self.Notify = None self.gobject = None - def init_pynotify(self): - if self.pynotify is not None: + def init_notify(self): + if self.Notify is not None: return True try: - import pynotify + from gi.repository import Notify except ImportError: - logger.log(u"Unable to import pynotify. libnotify notifications won't work.", logger.ERROR) + logger.log(u"Unable to import Notify from gi.repository. libnotify notifications won't work.", logger.ERROR) return False try: import gobject except ImportError: logger.log(u"Unable to import gobject. We can't catch a GError in display.", logger.ERROR) return False - if not pynotify.init('SickRage'): - logger.log(u"Initialization of pynotify failed. libnotify notifications won't work.", logger.ERROR) + if not Notify.init('SickRage'): + logger.log(u"Initialization of Notify failed. libnotify notifications won't work.", logger.ERROR) return False - self.pynotify = pynotify + self.Notify = Notify self.gobject = gobject return True @@ -105,18 +106,17 @@ class LibnotifyNotifier: def _notify(self, title, message, force=False): if not sickbeard.USE_LIBNOTIFY and not force: return False - if not self.init_pynotify(): + if not self.init_notify(): return False # Can't make this a global constant because PROG_DIR isn't available # when the module is imported. - icon_path = os.path.join(sickbeard.PROG_DIR, "data/images/sickbeard_touch_icon.png") - icon_uri = 'file://' + os.path.abspath(icon_path) + icon_path = os.path.join(sickbeard.PROG_DIR, 'gui', 'slick', 'images', 'ico', 'favicon-120.png') # If the session bus can't be acquired here a bunch of warning messages # will be printed but the call to show() will still return True. # pynotify doesn't seem too keen on error handling. - n = self.pynotify.Notification(title, message, icon_uri) + n = self.Notify.Notification.new(title, message, icon_path) try: return n.show() except self.gobject.GError: diff --git a/sickbeard/notifiers/plex.py b/sickbeard/notifiers/plex.py index 735469a93dd8b386b21bc30050846788f0164ee3..a878801854f85826193b8d74a59ced4060c9d7a7 100644 --- a/sickbeard/notifiers/plex.py +++ b/sickbeard/notifiers/plex.py @@ -19,41 +19,122 @@ import urllib import urllib2 import base64 +import re import sickbeard from sickbeard import logger from sickbeard import common from sickbeard.exceptions import ex -from sickbeard import encodingKludge as ek +from sickbeard.encodingKludge import fixStupidEncodings -from sickbeard.notifiers.kodi import KODINotifier +try: + import xml.etree.cElementTree as etree +except ImportError: + import elementtree.ElementTree as etree -# TODO: switch over to using ElementTree -from xml.dom import minidom +class PLEXNotifier: + + def _send_to_plex(self, command, host, username=None, password=None): + """Handles communication to Plex hosts via HTTP API + + Args: + command: Dictionary of field/data pairs, encoded via urllib and passed to the legacy xbmcCmds HTTP API + host: Plex host:port + username: Plex API username + password: Plex API password + + Returns: + Returns 'OK' for successful commands or False if there was an error + + """ -class PLEXNotifier(KODINotifier): - def _notify_pmc(self, message, title="SickRage", host=None, username=None, password=None, force=False): # fill in omitted parameters - if not host: - if sickbeard.PLEX_HOST: - host = sickbeard.PLEX_HOST # Use the default Plex host - else: - logger.log(u"No Plex host specified, check your settings", logger.DEBUG) - return False if not username: username = sickbeard.PLEX_USERNAME if not password: password = sickbeard.PLEX_PASSWORD + if not host: + logger.log(u'PLEX: No host specified, check your settings', logger.ERROR) + return False + + for key in command: + if type(command[key]) == unicode: + command[key] = command[key].encode('utf-8') + + enc_command = urllib.urlencode(command) + logger.log(u'PLEX: Encoded API command: ' + enc_command, logger.DEBUG) + + url = 'http://%s/xbmcCmds/xbmcHttp/?%s' % (host, enc_command) + try: + req = urllib2.Request(url) + # if we have a password, use authentication + if password: + base64string = base64.encodestring('%s:%s' % (username, password))[:-1] + authheader = 'Basic %s' % base64string + req.add_header('Authorization', authheader) + logger.log(u'PLEX: Contacting (with auth header) via url: ' + url, logger.DEBUG) + else: + logger.log(u'PLEX: Contacting via url: ' + url, logger.DEBUG) + + response = urllib2.urlopen(req) + + result = response.read().decode(sickbeard.SYS_ENCODING) + response.close() + + logger.log(u'PLEX: HTTP response: ' + result.replace('\n', ''), logger.DEBUG) + # could return result response = re.compile('<html><li>(.+\w)</html>').findall(result) + return 'OK' + + except (urllib2.URLError, IOError), e: + logger.log(u'PLEX: Warning: Couldn\'t contact Plex at ' + fixStupidEncodings(url) + ' ' + ex(e), logger.WARNING) + return False + + def _notify_pmc(self, message, title='SickRage', host=None, username=None, password=None, force=False): + """Internal wrapper for the notify_snatch and notify_download functions + + Args: + message: Message body of the notice to send + title: Title of the notice to send + host: Plex Media Client(s) host:port + username: Plex username + password: Plex password + force: Used for the Test method to override config safety checks + + Returns: + Returns a list results in the format of host:ip:result + The result will either be 'OK' or False, this is used to be parsed by the calling function. + + """ + # suppress notifications if the notifier is disabled but the notify options are checked if not sickbeard.USE_PLEX and not force: - logger.log("Notification for Plex not enabled, skipping this notification", logger.DEBUG) return False - return self._notify_kodi(message=message, title=title, host=host, username=username, password=password, - force=True) + # fill in omitted parameters + if not host: + host = sickbeard.PLEX_HOST + if not username: + username = sickbeard.PLEX_USERNAME + if not password: + password = sickbeard.PLEX_PASSWORD + + result = '' + for curHost in [x.strip() for x in host.split(',')]: + logger.log(u'PLEX: Sending notification to \'%s\' - %s' % (curHost, message), logger.DEBUG) + + command = {'command': 'ExecBuiltIn', 'parameter': 'Notification(%s,%s)' % (title.encode('utf-8'), message.encode('utf-8'))} + notify_result = self._send_to_plex(command, curHost, username, password) + if notify_result: + result += '%s:%s' % (curHost, str(notify_result)) + + return result + +############################################################################## +# Public functions +############################################################################## def notify_snatch(self, ep_name): if sickbeard.PLEX_NOTIFY_ONSNATCH: @@ -65,62 +146,128 @@ class PLEXNotifier(KODINotifier): def notify_subtitle_download(self, ep_name, lang): if sickbeard.PLEX_NOTIFY_ONSUBTITLEDOWNLOAD: - self._notify_pmc(ep_name + ": " + lang, common.notifyStrings[common.NOTIFY_SUBTITLE_DOWNLOAD]) + self._notify_pmc(ep_name + ': ' + lang, common.notifyStrings[common.NOTIFY_SUBTITLE_DOWNLOAD]) - def notify_git_update(self, new_version="??"): + def notify_git_update(self, new_version='??'): if sickbeard.USE_PLEX: update_text = common.notifyStrings[common.NOTIFY_GIT_UPDATE_TEXT] title = common.notifyStrings[common.NOTIFY_GIT_UPDATE] self._notify_pmc(update_text + new_version, title) - def test_notify(self, host, username, password): - return self._notify_pmc("Testing Plex notifications from SickRage", "Test Notification", host, username, - password, force=True) + def test_notify_pmc(self, host, username, password): + return self._notify_pmc('This is a test notification from SickRage', 'Test Notification', host, username, password, force=True) + + def test_notify_pms(self, host, username, password, plex_server_token): + return self.update_library(host=host, username=username, password=password, plex_server_token=plex_server_token, force=False) - def update_library(self, showName=None): + def update_library(self, ep_obj=None, host=None, username=None, password=None, plex_server_token=None, force=True): """Handles updating the Plex Media Server host via HTTP API Plex Media Server currently only supports updating the whole video library and not a specific path. Returns: - Returns True or False + Returns None for no issue, else a string of host with connection issues """ if sickbeard.USE_PLEX and sickbeard.PLEX_UPDATE_LIBRARY: + if not sickbeard.PLEX_SERVER_HOST: - logger.log(u"No Plex Media Server host specified, check your settings", logger.DEBUG) + logger.log(u'PLEX: No Plex Media Server host specified, check your settings', logger.DEBUG) return False - logger.log(u"Updating library for the Plex Media Server host: " + sickbeard.PLEX_SERVER_HOST, - logger.INFO) + if not host: + host = sickbeard.PLEX_SERVER_HOST + if not username: + username = sickbeard.PLEX_USERNAME + if not password: + password = sickbeard.PLEX_PASSWORD + + if not plex_server_token: + token = sickbeard.PLEX_SERVER_TOKEN + + # if username and password were provided, fetch the auth token from plex.tv + token_arg = '' + if plex_server_token: + token_arg = '?X-Plex-Token=' + sickbeard.PLEX_SERVER_TOKEN + elif username and password: + logger.log(u'PLEX: fetching plex.tv credentials for user: ' + username, logger.DEBUG) + req = urllib2.Request('https://plex.tv/users/sign_in.xml', data='') + authheader = 'Basic %s' % base64.encodestring('%s:%s' % (username, password))[:-1] + req.add_header('Authorization', authheader) + req.add_header('X-Plex-Device-Name', 'SickRage') + req.add_header('X-Plex-Product', 'SickRage Notifier') + req.add_header('X-Plex-Client-Identifier', sickbeard.CUR_COMMIT_HASH) + req.add_header('X-Plex-Version', '1.0') - url = "http://%s/library/sections" % sickbeard.PLEX_SERVER_HOST - if sickbeard.PLEX_SERVER_TOKEN: - url += "/?X-Plex-Token=" + sickbeard.PLEX_SERVER_TOKEN - try: - xml_sections = minidom.parse(urllib.urlopen(url)) - except IOError, e: - logger.log(u"Error while trying to contact Plex Media Server: " + ex(e), logger.ERROR) - return False + try: + response = urllib2.urlopen(req) + auth_tree = etree.parse(response) + token = auth_tree.findall('.//authentication-token')[0].text + token_arg = '?X-Plex-Token=' + token - sections = xml_sections.getElementsByTagName('Directory') - if not sections: - logger.log(u"Plex Media Server not running on: " + sickbeard.PLEX_SERVER_HOST, logger.INFO) - return False + except urllib2.URLError as e: + logger.log(u'PLEX: Error fetching credentials from from plex.tv for user %s: %s' % (username, ex(e)), logger.DEBUG) + + except (ValueError, IndexError) as e: + logger.log(u'PLEX: Error parsing plex.tv response: ' + ex(e), logger.DEBUG) + + file_location = '' if None is ep_obj else ep_obj.location + host_list = [x.strip() for x in host.split(',')] + hosts_all = {} + hosts_match = {} + hosts_failed = [] + for cur_host in host_list: + + url = 'http://%s/library/sections%s' % (cur_host, token_arg) + try: + xml_tree = etree.parse(urllib.urlopen(url)) + media_container = xml_tree.getroot() + except IOError, e: + logger.log(u'PLEX: Error while trying to contact Plex Media Server: ' + ex(e), logger.ERROR) + hosts_failed.append(cur_host) + continue + + sections = media_container.findall('.//Directory') + if not sections: + logger.log(u'PLEX: Plex Media Server not running on: ' + cur_host, logger.DEBUG) + hosts_failed.append(cur_host) + continue - for s in sections: - if s.getAttribute('type') == "show": - url = "http://%s/library/sections/%s/refresh" % (sickbeard.PLEX_SERVER_HOST, s.getAttribute('key')) - if sickbeard.PLEX_SERVER_TOKEN: - url += "/?X-Plex-Token=" + sickbeard.PLEX_SERVER_TOKEN - try: - urllib.urlopen(url) - except Exception, e: - logger.log(u"Error updating library section for Plex Media Server: " + ex(e), logger.ERROR) - return False + for section in sections: + if 'show' == section.attrib['type']: - return True + keyed_host = [(str(section.attrib['key']), cur_host)] + hosts_all.update(keyed_host) + if not file_location: + continue + + for section_location in section.findall('.//Location'): + section_path = re.sub(r'[/\\]+', '/', section_location.attrib['path'].lower()) + section_path = re.sub(r'^(.{,2})[/\\]', '', section_path) + location_path = re.sub(r'[/\\]+', '/', file_location.lower()) + location_path = re.sub(r'^(.{,2})[/\\]', '', location_path) + + if section_path in location_path: + hosts_match.update(keyed_host) + + hosts_try = (hosts_all.copy(), hosts_match.copy())[len(hosts_match)] + host_list = [] + for section_key, cur_host in hosts_try.items(): + + url = 'http://%s/library/sections/%s/refresh%s' % (cur_host, section_key, token_arg) + try: + force and urllib.urlopen(url) + host_list.append(cur_host) + except Exception, e: + logger.log(u'PLEX: Error updating library section for Plex Media Server: ' + ex(e), logger.ERROR) + hosts_failed.append(cur_host) + + if len(hosts_match): + logger.log(u'PLEX: Updating hosts where TV section paths match the downloaded show: ' + ', '.join(set(host_list)), logger.DEBUG) + else: + logger.log(u'PLEX: Updating all hosts with TV sections: ' + ', '.join(set(host_list)), logger.DEBUG) + return (', '.join(set(hosts_failed)), None)[not len(hosts_failed)] notifier = PLEXNotifier diff --git a/sickbeard/postProcessor.py b/sickbeard/postProcessor.py index 173c47a2a7670793288b17c61055e5c083f3f16c..eede71fff326a0564381b4863b39490c26215555 100644 --- a/sickbeard/postProcessor.py +++ b/sickbeard/postProcessor.py @@ -183,10 +183,19 @@ class PostProcessor(object): # don't confuse glob with chars we didn't mean to use base_name = re.sub(r'[\[\]\*\?]', r'[\g<0>]', base_name) - if subfolders: - filelist = ek.ek(recursive_glob, ek.ek(os.path.dirname, file_path), base_name + '*') - else: - filelist = ek.ek(glob.glob, base_name + '*') + if subfolders: # subfolders are only checked in show folder, so names will always be exactly alike + filelist = ek.ek(recursive_glob, ek.ek(os.path.dirname, file_path), base_name + '*') # just create the list of all files starting with the basename + else: # this is called when PP, so we need to do the filename check case-insensitive + filelist = [] + checklist = ek.ek(glob.glob, ek.ek(os.path.join, ek.ek(os.path.dirname, file_path), '*')) # get a list of all the files in the folder + for filefound in checklist: # loop through all the files in the folder, and check if they are the same name even when the cases don't match + file_name = filefound.rpartition('.')[0] + if not base_name_only: + file_name = file_name + '.' + if file_name.lower() == base_name.lower(): # if there's no difference in the filename add it to the filelist + filelist.append(filefound) + + for associated_file_path in filelist: # only add associated to list if associated_file_path == file_path: @@ -201,7 +210,12 @@ class PostProcessor(object): if ek.ek(os.path.isfile, associated_file_path): file_path_list.append(associated_file_path) - + + if file_path_list: + self._log(u"Found the following associated files: " + str(file_path_list), logger.DEBUG) + else: + self._log(u"No associated files were during this pass", logger.DEBUG) + return file_path_list def _delete(self, file_path, associated_files=False): @@ -841,7 +855,7 @@ class PostProcessor(object): old_ep_status, old_ep_quality = common.Quality.splitCompositeStatus(ep_obj.status) # get the quality of the episode we're processing - if quality: + if quality and not common.Quality.qualityStrings[quality] == 'Unknown': self._log(u"Snatch history had a quality in it, using that: " + common.Quality.qualityStrings[quality], logger.DEBUG) new_ep_quality = quality diff --git a/sickbeard/providers/__init__.py b/sickbeard/providers/__init__.py index 02f150c0bf4b584a01ea64ad2913fe6384c33b96..453a34eb3b0836e4791abb12158d961789ea2ca1 100755 --- a/sickbeard/providers/__init__.py +++ b/sickbeard/providers/__init__.py @@ -47,6 +47,7 @@ __all__ = ['ezrss', 'tntvillage', 'binsearch', 'eztv', + 'scenetime', ] import sickbeard diff --git a/sickbeard/providers/eztv.py b/sickbeard/providers/eztv.py index 909725c5daca12bfabd39607f701cbf221df3180..22a32f4eb3376fb9c8235b2ebcc34ec7424fcddc 100644 --- a/sickbeard/providers/eztv.py +++ b/sickbeard/providers/eztv.py @@ -16,7 +16,6 @@ # # You should have received a copy of the GNU General Public License # along with Sick Beard. If not, see <http://www.gnu.org/licenses/>. - import traceback import re, datetime @@ -26,6 +25,7 @@ from sickbeard import classes from sickbeard import helpers from sickbeard import logger, tvcache, db from sickbeard.common import Quality +from sickbeard.bs4_parser import BS4Parser class EZTVProvider(generic.TorrentProvider): @@ -39,8 +39,9 @@ class EZTVProvider(generic.TorrentProvider): self.cache = EZTVCache(self) self.urls = { - 'base_url': 'http://eztvapi.re/', - 'show': 'http://eztvapi.re/show/%s', + 'base_url': 'https://eztv.ch/', + 'rss': 'https://eztv.ch/', + 'episode': 'http://eztvapi.re/show/%s', } self.url = self.urls['base_url'] @@ -68,12 +69,15 @@ class EZTVProvider(generic.TorrentProvider): return [search_string] def getQuality(self, item, anime=False): - if item.get('quality') == "480p": - return Quality.SDTV - elif item.get('quality') == "720p": - return Quality.HDWEBDL - elif item.get('quality') == "1080p": - return Quality.FULLHDWEBDL + if 'quality' in item: + if item.get('quality') == "480p": + return Quality.SDTV + elif item.get('quality') == "720p": + return Quality.HDWEBDL + elif item.get('quality') == "1080p": + return Quality.FULLHDWEBDL + else: + return Quality.sceneQuality(item.get('title'), anime) else: return Quality.sceneQuality(item.get('title'), anime) @@ -84,49 +88,83 @@ class EZTVProvider(generic.TorrentProvider): for mode in search_params.keys(): - if mode != 'Episode': - logger.log(u"" + self.name + " does not accept " + mode + " mode", logger.DEBUG) - return results + if mode == 'RSS': + for search_string in search_params[mode]: + searchURL = self.urls['rss'] + logger.log(u"" + self.name + " search page URL: " + searchURL, logger.DEBUG) - for search_string in search_params[mode]: + HTML = self.getURL(searchURL) + if not HTML: + logger.log(u"" + self.name + " could not retrieve page URL:" + searchURL, logger.DEBUG) + return results - searchURL = self.urls['show'] % (search_string['imdb_id']) - logger.log(u"" + self.name + " search page URL: " + searchURL, logger.DEBUG) + try: + with BS4Parser(HTML, features=["html5lib", "permissive"]) as parsedHTML: + resultsTable = parsedHTML.find_all('tr', attrs={'name': 'hover', 'class': 'header_brd'}) - try: - parsedJSON = self.getURL(searchURL, json=True) - except ValueError as e: - parsedJSON = None + if not resultsTable: + logger.log(u"The Data returned from " + self.name + " do not contains any torrent", + logger.DEBUG) + continue - if not parsedJSON: - logger.log(u"" + self.name + " could not retrieve page URL:" + searchURL, logger.DEBUG) - return results + for entries in resultsTable: + title = entries.find('a', attrs={'class': 'epinfo'}).contents[0] + link = entries.find('a', attrs={'class': 'magnet'}).get('href') - try: - for episode in parsedJSON['episodes']: - if int(episode.get('season')) == search_string.get('season') and \ - int(episode.get('episode')) == search_string.get('episode'): + item = { + 'title': title, + 'link': link, + } - for quality in episode['torrents'].keys(): - link = episode['torrents'][quality]['url'] - title = re.search('&dn=(.*?)&', link).group(1) + items[mode].append(item) - item = { - 'title': title, - 'link': link, - 'quality': quality - } + except Exception, e: + logger.log(u"Failed parsing " + self.name + " Traceback: " + traceback.format_exc(), + logger.ERROR) - # re.search in case of PROPER|REPACK. In other cases - # add_string is empty, so condition is met. - if re.search(search_string.get('add_string'), title): - items[mode].append(item) + elif mode == 'Episode': + for search_string in search_params[mode]: + searchURL = self.urls['episode'] % (search_string['imdb_id']) + logger.log(u"" + self.name + " search page URL: " + searchURL, logger.DEBUG) + + try: + parsedJSON = self.getURL(searchURL, json=True) + except ValueError as e: + parsedJSON = None + + if not parsedJSON: + logger.log(u"" + self.name + " could not retrieve page URL:" + searchURL, logger.DEBUG) + return results - break + try: + for episode in parsedJSON['episodes']: + if int(episode.get('season')) == search_string.get('season') and \ + int(episode.get('episode')) == search_string.get('episode'): - except Exception, e: - logger.log(u"Failed parsing " + self.name + " Traceback: " + traceback.format_exc(), - logger.ERROR) + for quality in episode['torrents'].keys(): + link = episode['torrents'][quality]['url'] + title = re.search('&dn=(.*?)&', link).group(1) + + item = { + 'title': title, + 'link': link, + 'quality': quality + } + + # re.search in case of PROPER|REPACK. In other cases + # add_string is empty, so condition is met. + if 'add_string' in search_string and re.search(search_string.get('add_string'), title): + items[mode].append(item) + + break + + except Exception, e: + logger.log(u"Failed parsing " + self.name + " Traceback: " + traceback.format_exc(), + logger.ERROR) + + else: + logger.log(u"" + self.name + " does not accept " + mode + " mode", logger.DEBUG) + return results results += items[mode] diff --git a/sickbeard/providers/iptorrents.py b/sickbeard/providers/iptorrents.py index 1af7e8ca8008e926371ea5fde8efaffc9990c2c6..28a8f3c1038d54d0de6a39f96a20a62738811b53 100644 --- a/sickbeard/providers/iptorrents.py +++ b/sickbeard/providers/iptorrents.py @@ -24,6 +24,7 @@ import itertools import sickbeard import generic +from sickbeard.common import MULTI_EP_RESULT, SEASON_RESULT from sickbeard.common import Quality from sickbeard import logger from sickbeard import tvcache @@ -59,12 +60,12 @@ class IPTorrentsProvider(generic.TorrentProvider): self.urls = {'base_url': 'https://iptorrents.eu', 'login': 'https://iptorrents.eu/torrents/', - 'search': 'https://iptorrents.eu/torrents/?%s%s&q=%s&qf=ti', + 'search': 'https://iptorrents.eu/t?%s%s&q=%s&qf=#torrents', } self.url = self.urls['base_url'] - self.categorie = 'l73=1&l78=1&l66=1&l65=1&l79=1&l5=1&l4=1' + self.categorie = 'l73=' def isEnabled(self): return self.enabled diff --git a/sickbeard/providers/rarbg.py b/sickbeard/providers/rarbg.py index ad8e63d6a7e7302c4b8b8d859b82f48ded7c395f..c56a3ccc672cdc691632c35a66fd7bb5312fc01c 100644 --- a/sickbeard/providers/rarbg.py +++ b/sickbeard/providers/rarbg.py @@ -30,7 +30,7 @@ from lib import requests from lib.requests import exceptions import sickbeard -from sickbeard.common import Quality +from sickbeard.common import Quality, USER_AGENT from sickbeard import logger from sickbeard import tvcache from sickbeard import show_name_helpers @@ -62,13 +62,13 @@ class RarbgProvider(generic.TorrentProvider): self.token = None self.tokenExpireDate = None - self.urls = {'url': 'https://rarbg.com', - 'token': 'https://torrentapi.org/pubapi.php?get_token=get_token&format=json', - 'listing': 'https://torrentapi.org/pubapi.php?mode=list', - 'search': 'https://torrentapi.org/pubapi.php?mode=search&search_string={search_string}', - 'search_tvdb': 'https://torrentapi.org/pubapi.php?mode=search&search_tvdb={tvdb}&search_string={search_string}', - 'search_tvrage': 'https://torrentapi.org/pubapi.php?mode=search&search_tvrage={tvrage}&search_string={search_string}', - 'api_spec': 'https://rarbg.com/pubapi/apidocs.txt', + self.urls = {'url': u'https://rarbg.com', + 'token': u'https://torrentapi.org/pubapi.php?get_token=get_token&format=json', + 'listing': u'https://torrentapi.org/pubapi.php?mode=list', + 'search': u'https://torrentapi.org/pubapi.php?mode=search&search_string={search_string}', + 'search_tvdb': u'https://torrentapi.org/pubapi.php?mode=search&search_tvdb={tvdb}&search_string={search_string}', + 'search_tvrage': u'https://torrentapi.org/pubapi.php?mode=search&search_tvrage={tvrage}&search_string={search_string}', + 'api_spec': u'https://rarbg.com/pubapi/apidocs.txt', } self.url = self.urls['listing'] @@ -92,6 +92,8 @@ class RarbgProvider(generic.TorrentProvider): self.next_request = datetime.datetime.now() self.cache = RarbgCache(self) + + self.headers = {'User-Agent': USER_AGENT} def isEnabled(self): return self.enabled @@ -107,7 +109,7 @@ class RarbgProvider(generic.TorrentProvider): resp_json = None try: - response = self.session.get(self.urls['token'], timeout=30, verify=False) + response = self.session.get(self.urls['token'], timeout=30, verify=False, headers=self.headers) response.raise_for_status() resp_json = response.json() except RequestException as e: @@ -204,18 +206,18 @@ class RarbgProvider(generic.TorrentProvider): searchURL = self.urls['listing'] + self.defaultOptions elif mode == 'Season': if ep_indexer == INDEXER_TVDB: - searchURL = self.urls['search_tvdb'].format(search_string=urllib.quote(search_string), tvdb=ep_indexerid) + self.defaultOptions + searchURL = self.urls['search_tvdb'].format(search_string=search_string, tvdb=ep_indexerid) + self.defaultOptions elif ep_indexer == INDEXER_TVRAGE: - searchURL = self.urls['search_tvrage'].format(search_string=urllib.quote(search_string), tvrage=ep_indexerid) + self.defaultOptions + searchURL = self.urls['search_tvrage'].format(search_string=search_string, tvrage=ep_indexerid) + self.defaultOptions else: - searchURL = self.urls['search'].format(search_string=urllib.quote(search_string)) + self.defaultOptions + searchURL = self.urls['search'].format(search_string=search_string) + self.defaultOptions elif mode == 'Episode': if ep_indexer == INDEXER_TVDB: - searchURL = self.urls['search_tvdb'].format(search_string=urllib.quote(search_string), tvdb=ep_indexerid) + self.defaultOptions + searchURL = self.urls['search_tvdb'].format(search_string=search_string, tvdb=ep_indexerid) + self.defaultOptions elif ep_indexer == INDEXER_TVRAGE: - searchURL = self.urls['search_tvrage'].format(search_string=urllib.quote(search_string), tvrage=ep_indexerid) + self.defaultOptions + searchURL = self.urls['search_tvrage'].format(search_string=search_string, tvrage=ep_indexerid) + self.defaultOptions else: - searchURL = self.urls['search'].format(search_string=urllib.quote(search_string)) + self.defaultOptions + searchURL = self.urls['search'].format(search_string=search_string) + self.defaultOptions else: logger.log(u'{name} invalid search mode:{mode}'.format(name=self.name, mode=mode), logger.ERROR) diff --git a/sickbeard/providers/scenetime.py b/sickbeard/providers/scenetime.py new file mode 100644 index 0000000000000000000000000000000000000000..c3aaca9d341022e5f457e145904e382f3f3f0970 --- /dev/null +++ b/sickbeard/providers/scenetime.py @@ -0,0 +1,285 @@ +# Author: Idan Gutman +# URL: http://code.google.com/p/sickbeard/ +# +# This file is part of SickRage. +# +# SickRage is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# SickRage is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with SickRage. If not, see <http://www.gnu.org/licenses/>. + +import re +import traceback +import datetime +import urlparse +import sickbeard +import generic +import urllib +from sickbeard.common import Quality +from sickbeard import logger +from sickbeard import tvcache +from sickbeard import db +from sickbeard import classes +from sickbeard import helpers +from sickbeard import show_name_helpers +from sickbeard.exceptions import ex +from sickbeard import clients +from lib import requests +from lib.requests import exceptions +from sickbeard.bs4_parser import BS4Parser +from lib.unidecode import unidecode +from sickbeard.helpers import sanitizeSceneName + + +class SceneTimeProvider(generic.TorrentProvider): + + def __init__(self): + + generic.TorrentProvider.__init__(self, "SceneTime") + + self.supportsBacklog = True + + self.enabled = False + self.username = None + self.password = None + self.ratio = None + self.minseed = None + self.minleech = None + + self.cache = SceneTimeCache(self) + + self.urls = {'base_url': 'https://www.scenetime.com', + 'login': 'https://www.scenetime.com/takelogin.php', + 'detail': 'https://www.scenetime.com/details.php?id=%s', + 'search': 'https://www.scenetime.com/browse.php?search=%s%s', + 'download': 'https://www.scenetime.com/download.php/%s/%s', + } + + self.url = self.urls['base_url'] + + self.categories = "&c2=1&c43=13&c9=1&c63=1&c77=1&c79=1&c100=1&c101=1" + + def isEnabled(self): + return self.enabled + + def imageName(self): + return 'scenetime.png' + + def getQuality(self, item, anime=False): + + quality = Quality.sceneQuality(item[0], anime) + return quality + + def _doLogin(self): + + login_params = {'username': self.username, + 'password': self.password + } + + self.session = requests.Session() + + try: + response = self.session.post(self.urls['login'], data=login_params, timeout=30, verify=False) + except (requests.exceptions.ConnectionError, requests.exceptions.HTTPError), e: + logger.log(u'Unable to connect to ' + self.name + ' provider: ' + ex(e), logger.ERROR) + return False + + if re.search('Username or password incorrect', response.text): + logger.log(u'Invalid username or password for ' + self.name + ' Check your settings', logger.ERROR) + return False + + return True + + def _get_season_search_strings(self, ep_obj): + + search_string = {'Season': []} + for show_name in set(show_name_helpers.allPossibleShowNames(self.show)): + if ep_obj.show.air_by_date or ep_obj.show.sports: + ep_string = show_name + '.' + str(ep_obj.airdate).split('-')[0] + elif ep_obj.show.anime: + ep_string = show_name + '.' + "%d" % ep_obj.scene_absolute_number + else: + ep_string = show_name + '.S%02d' % int(ep_obj.scene_season) #1) showName SXX + + search_string['Season'].append(ep_string) + + return [search_string] + + def _get_episode_search_strings(self, ep_obj, add_string=''): + + search_string = {'Episode': []} + + if not ep_obj: + return [] + + if self.show.air_by_date: + for show_name in set(show_name_helpers.allPossibleShowNames(self.show)): + ep_string = sanitizeSceneName(show_name) + ' ' + \ + str(ep_obj.airdate).replace('-', '|') + search_string['Episode'].append(ep_string) + elif self.show.sports: + for show_name in set(show_name_helpers.allPossibleShowNames(self.show)): + ep_string = sanitizeSceneName(show_name) + ' ' + \ + str(ep_obj.airdate).replace('-', '|') + '|' + \ + ep_obj.airdate.strftime('%b') + search_string['Episode'].append(ep_string) + elif self.show.anime: + for show_name in set(show_name_helpers.allPossibleShowNames(self.show)): + ep_string = sanitizeSceneName(show_name) + ' ' + \ + "%i" % int(ep_obj.scene_absolute_number) + search_string['Episode'].append(ep_string) + else: + for show_name in set(show_name_helpers.allPossibleShowNames(self.show)): + ep_string = show_name_helpers.sanitizeSceneName(show_name) + ' ' + \ + sickbeard.config.naming_ep_type[2] % {'seasonnumber': ep_obj.scene_season, + 'episodenumber': ep_obj.scene_episode} + ' %s' % add_string + + search_string['Episode'].append(re.sub('\s+', ' ', ep_string)) + + return [search_string] + + def _doSearch(self, search_params, search_mode='eponly', epcount=0, age=0, epObj=None): + + results = [] + items = {'Season': [], 'Episode': [], 'RSS': []} + + if not self._doLogin(): + return results + + for mode in search_params.keys(): + for search_string in search_params[mode]: + + if isinstance(search_string, unicode): + search_string = unidecode(search_string) + + searchURL = self.urls['search'] % (urllib.quote(search_string), self.categories) + + logger.log(u"Search string: " + searchURL, logger.DEBUG) + + data = self.getURL(searchURL) + if not data: + continue + + try: + with BS4Parser(data, features=["html5lib", "permissive"]) as html: + torrent_table = html.select("#torrenttable table"); + torrent_rows = torrent_table[0].select("tr") if torrent_table else [] + + #Continue only if one Release is found + if len(torrent_rows) < 2: + logger.log(u"The Data returned from %s does not contain any torrent links" % self.name, + logger.DEBUG) + continue + + for result in torrent_rows[1:]: + cells = result.find_all('td') + + link = cells[1].find('a'); + + full_id = link['href'].replace('details.php?id=', '') + torrent_id = full_id.split("&")[0] + + try: + title = link.contents[0].get_text() + + filename = "%s.torrent" % title.replace(" ", ".") + + download_url = self.urls['download'] % (torrent_id, filename) + + id = int(torrent_id) + seeders = int(cells[6].get_text()) + leechers = int(cells[7].get_text()) + except (AttributeError, TypeError): + continue + + #Filter unseeded torrent + if mode != 'RSS' and (seeders < self.minseed or leechers < self.minleech): + continue + + if not title or not download_url: + continue + + item = title, download_url, id, seeders, leechers + logger.log(u"Found result: " + title.replace(' ','.') + " (" + searchURL + ")", logger.DEBUG) + + items[mode].append(item) + + except Exception, e: + logger.log(u"Failed parsing " + self.name + " Traceback: " + traceback.format_exc(), logger.ERROR) + + #For each search mode sort all the items by seeders + items[mode].sort(key=lambda tup: tup[3], reverse=True) + + results += items[mode] + + return results + + def _get_title_and_url(self, item): + + title, url, id, seeders, leechers = item + + if title: + title = u'' + title + title = title.replace(' ', '.') + title = self._clean_title_from_provider(title) + + if url: + url = str(url).replace('&', '&') + + return (title, url) + + def findPropers(self, search_date=datetime.datetime.today()): + + results = [] + + myDB = db.DBConnection() + sqlResults = myDB.select( + 'SELECT s.show_name, e.showid, e.season, e.episode, e.status, e.airdate FROM tv_episodes AS e' + + ' INNER JOIN tv_shows AS s ON (e.showid = s.indexer_id)' + + ' WHERE e.airdate >= ' + str(search_date.toordinal()) + + ' AND (e.status IN (' + ','.join([str(x) for x in Quality.DOWNLOADED]) + ')' + + ' OR (e.status IN (' + ','.join([str(x) for x in Quality.SNATCHED]) + ')))' + ) + + if not sqlResults: + return [] + + for sqlshow in sqlResults: + self.show = helpers.findCertainShow(sickbeard.showList, int(sqlshow["showid"])) + if self.show: + curEp = self.show.getEpisode(int(sqlshow["season"]), int(sqlshow["episode"])) + + searchString = self._get_episode_search_strings(curEp, add_string='PROPER|REPACK') + + for item in self._doSearch(searchString[0]): + title, url = self._get_title_and_url(item) + results.append(classes.Proper(title, url, datetime.datetime.today(), self.show)) + + return results + + def seedRatio(self): + return self.ratio + + +class SceneTimeCache(tvcache.TVCache): + def __init__(self, provider): + + tvcache.TVCache.__init__(self, provider) + + # only poll SceneTime every 20 minutes max + self.minTime = 20 + + def _getRSSData(self): + search_params = {'RSS': ['']} + return {'entries': self.provider._doSearch(search_params)} + + +provider = SceneTimeProvider() diff --git a/sickbeard/showUpdater.py b/sickbeard/showUpdater.py index a5dd91550afc7e3e46dcbc20acd76b68c0813ea4..6ceeefcd69bf3ab467dd4bdb32aafc1305b5e348 100644 --- a/sickbeard/showUpdater.py +++ b/sickbeard/showUpdater.py @@ -78,16 +78,14 @@ class ShowUpdater(): # if should_update returns True (not 'Ended') or show is selected stale 'Ended' then update, otherwise just refresh if curShow.should_update(update_date=update_date) or curShow.indexerid in stale_should_update: try: - curQueueItem = sickbeard.showQueueScheduler.action.updateShow(curShow, True) # @UndefinedVariable + piList.append(sickbeard.showQueueScheduler.action.updateShow(curShow, True)) # @UndefinedVariable except exceptions.CantUpdateException as e: logger.log("Unable to update show: {0}".format(str(e)),logger.DEBUG) else: logger.log( u"Not updating episodes for show " + curShow.name + " because it's marked as ended and last/next episode is not within the grace period.", logger.DEBUG) - curQueueItem = sickbeard.showQueueScheduler.action.refreshShow(curShow, True) # @UndefinedVariable - - piList.append(curQueueItem) + piList.append(sickbeard.showQueueScheduler.action.refreshShow(curShow, True)) # @UndefinedVariable except (exceptions.CantUpdateException, exceptions.CantRefreshException), e: logger.log(u"Automatic update failed: " + ex(e), logger.ERROR) diff --git a/sickbeard/tv.py b/sickbeard/tv.py index 512893fadf542c5852713d3cee4850e229b7e837..128a2eba4f8d6166a18107e544b71d6ee617784d 100644 --- a/sickbeard/tv.py +++ b/sickbeard/tv.py @@ -1437,7 +1437,8 @@ class TVEpisode(object): for subtitle in subtitles.get(video): added_subtitles.append(subtitle.language.alpha2) helpers.chmodAsParent(subtitle.path) - + except ServiceError as e: + logger.log("Service is unavailable: {0}".format(str(e)), logger.INFO) except Exception as e: logger.log("Error occurred when downloading subtitles: " + str(e), logger.ERROR) return diff --git a/sickbeard/tvcache.py b/sickbeard/tvcache.py index 4d642472ac861aea7fe68983df916331b43dc3b3..3e2804cba01b83d6181ddcdda6d9b1a3af9bd3a9 100644 --- a/sickbeard/tvcache.py +++ b/sickbeard/tvcache.py @@ -299,7 +299,7 @@ class TVCache(): def searchCache(self, episode, manualSearch=False, downCurQuality=False): neededEps = self.findNeededEpisodes(episode, manualSearch, downCurQuality) - return neededEps[episode] if len(neededEps) > 0 else [] + return neededEps[episode] if episode in neededEps else [] def listPropers(self, date=None, delimiter="."): myDB = self._getDB() diff --git a/sickbeard/webserve.py b/sickbeard/webserve.py index d373be2228159b237f5f089f881168b13c157556..fc8bfcbd6e3139723669b6623655d828a33ec094 100644 --- a/sickbeard/webserve.py +++ b/sickbeard/webserve.py @@ -287,6 +287,9 @@ class LoginHandler(BaseHandler): if api_key: remember_me = int(self.get_argument('remember_me', default=0) or 0) self.set_secure_cookie('sickrage_user', api_key, expires_days=30 if remember_me > 0 else None) + logger.log('User logged into the SickRage web interface from IP: ' + self.request.remote_ip, logger.INFO) + else: + logger.log('User attempted a failed login to the SickRage web interface from IP: ' + self.request.remote_ip, logger.WARNING) self.redirect('/home/') @@ -891,21 +894,44 @@ class Home(WebRoot): return finalResult - def testPLEX(self, host=None, username=None, password=None): - # self.set_header('Cache-Control', 'max-age=0,no-cache,no-store') + def testPMC(self, host=None, username=None, password=None): + self.set_header('Cache-Control', 'max-age=0,no-cache,no-store') + + if None is not password and set('*') == set(password): + password = sickbeard.PLEX_PASSWORD finalResult = '' - for curHost in [x.strip() for x in host.split(",")]: - curResult = notifiers.plex_notifier.test_notify(urllib.unquote_plus(curHost), username, password) - if len(curResult.split(":")) > 2 and 'OK' in curResult.split(":")[2]: - finalResult += "Test Plex notice sent successfully to " + urllib.unquote_plus(curHost) + for curHost in [x.strip() for x in host.split(',')]: + curResult = notifiers.plex_notifier.test_notify_pmc(urllib.unquote_plus(curHost), username, password) + if len(curResult.split(':')) > 2 and 'OK' in curResult.split(':')[2]: + finalResult += 'Successful test notice sent to Plex client ... ' + urllib.unquote_plus(curHost) else: - finalResult += "Test Plex notice failed to " + urllib.unquote_plus(curHost) - finalResult += "<br />\n" + finalResult += 'Test failed for Plex client ... ' + urllib.unquote_plus(curHost) + finalResult += '<br />' + '\n' + + ui.notifications.message('Tested Plex client(s): ', urllib.unquote_plus(host.replace(',', ', '))) return finalResult + def testPMS(self, host=None, username=None, password=None, plex_server_token=None): + self.set_header('Cache-Control', 'max-age=0,no-cache,no-store') + if None is not password and set('*') == set(password): + password = sickbeard.PLEX_PASSWORD + + finalResult = '' + + curResult = notifiers.plex_notifier.test_notify_pms(urllib.unquote_plus(host), username, password, plex_server_token) + if None is curResult: + finalResult += 'Successful test of Plex server(s) ... ' + urllib.unquote_plus(host.replace(',', ', ')) + else: + finalResult += 'Test failed for Plex server(s) ... ' + urllib.unquote_plus(curResult.replace(',', ', ')) + finalResult += '<br />' + '\n' + + ui.notifications.message('Tested Plex Media Server host(s): ', urllib.unquote_plus(host.replace(',', ', '))) + + return finalResult + def testLibnotify(self): # self.set_header('Cache-Control', 'max-age=0,no-cache,no-store') @@ -960,8 +986,7 @@ class Home(WebRoot): else: return '{"message": "Unable to find NMJ Database at location: %(dbloc)s. Is the right location selected and PCH running?", "database": ""}' % { "dbloc": dbloc} - - + def testTrakt(self, username=None, password=None, disable_ssl=None, blacklist_name=None): # self.set_header('Cache-Control', 'max-age=0,no-cache,no-store') if disable_ssl == 'true': @@ -1183,14 +1208,18 @@ class Home(WebRoot): if not sickbeard.showQueueScheduler.action.isBeingAdded(showObj): if not sickbeard.showQueueScheduler.action.isBeingUpdated(showObj): + if showObj.paused: + t.submenu.append({'title': 'Resume', 'path': 'home/togglePause?show=%d' % showObj.indexerid}) + else: + t.submenu.append({'title': 'Pause', 'path': 'home/togglePause?show=%d' % showObj.indexerid}) + t.submenu.append( {'title': 'Remove', 'path': 'home/deleteShow?show=%d' % showObj.indexerid, 'confirm': True}) t.submenu.append({'title': 'Re-scan files', 'path': 'home/refreshShow?show=%d' % showObj.indexerid}) t.submenu.append( {'title': 'Force Full Update', 'path': 'home/updateShow?show=%d&force=1' % showObj.indexerid}) t.submenu.append({'title': 'Update show in KODI', - 'path': 'home/updateKODI?showName=%s' % urllib.quote_plus( - showObj.name.encode('utf-8')), 'requires': self.haveKODI}) + 'path': 'home/updateKODI?show=%d' % showObj.indexerid, 'requires': self.haveKODI}) t.submenu.append({'title': 'Preview Rename', 'path': 'home/testRename?show=%d' % showObj.indexerid}) if sickbeard.USE_SUBTITLES and not sickbeard.showQueueScheduler.action.isBeingSubtitled( showObj) and showObj.subtitles: @@ -1516,7 +1545,7 @@ class Home(WebRoot): if not paused and (sickbeard.TRAKT_USE_ROLLING_DOWNLOAD and sickbeard.USE_TRAKT): # Checking if trakt and rolling_download are enable because updateWantedList() # doesn't do the distinction between a failuire and being not activated(Return false) - if not sickbeard.traktRollingScheduler.action.updateWantedList(): + if not sickbeard.traktRollingScheduler.action.updateWantedList(showObj.indexerid): errors.append("Unable to force an update on wanted episode") if do_update_scene_numbering: @@ -1536,6 +1565,31 @@ class Home(WebRoot): return self.redirect("/home/displayShow?show=" + show) + def togglePause(self, show=None): + if show is None: + return self._genericMessage("Error", "Invalid show ID") + + showObj = sickbeard.helpers.findCertainShow(sickbeard.showList, int(show)) + + if showObj is None: + return self._genericMessage("Error", "Unable to find the specified show") + + if showObj.paused: + showObj.paused = 0 + else: + showObj.paused = 1 + + showObj.saveToDB() + + if not showObj.paused and sickbeard.TRAKT_USE_ROLLING_DOWNLOAD and sickbeard.USE_TRAKT: + # Checking if trakt and rolling_download are enable because updateWantedList() + # doesn't do the distinction between a failuire and being not activated(Return false) + if not sickbeard.traktRollingScheduler.action.updateWantedList(showObj.indexerid): + errors.append("Unable to force an update on wanted episode") + + ui.notifications.message('<b>%s</b> has been %s' % (showObj.name,('resumed', 'paused')[showObj.paused])) + return self.redirect("/home/displayShow?show=" + show) + def deleteShow(self, show=None, full=0): if show is None: @@ -1628,8 +1682,14 @@ class Home(WebRoot): return self.redirect("/home/displayShow?show=" + str(showObj.indexerid)) - def updateKODI(self, showName=None): - + def updateKODI(self, show=None): + showName=None + + if show: + showObj = sickbeard.helpers.findCertainShow(sickbeard.showList, int(show)) + if showObj: + showName=urllib.quote_plus(showObj.name.encode('utf-8')) + # only send update to first host in the list -- workaround for kodi sql backend users if sickbeard.KODI_UPDATE_ONLYFIRST: # only send update to first host in the list -- workaround for kodi sql backend users @@ -1641,8 +1701,11 @@ class Home(WebRoot): ui.notifications.message("Library update command sent to KODI host(s): " + host) else: ui.notifications.error("Unable to contact one or more KODI host(s): " + host) - return self.redirect('/home/') - + + if showObj: + return self.redirect('/home/displayShow?show=' + str(showObj.indexerid)) + else: + return self.redirect('/home/') def updatePLEX(self): if notifiers.plex_notifier.update_library(): @@ -1914,7 +1977,11 @@ class Home(WebRoot): def getEpisodes(searchThread, searchstatus): results = [] showObj = sickbeard.helpers.findCertainShow(sickbeard.showList, int(searchThread.show.indexerid)) - + + if not showObj: + logger.log('No Show Object found for show with indexerID: ' + searchThread.show.indexerid, logger.ERROR) + return results + if isinstance(searchThread, sickbeard.search_queue.ManualSearchQueueItem): results.append({'show': searchThread.show.indexerid, 'episode': searchThread.segment.episode, @@ -3643,7 +3710,7 @@ class ConfigGeneral(Config): handle_reverse_proxy=None, sort_article=None, auto_update=None, notify_on_update=None, proxy_setting=None, proxy_indexers=None, anon_redirect=None, git_path=None, git_remote=None, calendar_unprotected=None, debug=None, no_restart=None, coming_eps_missed_range=None, - display_filesize=None, filter_row=None, fuzzy_dating=None, trim_zero=None, date_preset=None, date_preset_na=None, time_preset=None, + filter_row=None, fuzzy_dating=None, trim_zero=None, date_preset=None, date_preset_na=None, time_preset=None, indexer_timeout=None, download_url=None, rootDir=None, theme_name=None, git_reset=None, git_username=None, git_password=None, git_autoissues=None): @@ -3694,7 +3761,6 @@ class ConfigGeneral(Config): sickbeard.WEB_USERNAME = web_username sickbeard.WEB_PASSWORD = web_password - sickbeard.DISPLAY_FILESIZE = config.checkbox_to_value(display_filesize) sickbeard.FILTER_ROW = config.checkbox_to_value(filter_row) sickbeard.FUZZY_DATING = config.checkbox_to_value(fuzzy_dating) sickbeard.TRIM_ZERO = config.checkbox_to_value(trim_zero) @@ -3854,8 +3920,7 @@ class ConfigSearch(Config): sickbeard.RANDOMIZE_PROVIDERS = config.checkbox_to_value(randomize_providers) - sickbeard.DOWNLOAD_PROPERS = config.checkbox_to_value(download_propers) - config.change_DOWNLOAD_PROPERS(sickbeard.DOWNLOAD_PROPERS) + config.change_DOWNLOAD_PROPERS(download_propers) sickbeard.CHECK_PROPERS_INTERVAL = check_propers_interval @@ -4984,10 +5049,6 @@ class ErrorLogs(WebRoot): with ek.ek(codecs.open, *[logger.logFile + "." + str(i), 'r', 'utf-8']) as f: data += Get_Data(minLevel, f.readlines(), len(data), regex, logFilter, logSearch, maxLines) - - - - result = "".join(data) t.logLines = result