Merge pull request #673 from JackDandy/feature/ChangeValidateProv
Change validate and improve specific Torrent provider connections…
|
@ -64,6 +64,15 @@
|
||||||
* Change instantly use saved value from Search Settings/Episode Search/"Check propers every" instead of after a restart
|
* Change instantly use saved value from Search Settings/Episode Search/"Check propers every" instead of after a restart
|
||||||
* Change include OSError system messages in file system failure logs during post process
|
* Change include OSError system messages in file system failure logs during post process
|
||||||
* Fix find associated meta files to prevent orphan episode images
|
* Fix find associated meta files to prevent orphan episode images
|
||||||
|
* Add torrent provider HD4Free
|
||||||
|
* Change validate and improve specific Torrent provider connections, IPT, KAT, SCC, TPB, TB, TD, TT
|
||||||
|
* Change refactor cache for torrent providers to reduce code
|
||||||
|
* Change improve search category selection BMTV, FSH, FF, TB
|
||||||
|
* Change identify more SD release qualities
|
||||||
|
* Change update SpeedCD, MoreThan, TVChaosuk
|
||||||
|
* Change only create threads for providers needing a recent search instead of for all enabled
|
||||||
|
* Add 4489 as experimental value to "Recent search frequency" to use provider freqs instead of fixed width for all
|
||||||
|
* Change remove some logging cruft
|
||||||
|
|
||||||
|
|
||||||
### 0.11.11 (2016-04-05 19:20:00 UTC)
|
### 0.11.11 (2016-04-05 19:20:00 UTC)
|
||||||
|
|
Before Width: | Height: | Size: 345 B |
Before Width: | Height: | Size: 3.3 KiB |
Before Width: | Height: | Size: 1 KiB |
BIN
gui/slick/images/providers/hd4free.png
Normal file
After Width: | Height: | Size: 621 B |
Before Width: | Height: | Size: 398 B After Width: | Height: | Size: 600 B |
Before Width: | Height: | Size: 979 B |
BIN
gui/slick/images/providers/pfmonkey.png
Normal file
After Width: | Height: | Size: 1.2 KiB |
BIN
gui/slick/images/providers/simplynzbs.png
Normal file
After Width: | Height: | Size: 463 B |
Before Width: | Height: | Size: 3.5 KiB After Width: | Height: | Size: 673 B |
Before Width: | Height: | Size: 784 B After Width: | Height: | Size: 982 B |
Before Width: | Height: | Size: 349 B |
Before Width: | Height: | Size: 1,002 B After Width: | Height: | Size: 937 B |
Before Width: | Height: | Size: 754 B |
|
@ -37,9 +37,9 @@
|
||||||
|
|
||||||
var show_nzb_providers = <%= 'true' if sickbeard.USE_NZBS else 'false' %>;
|
var show_nzb_providers = <%= 'true' if sickbeard.USE_NZBS else 'false' %>;
|
||||||
|
|
||||||
#for $curNewznabProvider in $sickbeard.newznabProviderList:
|
#for $cur_newznab_provider in $sickbeard.newznabProviderList:
|
||||||
|
|
||||||
\$(this).addProvider('$curNewznabProvider.get_id()', '$curNewznabProvider.name', '$curNewznabProvider.url', '<%= starify(curNewznabProvider.key) %>', '$curNewznabProvider.cat_ids', $int($curNewznabProvider.default), show_nzb_providers);
|
\$(this).addProvider('$cur_newznab_provider.get_id()', '$cur_newznab_provider.name', '$cur_newznab_provider.url', '<%= starify(cur_newznab_provider.key) %>', '$cur_newznab_provider.cat_ids', $int($cur_newznab_provider.default), show_nzb_providers);
|
||||||
|
|
||||||
#end for
|
#end for
|
||||||
|
|
||||||
|
@ -47,9 +47,9 @@
|
||||||
|
|
||||||
#if $sickbeard.USE_TORRENTS
|
#if $sickbeard.USE_TORRENTS
|
||||||
|
|
||||||
#for $curTorrentRssProvider in $sickbeard.torrentRssProviderList:
|
#for $cur_torrent_rss_provider in $sickbeard.torrentRssProviderList:
|
||||||
|
|
||||||
\$(this).addTorrentRssProvider('$curTorrentRssProvider.get_id()', '$curTorrentRssProvider.name', '$curTorrentRssProvider.url', '<%= starify(curTorrentRssProvider.cookies) %>');
|
\$(this).addTorrentRssProvider('$cur_torrent_rss_provider.get_id()', '$cur_torrent_rss_provider.name', '$cur_torrent_rss_provider.url', '<%= starify(cur_torrent_rss_provider.cookies) %>');
|
||||||
|
|
||||||
#end for
|
#end for
|
||||||
|
|
||||||
|
@ -98,20 +98,20 @@
|
||||||
|
|
||||||
|
|
||||||
<ul id="provider_order_list" class="provider_order_panel">
|
<ul id="provider_order_list" class="provider_order_panel">
|
||||||
#for $curProvider in $sickbeard.providers.sortedProviderList()
|
#for $cur_provider in [$x for $x in $sickbeard.providers.sortedProviderList()
|
||||||
#if $curProvider.providerType == $GenericProvider.NZB and not $sickbeard.USE_NZBS
|
if $x.providerType == $GenericProvider.NZB and $sickbeard.USE_NZBS or
|
||||||
#continue
|
$x.providerType == $GenericProvider.TORRENT and $sickbeard.USE_TORRENTS]
|
||||||
#elif $curProvider.providerType == $GenericProvider.TORRENT and not $sickbeard.USE_TORRENTS
|
#set $cur_name = $cur_provider.get_id()
|
||||||
#continue
|
#set $cur_url = $cur_provider.url
|
||||||
#end if
|
#set $tip = ($cur_provider.name, 'Site Down')[not $cur_url]
|
||||||
#set $curName = $curProvider.get_id()
|
#set $state = ('', ' <span class="red-text">(Site Down?)</span>')[not $cur_url]
|
||||||
<li class="ui-state-default" id="$curName">
|
<li class="ui-state-default" id="$cur_name">
|
||||||
<input type="checkbox" id="enable_$curName" class="provider_enabler" <%= html_checked if curProvider.is_enabled() else '' %>/>
|
<input type="checkbox" id="enable_$cur_name" class="provider_enabler" <%= html_checked if cur_provider.is_enabled() else '' %>/>
|
||||||
<a href="<%= anon_url(curProvider.url) %>" class="imgLink" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;"><img src="$sbRoot/images/providers/$curProvider.image_name()" alt="$curProvider.name" title="$curProvider.name" width="16" height="16" style="vertical-align:middle;"/></a>
|
<a href="<%= anon_url(cur_url) %>" class="imgLink" rel="noreferrer" onclick="window.open(this.href,'_blank');return false;"><img src="$sbRoot/images/providers/$cur_provider.image_name()" alt="$tip" title="$tip" width="16" height="16" style="vertical-align:middle" /></a>
|
||||||
<span style="vertical-align:middle">$curProvider.name</span>
|
<span style="vertical-align:middle">$cur_provider.name$state</span>
|
||||||
<%= '*' if not curProvider.supportsBacklog else '' %>
|
<%= '*' if not cur_provider.supports_backlog else '' %>
|
||||||
<span class="ui-icon ui-icon-arrowthick-2-n-s pull-right" style="margin-top:3px"></span>
|
<span class="ui-icon ui-icon-arrowthick-2-n-s pull-right" style="margin-top:3px"></span>
|
||||||
</li>
|
</li>
|
||||||
#end for
|
#end for
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
|
@ -150,16 +150,13 @@
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
#set $provider_config_list_enabled = []
|
#set $provider_config_list_enabled = []
|
||||||
#set $provider_config_list = []
|
#set $provider_config_list = []
|
||||||
#for $curProvider in $sickbeard.providers.sortedProviderList()
|
#for $cur_provider in [$x for $x in $sickbeard.providers.sortedProviderList()
|
||||||
#if $curProvider.providerType == $GenericProvider.NZB and not $sickbeard.USE_NZBS
|
if $x.providerType == $GenericProvider.NZB and $sickbeard.USE_NZBS or
|
||||||
#continue
|
$x.providerType == $GenericProvider.TORRENT and $sickbeard.USE_TORRENTS]
|
||||||
#elif $curProvider.providerType == $GenericProvider.TORRENT and not $sickbeard.USE_TORRENTS
|
#if $cur_provider.is_enabled()
|
||||||
#continue
|
$provider_config_list_enabled.append($cur_provider)
|
||||||
#end if
|
|
||||||
#if $curProvider.is_enabled()
|
|
||||||
$provider_config_list_enabled.append($curProvider)
|
|
||||||
#else
|
#else
|
||||||
$provider_config_list.append($curProvider)
|
$provider_config_list.append($cur_provider)
|
||||||
#end if
|
#end if
|
||||||
#end for
|
#end for
|
||||||
|
|
||||||
|
@ -188,71 +185,71 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- start div for editing providers //-->
|
<!-- start div for editing providers //-->
|
||||||
#for $curNewznabProvider in [$curProvider for $curProvider in $sickbeard.newznabProviderList]
|
#for $cur_newznab_provider in [$cur_provider for $cur_provider in $sickbeard.newznabProviderList]
|
||||||
<div class="providerDiv" id="${curNewznabProvider.get_id()}Div">
|
<div class="providerDiv" id="${cur_newznab_provider.get_id()}Div">
|
||||||
#if $curNewznabProvider.default and $curNewznabProvider.needs_auth
|
#if $cur_newznab_provider.default and $cur_newznab_provider.needs_auth
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNewznabProvider.get_id()}_url">
|
<label for="${cur_newznab_provider.get_id()}_url">
|
||||||
<span class="component-title">URL</span>
|
<span class="component-title">URL</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="text" id="${curNewznabProvider.get_id()}_url" value="$curNewznabProvider.url" class="form-control input-sm input350" disabled/>
|
<input type="text" id="${cur_newznab_provider.get_id()}_url" value="$cur_newznab_provider.url" class="form-control input-sm input350" disabled/>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNewznabProvider.get_id()}_hash">
|
<label for="${cur_newznab_provider.get_id()}_hash">
|
||||||
<span class="component-title">API key</span>
|
<span class="component-title">API key</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="text" id="${curNewznabProvider.get_id()}_hash" value="<%= starify(curNewznabProvider.key) %>" newznab_name="${curNewznabProvider.get_id()}_hash" class="newznab_key form-control input-sm input350" />
|
<input type="text" id="${cur_newznab_provider.get_id()}_hash" value="<%= starify(cur_newznab_provider.key) %>" newznab_name="${cur_newznab_provider.get_id()}_hash" class="newznab_key form-control input-sm input350" />
|
||||||
<div class="clear-left"><p>get API key from provider website</p></div>
|
<div class="clear-left"><p>get API key from provider website</p></div>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curNewznabProvider, 'enable_recentsearch') and $curNewznabProvider.supportsBacklog:
|
#if $hasattr($cur_newznab_provider, 'enable_recentsearch') and $cur_newznab_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNewznabProvider.get_id()}_enable_recentsearch">
|
<label for="${cur_newznab_provider.get_id()}_enable_recentsearch">
|
||||||
<span class="component-title">Enable recent searches</span>
|
<span class="component-title">Enable recent searches</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curNewznabProvider.get_id()}_enable_recentsearch" id="${curNewznabProvider.get_id()}_enable_recentsearch" <%= html_checked if curNewznabProvider.enable_recentsearch else '' %>/>
|
<input type="checkbox" name="${cur_newznab_provider.get_id()}_enable_recentsearch" id="${cur_newznab_provider.get_id()}_enable_recentsearch" <%= html_checked if cur_newznab_provider.enable_recentsearch else '' %>/>
|
||||||
<p>perform recent searches at provider</p>
|
<p>perform recent searches at provider</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curNewznabProvider, 'enable_backlog') and $curNewznabProvider.supportsBacklog:
|
#if $hasattr($cur_newznab_provider, 'enable_backlog') and $cur_newznab_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNewznabProvider.get_id()}_enable_backlog">
|
<label for="${cur_newznab_provider.get_id()}_enable_backlog">
|
||||||
<span class="component-title">Enable backlog searches</span>
|
<span class="component-title">Enable backlog searches</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curNewznabProvider.get_id()}_enable_backlog" id="${curNewznabProvider.get_id()}_enable_backlog" <%= html_checked if curNewznabProvider.enable_backlog else '' %>/>
|
<input type="checkbox" name="${cur_newznab_provider.get_id()}_enable_backlog" id="${cur_newznab_provider.get_id()}_enable_backlog" <%= html_checked if cur_newznab_provider.enable_backlog else '' %>/>
|
||||||
<p>perform backlog searches at provider</p>
|
<p>perform backlog searches at provider</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curNewznabProvider, 'search_mode') and $curNewznabProvider.supportsBacklog:
|
#if $hasattr($cur_newznab_provider, 'search_mode') and $cur_newznab_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<span class="component-title">Season search mode</span>
|
<span class="component-title">Episode search mode</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<label class="space-right">
|
<label class="space-right">
|
||||||
<input type="radio" name="${curNewznabProvider.get_id()}_search_mode" id="${curNewznabProvider.get_id()}_search_mode_sponly" value="sponly" <%= html_checked if 'sponly' == curNewznabProvider.search_mode else '' %>/>season packs only
|
<input type="radio" name="${cur_newznab_provider.get_id()}_search_mode" id="${cur_newznab_provider.get_id()}_search_mode_eponly" value="eponly" <%= html_checked if 'eponly' == cur_newznab_provider.search_mode else '' %>/>episodes only (normal use)
|
||||||
</label>
|
</label>
|
||||||
<label>
|
<label>
|
||||||
<input type="radio" name="${curNewznabProvider.get_id()}_search_mode" id="${curNewznabProvider.get_id()}_search_mode_eponly" value="eponly" <%= html_checked if 'eponly' == curNewznabProvider.search_mode else '' %>/>episodes only
|
<input type="radio" name="${cur_newznab_provider.get_id()}_search_mode" id="${cur_newznab_provider.get_id()}_search_mode_sponly" value="sponly" <%= html_checked if 'sponly' == cur_newznab_provider.search_mode else '' %>/>season packs only
|
||||||
</label>
|
</label>
|
||||||
<p>when searching for complete seasons, search for packs or collect single episodes</p>
|
<p>when searching for episodes, collect single episodes or search for packs</p>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curNewznabProvider, 'search_fallback') and $curNewznabProvider.supportsBacklog:
|
#if $hasattr($cur_newznab_provider, 'search_fallback') and $cur_newznab_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNewznabProvider.get_id()}_search_fallback">
|
<label for="${cur_newznab_provider.get_id()}_search_fallback">
|
||||||
<span class="component-title">Season search fallback</span>
|
<span class="component-title">Episode search fallback</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curNewznabProvider.get_id()}_search_fallback" id="${curNewznabProvider.get_id()}_search_fallback" <%= html_checked if curNewznabProvider.search_fallback else '' %>/>
|
<input type="checkbox" name="${cur_newznab_provider.get_id()}_search_fallback" id="${cur_newznab_provider.get_id()}_search_fallback" <%= html_checked if cur_newznab_provider.search_fallback else '' %>/>
|
||||||
<p>run the alternate season search mode when a complete season is not found</p>
|
<p>use the alternate episode search mode if no match is found -- warning; may result in a heavy block hit -- best use: manually collect a season, disable after.</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
@ -262,80 +259,81 @@
|
||||||
##
|
##
|
||||||
|
|
||||||
##
|
##
|
||||||
#for $curNzbProvider in [$curProvider for $curProvider in $sickbeard.providers.sortedProviderList() if $curProvider.providerType == $GenericProvider.NZB and $curProvider not in $sickbeard.newznabProviderList]:
|
#for $cur_nzb_provider in [$cur_provider for $cur_provider in $sickbeard.providers.sortedProviderList()
|
||||||
<div class="providerDiv" id="${curNzbProvider.get_id()}Div">
|
if $cur_provider.providerType == $GenericProvider.NZB and $cur_provider not in $sickbeard.newznabProviderList]:
|
||||||
#if $hasattr($curNzbProvider, 'username'):
|
<div class="providerDiv" id="${cur_nzb_provider.get_id()}Div">
|
||||||
|
#if $hasattr($cur_nzb_provider, 'username'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNzbProvider.get_id()}_username">
|
<label for="${cur_nzb_provider.get_id()}_username">
|
||||||
<span class="component-title">Username</span>
|
<span class="component-title">Username</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="text" name="${curNzbProvider.get_id()}_username" value="$curNzbProvider.username" class="form-control input-sm input350" />
|
<input type="text" name="${cur_nzb_provider.get_id()}_username" value="$cur_nzb_provider.username" class="form-control input-sm input350" />
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curNzbProvider, 'api_key'):
|
#if $hasattr($cur_nzb_provider, 'api_key'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNzbProvider.get_id()}_api_key">
|
<label for="${cur_nzb_provider.get_id()}_api_key">
|
||||||
<span class="component-title">API key</span>
|
<span class="component-title">API key</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
#set $field_name = curNzbProvider.get_id() + '_api_key'
|
#set $field_name = cur_nzb_provider.get_id() + '_api_key'
|
||||||
<input type="text" name="$field_name" value="<%= starify(curNzbProvider.api_key) %>" class="form-control input-sm input350" />
|
<input type="text" name="$field_name" value="<%= starify(cur_nzb_provider.api_key) %>" class="form-control input-sm input350" />
|
||||||
#if callable(getattr(curNzbProvider, 'ui_string', None))
|
#if callable(getattr(cur_nzb_provider, 'ui_string', None))
|
||||||
<div class="clear-left"><p>${curNzbProvider.ui_string($field_name)}</p></div>
|
<div class="clear-left"><p>${cur_nzb_provider.ui_string($field_name)}</p></div>
|
||||||
#end if
|
#end if
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curNzbProvider, 'enable_recentsearch') and $curNzbProvider.supportsBacklog:
|
#if $hasattr($cur_nzb_provider, 'enable_recentsearch') and $cur_nzb_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNzbProvider.get_id()}_enable_recentsearch">
|
<label for="${cur_nzb_provider.get_id()}_enable_recentsearch">
|
||||||
<span class="component-title">Enable recent searches</span>
|
<span class="component-title">Enable recent searches</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curNzbProvider.get_id()}_enable_recentsearch" id="${curNzbProvider.get_id()}_enable_recentsearch" <%= html_checked if curNzbProvider.enable_recentsearch else '' %>/>
|
<input type="checkbox" name="${cur_nzb_provider.get_id()}_enable_recentsearch" id="${cur_nzb_provider.get_id()}_enable_recentsearch" <%= html_checked if cur_nzb_provider.enable_recentsearch else '' %>/>
|
||||||
<p>enable provider to perform recent searches.</p>
|
<p>enable provider to perform recent searches</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curNzbProvider, 'enable_backlog') and $curNzbProvider.supportsBacklog:
|
#if $hasattr($cur_nzb_provider, 'enable_backlog') and $cur_nzb_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNzbProvider.get_id()}_enable_backlog">
|
<label for="${cur_nzb_provider.get_id()}_enable_backlog">
|
||||||
<span class="component-title">Enable backlog searches</span>
|
<span class="component-title">Enable backlog searches</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curNzbProvider.get_id()}_enable_backlog" id="${curNzbProvider.get_id()}_enable_backlog" <%= html_checked if curNzbProvider.enable_backlog else '' %>/>
|
<input type="checkbox" name="${cur_nzb_provider.get_id()}_enable_backlog" id="${cur_nzb_provider.get_id()}_enable_backlog" <%= html_checked if cur_nzb_provider.enable_backlog else '' %>/>
|
||||||
<p>enable provider to perform backlog searches.</p>
|
<p>enable provider to perform backlog searches</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curNzbProvider, 'search_mode') and $curNzbProvider.supportsBacklog:
|
#if $hasattr($cur_nzb_provider, 'search_mode') and $cur_nzb_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<span class="component-title">Season search mode</span>
|
<span class="component-title">Episode search mode</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<label class="space-right">
|
<label class="space-right">
|
||||||
<input type="radio" name="${curNzbProvider.get_id()}_search_mode" id="${curNzbProvider.get_id()}_search_mode_sponly" value="sponly" <%= html_checked if 'sponly' == curNzbProvider.search_mode else '' %>/>season packs only
|
<input type="radio" name="${cur_nzb_provider.get_id()}_search_mode" id="${cur_nzb_provider.get_id()}_search_mode_eponly" value="eponly" <%= html_checked if 'eponly' == cur_nzb_provider.search_mode else '' %>/>episodes only (normal use)
|
||||||
</label>
|
</label>
|
||||||
<label>
|
<label>
|
||||||
<input type="radio" name="${curNzbProvider.get_id()}_search_mode" id="${curNzbProvider.get_id()}_search_mode_eponly" value="eponly" <%= html_checked if 'eponly' == curNzbProvider.search_mode else '' %>/>episodes only
|
<input type="radio" name="${cur_nzb_provider.get_id()}_search_mode" id="${cur_nzb_provider.get_id()}_search_mode_sponly" value="sponly" <%= html_checked if 'sponly' == cur_nzb_provider.search_mode else '' %>/>season packs only
|
||||||
</label>
|
</label>
|
||||||
<p>when searching for complete seasons, search for packs or collect single episodes</p>
|
<p>when searching for episodes, collect single episodes or search for packs</p>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curNzbProvider, 'search_fallback') and $curNzbProvider.supportsBacklog:
|
#if $hasattr($cur_nzb_provider, 'search_fallback') and $cur_nzb_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curNzbProvider.get_id()}_search_fallback">
|
<label for="${cur_nzb_provider.get_id()}_search_fallback">
|
||||||
<span class="component-title">Season search fallback</span>
|
<span class="component-title">Episode search fallback</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curNzbProvider.get_id()}_search_fallback" id="${curNzbProvider.get_id()}_search_fallback" <%= html_checked if curNzbProvider.search_fallback else '' %>/>
|
<input type="checkbox" name="${cur_nzb_provider.get_id()}_search_fallback" id="${cur_nzb_provider.get_id()}_search_fallback" <%= html_checked if cur_nzb_provider.search_fallback else '' %>/>
|
||||||
<p>run the alternate season search mode when a complete season is not found</p>
|
<p>use the alternate episode search mode if no match is found -- warning; may result in a heavy block hit -- best use: manually collect a season, disable after.</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if not $curNzbProvider.supportsBacklog:
|
#if not $cur_nzb_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<span class="component-desc">The latest releases are the focus of this provider, no backlog searching</span>
|
<span class="component-desc">The latest releases are the focus of this provider, no backlog searching</span>
|
||||||
</div>
|
</div>
|
||||||
|
@ -345,233 +343,229 @@
|
||||||
##
|
##
|
||||||
|
|
||||||
##
|
##
|
||||||
#for $curTorrentProvider in [$curProvider for $curProvider in $sickbeard.providers.sortedProviderList() if $curProvider.providerType == $GenericProvider.TORRENT]:
|
#for $cur_torrent_provider in [$cur_provider for $cur_provider in $sickbeard.providers.sortedProviderList()
|
||||||
<div class="providerDiv" id="${curTorrentProvider.get_id()}Div">
|
if $cur_provider.providerType == $GenericProvider.TORRENT]:
|
||||||
#if callable(getattr(curTorrentProvider, 'ui_string', None))
|
<div class="providerDiv" id="${cur_torrent_provider.get_id()}Div">
|
||||||
#set $field_name = curTorrentProvider.get_id() + '_tip'
|
#if callable(getattr(cur_torrent_provider, 'ui_string', None))
|
||||||
#set $tip_text = curTorrentProvider.ui_string($field_name)
|
#set $field_name = cur_torrent_provider.get_id() + '_tip'
|
||||||
|
#set $tip_text = cur_torrent_provider.ui_string($field_name)
|
||||||
#if $tip_text
|
#if $tip_text
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<span class="component-desc" style="margin:0;width:100%">
|
<span class="component-desc" style="margin:0;width:100%">
|
||||||
<div class="clear-left"><p class="grey-text"><span class="red-text">Important! ${curTorrentProvider.name}</span> $tip_text</p></div>
|
<div class="clear-left"><p class="grey-text"><span class="red-text">Important! ${cur_torrent_provider.name}</span> $tip_text</p></div>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'api_key'):
|
#if $getattr($cur_torrent_provider, 'url_edit', None):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_api_key">
|
<label for="url_base">
|
||||||
<span class="component-title">Api key:</span>
|
#set $url_label = callable(getattr(cur_torrent_provider, 'ui_string', None)) and cur_torrent_provider.ui_string(cur_torrent_provider.get_id() + '_site_url') or 'Site URL'
|
||||||
|
<span class="component-title">$url_label</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="text" name="${curTorrentProvider.get_id()}_api_key" id="${curTorrentProvider.get_id()}_api_key" value="<%= starify(curTorrentProvider.api_key) %>" class="form-control input-sm input350" />
|
#set $field_name = cur_torrent_provider.get_id() + '_url_edit'
|
||||||
</span>
|
<input type="text" name="$field_name" id="$field_name" value="<%= ', '.join(cur_torrent_provider.url_home) %>" class="form-control input-sm input350" />
|
||||||
</label>
|
#if callable(getattr(cur_torrent_provider, 'ui_string', None))
|
||||||
</div>
|
<div class="clear-left"><p>${cur_torrent_provider.ui_string($field_name)}</p></div>
|
||||||
#end if
|
|
||||||
#if $hasattr($curTorrentProvider, 'digest'):
|
|
||||||
<div class="field-pair">
|
|
||||||
<label for="${curTorrentProvider.get_id()}_digest">
|
|
||||||
<span class="component-title">Cookies:</span>
|
|
||||||
<span class="component-desc">
|
|
||||||
#set $field_name = curTorrentProvider.get_id() + '_digest'
|
|
||||||
<input type="text" name="$field_name" id="$field_name" value="<%= starify(curTorrentProvider.digest) %>" class="form-control input-sm input350" />
|
|
||||||
#if callable(getattr(curTorrentProvider, 'ui_string', None))
|
|
||||||
<div class="clear-left"><p>${curTorrentProvider.ui_string($field_name)}</p></div>
|
|
||||||
#end if
|
#end if
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'hash'):
|
#if $hasattr($cur_torrent_provider, 'api_key'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_hash">
|
<label for="${cur_torrent_provider.get_id()}_api_key">
|
||||||
|
<span class="component-title">Api key:</span>
|
||||||
|
<span class="component-desc">
|
||||||
|
<input type="text" name="${cur_torrent_provider.get_id()}_api_key" id="${cur_torrent_provider.get_id()}_api_key" value="<%= starify(cur_torrent_provider.api_key) %>" class="form-control input-sm input350" />
|
||||||
|
</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
#end if
|
||||||
|
#if $hasattr($cur_torrent_provider, 'digest'):
|
||||||
|
#set $field_name = cur_torrent_provider.get_id() + '_digest'
|
||||||
|
<div class="field-pair">
|
||||||
|
<label for="$field_name">
|
||||||
|
<span class="component-title">Cookies:</span>
|
||||||
|
<span class="component-desc">
|
||||||
|
<input type="text" name="$field_name" id="$field_name" value="<%= starify(cur_torrent_provider.digest) %>" class="form-control input-sm input350" />
|
||||||
|
#if callable(getattr(cur_torrent_provider, 'ui_string', None))
|
||||||
|
<div class="clear-left"><p>${cur_torrent_provider.ui_string($field_name)}</p></div>
|
||||||
|
#end if
|
||||||
|
</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
#end if
|
||||||
|
#if $hasattr($cur_torrent_provider, 'hash'):
|
||||||
|
<div class="field-pair">
|
||||||
|
<label for="${cur_torrent_provider.get_id()}_hash">
|
||||||
<span class="component-title">Hash:</span>
|
<span class="component-title">Hash:</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="text" name="${curTorrentProvider.get_id()}_hash" id="${curTorrentProvider.get_id()}_hash" value="$curTorrentProvider.hash" class="form-control input-sm input350" />
|
<input type="text" name="${cur_torrent_provider.get_id()}_hash" id="${cur_torrent_provider.get_id()}_hash" value="$cur_torrent_provider.hash" class="form-control input-sm input350" />
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'username'):
|
#for $user_type in ['username', 'uid']:
|
||||||
|
#if $hasattr($cur_torrent_provider, $user_type):
|
||||||
|
#set $prov_type = '%s_%s' % ($cur_torrent_provider.get_id(), $user_type)
|
||||||
|
#set $user_value = $getattr($cur_torrent_provider, $user_type) or ''
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_username">
|
<label for="$prov_type">
|
||||||
<span class="component-title">Username:</span>
|
<span class="component-title">$user_type.capitalize():</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="text" name="${curTorrentProvider.get_id()}_username" id="${curTorrentProvider.get_id()}_username" value="$curTorrentProvider.username" class="form-control input-sm input350" />
|
<input type="text" name="$prov_type" id="$prov_type" value="$user_value" class="form-control input-sm input350" />
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#break
|
||||||
#if $hasattr($curTorrentProvider, 'password'):
|
#end if
|
||||||
|
#end for
|
||||||
|
#if $hasattr($cur_torrent_provider, 'password'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_password">
|
<label for="${cur_torrent_provider.get_id()}_password">
|
||||||
<span class="component-title">Password:</span>
|
<span class="component-title">Password:</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="password" name="${curTorrentProvider.get_id()}_password" id="${curTorrentProvider.get_id()}_password" value="#echo $curTorrentProvider.password and '*' * len($curTorrentProvider.password) or ''#" class="form-control input-sm input350" />
|
<input type="password" name="${cur_torrent_provider.get_id()}_password" id="${cur_torrent_provider.get_id()}_password" value="#echo $cur_torrent_provider.password and '*' * len($cur_torrent_provider.password) or ''#" class="form-control input-sm input350" />
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'passkey'):
|
#if $hasattr($cur_torrent_provider, 'passkey'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_passkey">
|
<label for="${cur_torrent_provider.get_id()}_passkey">
|
||||||
<span class="component-title">Passkey:</span>
|
<span class="component-title">Passkey:</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="text" name="${curTorrentProvider.get_id()}_passkey" id="${curTorrentProvider.get_id()}_passkey" value="<%= starify(curTorrentProvider.passkey) %>" class="form-control input-sm input350" />
|
<input type="text" name="${cur_torrent_provider.get_id()}_passkey" id="${cur_torrent_provider.get_id()}_passkey" value="<%= starify(cur_torrent_provider.passkey) %>" class="form-control input-sm input350" />
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, '_seed_ratio') and 'blackhole' != $sickbeard.TORRENT_METHOD:
|
#if $hasattr($cur_torrent_provider, '_seed_ratio') and 'blackhole' != $sickbeard.TORRENT_METHOD:
|
||||||
#set $torrent_method_text = {'utorrent': 'uTorrent', 'transmission': 'Transmission', 'deluge': 'Deluge', 'download_station': 'Synology DS', 'rtorrent': 'rTorrent'}
|
#set $torrent_method_text = {'utorrent': 'uTorrent', 'transmission': 'Transmission', 'deluge': 'Deluge', 'download_station': 'Synology DS', 'rtorrent': 'rTorrent'}
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_ratio">
|
<label for="${cur_torrent_provider.get_id()}_ratio">
|
||||||
<span class="component-title" id="${curTorrentProvider.get_id()}_ratio_desc">Seed until ratio (the goal)</span>
|
<span class="component-title" id="${cur_torrent_provider.get_id()}_ratio_desc">Seed until ratio (the goal)</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="number" name="${curTorrentProvider.get_id()}_ratio" id="${curTorrentProvider.get_id()}_ratio" value="$curTorrentProvider._seed_ratio" class="form-control input-sm input75" />
|
<input type="number" name="${cur_torrent_provider.get_id()}_ratio" id="${cur_torrent_provider.get_id()}_ratio" value="$cur_torrent_provider._seed_ratio" class="form-control input-sm input75" />
|
||||||
<p>this ratio is requested of each item sent to $torrent_method_text[$sickbeard.TORRENT_METHOD]</p>
|
<p>this ratio is requested of each item sent to $torrent_method_text[$sickbeard.TORRENT_METHOD]</p>
|
||||||
<div class="clear-left"><p>(#if 'Transmission' in $torrent_method_text[$sickbeard.TORRENT_METHOD]#set -1 to seed forever, or #end if#leave blank for the $torrent_method_text[$sickbeard.TORRENT_METHOD] setting)</p></div>
|
<div class="clear-left"><p>(#if 'Transmission' in $torrent_method_text[$sickbeard.TORRENT_METHOD]#set -1 to seed forever, or #end if#leave blank for the $torrent_method_text[$sickbeard.TORRENT_METHOD] setting)</p></div>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'seed_time') and 'utorrent' == $sickbeard.TORRENT_METHOD:
|
#if $hasattr($cur_torrent_provider, 'seed_time') and 'utorrent' == $sickbeard.TORRENT_METHOD:
|
||||||
#set $torrent_method_text = {'utorrent': 'uTorrent'}
|
#set $torrent_method_text = {'utorrent': 'uTorrent'}
|
||||||
#set $use_default = 'to use the %s min <a href="%s/config/search/#core-component-group3">torrent search setting minumum default</a>' % ($sickbeard.TORRENT_SEED_TIME, $sbRoot) if $sickbeard.TORRENT_SEED_TIME else 'for the %s setting' % $torrent_method_text[$sickbeard.TORRENT_METHOD]
|
#set $use_default = 'to use the %s min <a href="%s/config/search/#core-component-group3">torrent search setting minumum default</a>' % ($sickbeard.TORRENT_SEED_TIME, $sbRoot) if $sickbeard.TORRENT_SEED_TIME else 'for the %s setting' % $torrent_method_text[$sickbeard.TORRENT_METHOD]
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_seed_time">
|
<label for="${cur_torrent_provider.get_id()}_seed_time">
|
||||||
<span class="component-title" id="${curTorrentProvider.get_id()}_seed_time_desc">Seed time (provider default)</span>
|
<span class="component-title" id="${cur_torrent_provider.get_id()}_seed_time_desc">Seed time (provider default)</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="number" name="${curTorrentProvider.get_id()}_seed_time" id="${curTorrentProvider.get_id()}_seed_time" value="$curTorrentProvider.seed_time" class="form-control input-sm input75" />
|
<input type="number" name="${cur_torrent_provider.get_id()}_seed_time" id="${cur_torrent_provider.get_id()}_seed_time" value="$cur_torrent_provider.seed_time" class="form-control input-sm input75" />
|
||||||
<p>set 1 or more minimum minutes for each item sent to $torrent_method_text[$sickbeard.TORRENT_METHOD]</p>
|
<p>set 1 or more minimum minutes for each item sent to $torrent_method_text[$sickbeard.TORRENT_METHOD]</p>
|
||||||
<div class="clear-left"><p>(leave blank $use_default)</p></div>
|
<div class="clear-left"><p>(leave blank $use_default)</p></div>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'minseed'):
|
#if $hasattr($cur_torrent_provider, 'minseed'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_minseed">
|
<label for="${cur_torrent_provider.get_id()}_minseed">
|
||||||
<span class="component-title" id="${curTorrentProvider.get_id()}_minseed_desc">Minimum seeders</span>
|
<span class="component-title" id="${cur_torrent_provider.get_id()}_minseed_desc">Minimum seeders</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="number" name="${curTorrentProvider.get_id()}_minseed" id="${curTorrentProvider.get_id()}_minseed" value="$curTorrentProvider.minseed" class="form-control input-sm input75" />
|
<input type="number" name="${cur_torrent_provider.get_id()}_minseed" id="${cur_torrent_provider.get_id()}_minseed" value="$cur_torrent_provider.minseed" class="form-control input-sm input75" />
|
||||||
<p>a release must have to be snatch worthy</p>
|
<p>a release must have to be snatch worthy</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'minleech'):
|
#if $hasattr($cur_torrent_provider, 'minleech'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_minleech">
|
<label for="${cur_torrent_provider.get_id()}_minleech">
|
||||||
<span class="component-title" id="${curTorrentProvider.get_id()}_minleech_desc">Minimum leechers</span>
|
<span class="component-title" id="${cur_torrent_provider.get_id()}_minleech_desc">Minimum leechers</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="number" name="${curTorrentProvider.get_id()}_minleech" id="${curTorrentProvider.get_id()}_minleech" value="$curTorrentProvider.minleech" class="form-control input-sm input75" />
|
<input type="number" name="${cur_torrent_provider.get_id()}_minleech" id="${cur_torrent_provider.get_id()}_minleech" value="$cur_torrent_provider.minleech" class="form-control input-sm input75" />
|
||||||
<p>a release must have to be snatch worthy</p>
|
<p>a release must have to be snatch worthy</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'proxy'):
|
#if $hasattr($cur_torrent_provider, 'confirmed'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_proxy">
|
<label for="${cur_torrent_provider.get_id()}_confirmed">
|
||||||
<span class="component-title">Access provider via proxy</span>
|
|
||||||
<span class="component-desc">
|
|
||||||
<input type="checkbox" class="enabler" name="${curTorrentProvider.get_id()}_proxy" id="${curTorrentProvider.get_id()}_proxy" <%= html_checked if curTorrentProvider.proxy.enabled else '' %>/>
|
|
||||||
<p>to bypass country blocking mechanisms</p>
|
|
||||||
</span>
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
#if $hasattr($curTorrentProvider.proxy, 'url'):
|
|
||||||
<div class="field-pair content_${curTorrentProvider.get_id()}_proxy" id="content_${curTorrentProvider.get_id()}_proxy">
|
|
||||||
<label for="${curTorrentProvider.get_id()}_proxy_url">
|
|
||||||
<span class="component-title">Proxy URL:</span>
|
|
||||||
<span class="component-desc">
|
|
||||||
<select name="${curTorrentProvider.get_id()}_proxy_url" id="${curTorrentProvider.get_id()}_proxy_url" class="form-control input-sm">
|
|
||||||
#for $i in $curTorrentProvider.proxy.urls.keys():
|
|
||||||
<option value="$curTorrentProvider.proxy.urls[$i]" <%= html_selected if curTorrentProvider.proxy.url == curTorrentProvider.proxy.urls[i] else '' %>>$i</option>
|
|
||||||
#end for
|
|
||||||
</select>
|
|
||||||
</span>
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
#end if
|
|
||||||
#end if
|
|
||||||
#if $hasattr($curTorrentProvider, 'confirmed'):
|
|
||||||
<div class="field-pair">
|
|
||||||
<label for="${curTorrentProvider.get_id()}_confirmed">
|
|
||||||
<span class="component-title">Confirmed download</span>
|
<span class="component-title">Confirmed download</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curTorrentProvider.get_id()}_confirmed" id="${curTorrentProvider.get_id()}_confirmed" <%= html_checked if curTorrentProvider.confirmed else '' %>/>
|
<input type="checkbox" name="${cur_torrent_provider.get_id()}_confirmed" id="${cur_torrent_provider.get_id()}_confirmed" <%= html_checked if cur_torrent_provider.confirmed else '' %>/>
|
||||||
<p>only download torrents from trusted or verified uploaders ?</p>
|
<p>only download torrents from trusted or verified uploaders ?</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'freeleech'):
|
#if $hasattr($cur_torrent_provider, 'freeleech'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_freeleech">
|
<label for="${cur_torrent_provider.get_id()}_freeleech">
|
||||||
<span class="component-title">Freeleech</span>
|
<span class="component-title">Freeleech</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curTorrentProvider.get_id()}_freeleech" id="${curTorrentProvider.get_id()}_freeleech" <%= html_checked if curTorrentProvider.freeleech else '' %>/>
|
<input type="checkbox" name="${cur_torrent_provider.get_id()}_freeleech" id="${cur_torrent_provider.get_id()}_freeleech" <%= html_checked if cur_torrent_provider.freeleech else '' %>/>
|
||||||
<p>only download <b>[FreeLeech]</b> torrents.</p>
|
<p>only download <b>[FreeLeech]</b> torrents</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'reject_m2ts'):
|
#if $hasattr($cur_torrent_provider, 'reject_m2ts'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_reject_m2ts">
|
<label for="${cur_torrent_provider.get_id()}_reject_m2ts">
|
||||||
<span class="component-title">Reject Blu-ray M2TS releases</span>
|
<span class="component-title">Reject Blu-ray M2TS releases</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curTorrentProvider.get_id()}_reject_m2ts" id="${curTorrentProvider.get_id()}_reject_m2ts" <%= html_checked if curTorrentProvider.reject_m2ts else '' %>/>
|
<input type="checkbox" name="${cur_torrent_provider.get_id()}_reject_m2ts" id="${cur_torrent_provider.get_id()}_reject_m2ts" <%= html_checked if cur_torrent_provider.reject_m2ts else '' %>/>
|
||||||
<p>enable to ignore Blu-ray MPEG-2 Transport Stream container releases</p>
|
<p>enable to ignore Blu-ray MPEG-2 Transport Stream container releases</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'enable_recentsearch') and $curTorrentProvider.supportsBacklog:
|
#if $hasattr($cur_torrent_provider, 'enable_recentsearch'):
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_enable_recentsearch">
|
<label for="${cur_torrent_provider.get_id()}_enable_recentsearch">
|
||||||
<span class="component-title">Enable recent searches</span>
|
<span class="component-title">Enable recent searches</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curTorrentProvider.get_id()}_enable_recentsearch" id="${curTorrentProvider.get_id()}_enable_recentsearch" <%= html_checked if curTorrentProvider.enable_recentsearch else '' %>/>
|
<input type="checkbox" name="${cur_torrent_provider.get_id()}_enable_recentsearch" id="${cur_torrent_provider.get_id()}_enable_recentsearch" <%= html_checked if cur_torrent_provider.enable_recentsearch else '' %>/>
|
||||||
<p>enable provider to perform recent searches.</p>
|
<p>enable provider to perform recent searches</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'enable_backlog') and $curTorrentProvider.supportsBacklog:
|
#if $hasattr($cur_torrent_provider, 'enable_backlog') and $cur_torrent_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_enable_backlog">
|
<label for="${cur_torrent_provider.get_id()}_enable_backlog">
|
||||||
<span class="component-title">Enable backlog searches</span>
|
<span class="component-title">Enable backlog searches</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curTorrentProvider.get_id()}_enable_backlog" id="${curTorrentProvider.get_id()}_enable_backlog" <%= html_checked if curTorrentProvider.enable_backlog else '' %>/>
|
<input type="checkbox" name="${cur_torrent_provider.get_id()}_enable_backlog" id="${cur_torrent_provider.get_id()}_enable_backlog" <%= html_checked if cur_torrent_provider.enable_backlog else '' %>/>
|
||||||
<p>enable provider to perform backlog searches.</p>
|
<p>enable provider to perform backlog searches</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'search_mode') and $curTorrentProvider.supportsBacklog:
|
#if $hasattr($cur_torrent_provider, 'search_mode') and $cur_torrent_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<span class="component-title">Season search mode</span>
|
<span class="component-title">Episode search mode</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<label class="space-right">
|
<label class="space-right">
|
||||||
<input type="radio" name="${curTorrentProvider.get_id()}_search_mode" id="${curTorrentProvider.get_id()}_search_mode_sponly" value="sponly" <%= html_checked if 'sponly' == curTorrentProvider.search_mode else '' %>/>season packs only
|
<input type="radio" name="${cur_torrent_provider.get_id()}_search_mode" id="${cur_torrent_provider.get_id()}_search_mode_eponly" value="eponly" <%= html_checked if 'eponly' == cur_torrent_provider.search_mode else '' %>/>episodes only (normal use)
|
||||||
</label>
|
</label>
|
||||||
<label>
|
<label>
|
||||||
<input type="radio" name="${curTorrentProvider.get_id()}_search_mode" id="${curTorrentProvider.get_id()}_search_mode_eponly" value="eponly" <%= html_checked if 'eponly' == curTorrentProvider.search_mode else '' %>/>episodes only
|
<input type="radio" name="${cur_torrent_provider.get_id()}_search_mode" id="${cur_torrent_provider.get_id()}_search_mode_sponly" value="sponly" <%= html_checked if 'sponly' == cur_torrent_provider.search_mode else '' %>/>season packs only
|
||||||
</label>
|
</label>
|
||||||
<p>when searching for complete seasons, search for packs or collect single episodes</p>
|
<p>when searching for episodes, collect single episodes or search for packs</p>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
#end if
|
#end if
|
||||||
#if $hasattr($curTorrentProvider, 'search_fallback') and $curTorrentProvider.supportsBacklog:
|
#if $hasattr($cur_torrent_provider, 'search_fallback') and $cur_torrent_provider.supports_backlog:
|
||||||
<div class="field-pair">
|
<div class="field-pair">
|
||||||
<label for="${curTorrentProvider.get_id()}_search_fallback">
|
<label for="${cur_torrent_provider.get_id()}_search_fallback">
|
||||||
<span class="component-title">Season search fallback</span>
|
<span class="component-title">Episode search fallback</span>
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="checkbox" name="${curTorrentProvider.get_id()}_search_fallback" id="${curTorrentProvider.get_id()}_search_fallback" <%= html_checked if curTorrentProvider.search_fallback else '' %>/>
|
<input type="checkbox" name="${cur_torrent_provider.get_id()}_search_fallback" id="${cur_torrent_provider.get_id()}_search_fallback" <%= html_checked if cur_torrent_provider.search_fallback else '' %>/>
|
||||||
<p>run the alternate season search mode when a complete season is not found</p>
|
<p>use the alternate episode search mode if no match is found -- warning; may result in duplicates or a heavy ratio hit -- best use: manually collect a season, disable after.</p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -73,6 +73,7 @@
|
||||||
<span class="component-desc">
|
<span class="component-desc">
|
||||||
<input type="text" name="recentsearch_frequency" value="$sickbeard.RECENTSEARCH_FREQUENCY" class="form-control input-sm input75">
|
<input type="text" name="recentsearch_frequency" value="$sickbeard.RECENTSEARCH_FREQUENCY" class="form-control input-sm input75">
|
||||||
<p>minutes between checking recent updated shows (minimum $sickbeard.MIN_RECENTSEARCH_FREQUENCY)</p>
|
<p>minutes between checking recent updated shows (minimum $sickbeard.MIN_RECENTSEARCH_FREQUENCY)</p>
|
||||||
|
<p><em class="grey-text">enter 4489 for experimental internal provider frequencies</em></p>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -30,6 +30,7 @@ from threading import Lock
|
||||||
import sys
|
import sys
|
||||||
import os.path
|
import os.path
|
||||||
import uuid
|
import uuid
|
||||||
|
import ast
|
||||||
import base64
|
import base64
|
||||||
sys.path.insert(1, os.path.abspath('../lib'))
|
sys.path.insert(1, os.path.abspath('../lib'))
|
||||||
from sickbeard import helpers, logger, db, naming, metadata, providers, scene_exceptions, scene_numbering, \
|
from sickbeard import helpers, logger, db, naming, metadata, providers, scene_exceptions, scene_numbering, \
|
||||||
|
@ -174,6 +175,7 @@ IMDB_ACCOUNTS = []
|
||||||
IMDB_DEFAULT_LIST_ID = '64552276'
|
IMDB_DEFAULT_LIST_ID = '64552276'
|
||||||
IMDB_DEFAULT_LIST_NAME = 'SickGear'
|
IMDB_DEFAULT_LIST_NAME = 'SickGear'
|
||||||
PROVIDER_ORDER = []
|
PROVIDER_ORDER = []
|
||||||
|
PROVIDER_HOMES = {}
|
||||||
|
|
||||||
NAMING_MULTI_EP = False
|
NAMING_MULTI_EP = False
|
||||||
NAMING_ANIME_MULTI_EP = False
|
NAMING_ANIME_MULTI_EP = False
|
||||||
|
@ -520,7 +522,7 @@ def initialize(consoleLogging=True):
|
||||||
KEEP_PROCESSED_DIR, PROCESS_METHOD, TV_DOWNLOAD_DIR, MIN_RECENTSEARCH_FREQUENCY, DEFAULT_UPDATE_FREQUENCY, MIN_UPDATE_FREQUENCY, UPDATE_FREQUENCY, \
|
KEEP_PROCESSED_DIR, PROCESS_METHOD, TV_DOWNLOAD_DIR, MIN_RECENTSEARCH_FREQUENCY, DEFAULT_UPDATE_FREQUENCY, MIN_UPDATE_FREQUENCY, UPDATE_FREQUENCY, \
|
||||||
showQueueScheduler, searchQueueScheduler, ROOT_DIRS, CACHE_DIR, ACTUAL_CACHE_DIR, ZONEINFO_DIR, TIMEZONE_DISPLAY, \
|
showQueueScheduler, searchQueueScheduler, ROOT_DIRS, CACHE_DIR, ACTUAL_CACHE_DIR, ZONEINFO_DIR, TIMEZONE_DISPLAY, \
|
||||||
NAMING_PATTERN, NAMING_MULTI_EP, NAMING_ANIME_MULTI_EP, NAMING_FORCE_FOLDERS, NAMING_ABD_PATTERN, NAMING_CUSTOM_ABD, NAMING_SPORTS_PATTERN, NAMING_CUSTOM_SPORTS, NAMING_ANIME_PATTERN, NAMING_CUSTOM_ANIME, NAMING_STRIP_YEAR, \
|
NAMING_PATTERN, NAMING_MULTI_EP, NAMING_ANIME_MULTI_EP, NAMING_FORCE_FOLDERS, NAMING_ABD_PATTERN, NAMING_CUSTOM_ABD, NAMING_SPORTS_PATTERN, NAMING_CUSTOM_SPORTS, NAMING_ANIME_PATTERN, NAMING_CUSTOM_ANIME, NAMING_STRIP_YEAR, \
|
||||||
RENAME_EPISODES, AIRDATE_EPISODES, properFinderScheduler, PROVIDER_ORDER, autoPostProcesserScheduler, \
|
RENAME_EPISODES, AIRDATE_EPISODES, properFinderScheduler, PROVIDER_ORDER, PROVIDER_HOMES, autoPostProcesserScheduler, \
|
||||||
providerList, newznabProviderList, torrentRssProviderList, \
|
providerList, newznabProviderList, torrentRssProviderList, \
|
||||||
EXTRA_SCRIPTS, USE_TWITTER, TWITTER_USERNAME, TWITTER_PASSWORD, TWITTER_PREFIX, RECENTSEARCH_FREQUENCY, \
|
EXTRA_SCRIPTS, USE_TWITTER, TWITTER_USERNAME, TWITTER_PASSWORD, TWITTER_PREFIX, RECENTSEARCH_FREQUENCY, \
|
||||||
USE_BOXCAR2, BOXCAR2_ACCESSTOKEN, BOXCAR2_NOTIFY_ONDOWNLOAD, BOXCAR2_NOTIFY_ONSUBTITLEDOWNLOAD, BOXCAR2_NOTIFY_ONSNATCH, BOXCAR2_SOUND, \
|
USE_BOXCAR2, BOXCAR2_ACCESSTOKEN, BOXCAR2_NOTIFY_ONDOWNLOAD, BOXCAR2_NOTIFY_ONSUBTITLEDOWNLOAD, BOXCAR2_NOTIFY_ONSNATCH, BOXCAR2_SOUND, \
|
||||||
|
@ -580,10 +582,6 @@ def initialize(consoleLogging=True):
|
||||||
|
|
||||||
ACTUAL_CACHE_DIR = check_setting_str(CFG, 'General', 'cache_dir', 'cache')
|
ACTUAL_CACHE_DIR = check_setting_str(CFG, 'General', 'cache_dir', 'cache')
|
||||||
|
|
||||||
# fix bad configs due to buggy code
|
|
||||||
if ACTUAL_CACHE_DIR == 'None':
|
|
||||||
ACTUAL_CACHE_DIR = 'cache'
|
|
||||||
|
|
||||||
# unless they specify, put the cache dir inside the data dir
|
# unless they specify, put the cache dir inside the data dir
|
||||||
if not os.path.isabs(ACTUAL_CACHE_DIR):
|
if not os.path.isabs(ACTUAL_CACHE_DIR):
|
||||||
CACHE_DIR = os.path.join(DATA_DIR, ACTUAL_CACHE_DIR)
|
CACHE_DIR = os.path.join(DATA_DIR, ACTUAL_CACHE_DIR)
|
||||||
|
@ -699,6 +697,7 @@ def initialize(consoleLogging=True):
|
||||||
SCENE_DEFAULT = bool(check_setting_int(CFG, 'General', 'scene_default', 0))
|
SCENE_DEFAULT = bool(check_setting_int(CFG, 'General', 'scene_default', 0))
|
||||||
|
|
||||||
PROVIDER_ORDER = check_setting_str(CFG, 'General', 'provider_order', '').split()
|
PROVIDER_ORDER = check_setting_str(CFG, 'General', 'provider_order', '').split()
|
||||||
|
PROVIDER_HOMES = ast.literal_eval(check_setting_str(CFG, 'General', 'provider_homes', None) or '{}')
|
||||||
|
|
||||||
NAMING_PATTERN = check_setting_str(CFG, 'General', 'naming_pattern', 'Season %0S/%SN - S%0SE%0E - %EN')
|
NAMING_PATTERN = check_setting_str(CFG, 'General', 'naming_pattern', 'Season %0S/%SN - S%0SE%0E - %EN')
|
||||||
NAMING_ABD_PATTERN = check_setting_str(CFG, 'General', 'naming_abd_pattern', '%SN - %A.D - %EN')
|
NAMING_ABD_PATTERN = check_setting_str(CFG, 'General', 'naming_abd_pattern', '%SN - %A.D - %EN')
|
||||||
|
@ -1029,28 +1028,28 @@ def initialize(consoleLogging=True):
|
||||||
prov_id = torrent_prov.get_id()
|
prov_id = torrent_prov.get_id()
|
||||||
prov_id_uc = torrent_prov.get_id().upper()
|
prov_id_uc = torrent_prov.get_id().upper()
|
||||||
torrent_prov.enabled = bool(check_setting_int(CFG, prov_id_uc, prov_id, 0))
|
torrent_prov.enabled = bool(check_setting_int(CFG, prov_id_uc, prov_id, 0))
|
||||||
|
if getattr(torrent_prov, 'url_edit', None):
|
||||||
|
torrent_prov.url_home = check_setting_str(CFG, prov_id_uc, prov_id + '_url_home', [])
|
||||||
if hasattr(torrent_prov, 'api_key'):
|
if hasattr(torrent_prov, 'api_key'):
|
||||||
torrent_prov.api_key = check_setting_str(CFG, prov_id_uc, prov_id + '_api_key', '')
|
torrent_prov.api_key = check_setting_str(CFG, prov_id_uc, prov_id + '_api_key', '')
|
||||||
if hasattr(torrent_prov, 'hash'):
|
if hasattr(torrent_prov, 'hash'):
|
||||||
torrent_prov.hash = check_setting_str(CFG, prov_id_uc, prov_id + '_hash', '')
|
torrent_prov.hash = check_setting_str(CFG, prov_id_uc, prov_id + '_hash', '')
|
||||||
if hasattr(torrent_prov, 'digest'):
|
if hasattr(torrent_prov, 'digest'):
|
||||||
torrent_prov.digest = check_setting_str(CFG, prov_id_uc, prov_id + '_digest', '')
|
torrent_prov.digest = check_setting_str(CFG, prov_id_uc, prov_id + '_digest', '')
|
||||||
if hasattr(torrent_prov, 'username'):
|
for user_type in ['username', 'uid']:
|
||||||
torrent_prov.username = check_setting_str(CFG, prov_id_uc, prov_id + '_username', '')
|
if hasattr(torrent_prov, user_type):
|
||||||
|
setattr(torrent_prov, user_type,
|
||||||
|
check_setting_str(CFG, prov_id_uc, '%s_%s' % (prov_id, user_type), ''))
|
||||||
if hasattr(torrent_prov, 'password'):
|
if hasattr(torrent_prov, 'password'):
|
||||||
torrent_prov.password = check_setting_str(CFG, prov_id_uc, prov_id + '_password', '')
|
torrent_prov.password = check_setting_str(CFG, prov_id_uc, prov_id + '_password', '')
|
||||||
if hasattr(torrent_prov, 'passkey'):
|
if hasattr(torrent_prov, 'passkey'):
|
||||||
torrent_prov.passkey = check_setting_str(CFG, prov_id_uc, prov_id + '_passkey', '')
|
torrent_prov.passkey = check_setting_str(CFG, prov_id_uc, prov_id + '_passkey', '')
|
||||||
if hasattr(torrent_prov, 'proxy'):
|
|
||||||
torrent_prov.proxy.enabled = bool(check_setting_int(CFG, prov_id_uc, prov_id + '_proxy', 0))
|
|
||||||
if hasattr(torrent_prov.proxy, 'url'):
|
|
||||||
torrent_prov.proxy.url = check_setting_str(CFG, prov_id_uc, prov_id + '_proxy_url', '')
|
|
||||||
if hasattr(torrent_prov, 'confirmed'):
|
if hasattr(torrent_prov, 'confirmed'):
|
||||||
torrent_prov.confirmed = bool(check_setting_int(CFG, prov_id_uc, prov_id + '_confirmed', 0))
|
torrent_prov.confirmed = bool(check_setting_int(CFG, prov_id_uc, prov_id + '_confirmed', 0))
|
||||||
if hasattr(torrent_prov, 'options'):
|
if hasattr(torrent_prov, 'options'):
|
||||||
torrent_prov.options = check_setting_str(CFG, prov_id_uc, prov_id + '_options', '')
|
torrent_prov.options = check_setting_str(CFG, prov_id_uc, prov_id + '_options', '')
|
||||||
if hasattr(torrent_prov, '_seed_ratio'):
|
if hasattr(torrent_prov, '_seed_ratio'):
|
||||||
torrent_prov._seed_ratio = check_setting_str(CFG, prov_id_uc, prov_id + '_seed_ratio', '').replace('None', '')
|
torrent_prov._seed_ratio = check_setting_str(CFG, prov_id_uc, prov_id + '_seed_ratio', '')
|
||||||
if hasattr(torrent_prov, 'seed_time'):
|
if hasattr(torrent_prov, 'seed_time'):
|
||||||
torrent_prov.seed_time = check_setting_int(CFG, prov_id_uc, prov_id + '_seed_time', '')
|
torrent_prov.seed_time = check_setting_int(CFG, prov_id_uc, prov_id + '_seed_time', '')
|
||||||
if hasattr(torrent_prov, 'minseed'):
|
if hasattr(torrent_prov, 'minseed'):
|
||||||
|
@ -1087,7 +1086,8 @@ def initialize(consoleLogging=True):
|
||||||
nzb_prov.search_fallback = bool(check_setting_int(CFG, prov_id_uc, prov_id + '_search_fallback', 0))
|
nzb_prov.search_fallback = bool(check_setting_int(CFG, prov_id_uc, prov_id + '_search_fallback', 0))
|
||||||
if hasattr(nzb_prov, 'enable_recentsearch'):
|
if hasattr(nzb_prov, 'enable_recentsearch'):
|
||||||
nzb_prov.enable_recentsearch = bool(check_setting_int(CFG, prov_id_uc,
|
nzb_prov.enable_recentsearch = bool(check_setting_int(CFG, prov_id_uc,
|
||||||
prov_id + '_enable_recentsearch', 1))
|
prov_id + '_enable_recentsearch', 1)) or \
|
||||||
|
not getattr(nzb_prov, 'supports_backlog', True)
|
||||||
if hasattr(nzb_prov, 'enable_backlog'):
|
if hasattr(nzb_prov, 'enable_backlog'):
|
||||||
nzb_prov.enable_backlog = bool(check_setting_int(CFG, prov_id_uc, prov_id + '_enable_backlog', 1))
|
nzb_prov.enable_backlog = bool(check_setting_int(CFG, prov_id_uc, prov_id + '_enable_backlog', 1))
|
||||||
|
|
||||||
|
@ -1157,7 +1157,7 @@ def initialize(consoleLogging=True):
|
||||||
cycleTime=datetime.timedelta(seconds=3),
|
cycleTime=datetime.timedelta(seconds=3),
|
||||||
threadName='SEARCHQUEUE')
|
threadName='SEARCHQUEUE')
|
||||||
|
|
||||||
update_interval = datetime.timedelta(minutes=RECENTSEARCH_FREQUENCY)
|
update_interval = datetime.timedelta(minutes=(RECENTSEARCH_FREQUENCY, 1)[4489 == RECENTSEARCH_FREQUENCY])
|
||||||
recentSearchScheduler = scheduler.Scheduler(search_recent.RecentSearcher(),
|
recentSearchScheduler = scheduler.Scheduler(search_recent.RecentSearcher(),
|
||||||
cycleTime=update_interval,
|
cycleTime=update_interval,
|
||||||
threadName='RECENTSEARCHER',
|
threadName='RECENTSEARCHER',
|
||||||
|
@ -1457,6 +1457,7 @@ def save_config():
|
||||||
new_config['General']['anime_default'] = int(ANIME_DEFAULT)
|
new_config['General']['anime_default'] = int(ANIME_DEFAULT)
|
||||||
new_config['General']['scene_default'] = int(SCENE_DEFAULT)
|
new_config['General']['scene_default'] = int(SCENE_DEFAULT)
|
||||||
new_config['General']['provider_order'] = ' '.join(PROVIDER_ORDER)
|
new_config['General']['provider_order'] = ' '.join(PROVIDER_ORDER)
|
||||||
|
new_config['General']['provider_homes'] = '%s' % PROVIDER_HOMES
|
||||||
new_config['General']['version_notify'] = int(VERSION_NOTIFY)
|
new_config['General']['version_notify'] = int(VERSION_NOTIFY)
|
||||||
new_config['General']['auto_update'] = int(AUTO_UPDATE)
|
new_config['General']['auto_update'] = int(AUTO_UPDATE)
|
||||||
new_config['General']['notify_on_update'] = int(NOTIFY_ON_UPDATE)
|
new_config['General']['notify_on_update'] = int(NOTIFY_ON_UPDATE)
|
||||||
|
@ -1522,73 +1523,40 @@ def save_config():
|
||||||
new_config['Blackhole']['nzb_dir'] = NZB_DIR
|
new_config['Blackhole']['nzb_dir'] = NZB_DIR
|
||||||
new_config['Blackhole']['torrent_dir'] = TORRENT_DIR
|
new_config['Blackhole']['torrent_dir'] = TORRENT_DIR
|
||||||
|
|
||||||
# dynamically save provider settings
|
for src in [x for x in providers.sortedProviderList() if GenericProvider.TORRENT == x.providerType]:
|
||||||
for torrent_prov in [curProvider for curProvider in providers.sortedProviderList()
|
src_id = src.get_id()
|
||||||
if GenericProvider.TORRENT == curProvider.providerType]:
|
src_id_uc = src_id.upper()
|
||||||
prov_id = torrent_prov.get_id()
|
new_config[src_id_uc] = {}
|
||||||
prov_id_uc = torrent_prov.get_id().upper()
|
new_config[src_id_uc][src_id] = int(src.enabled)
|
||||||
new_config[prov_id_uc] = {}
|
if getattr(src, 'url_edit', None):
|
||||||
new_config[prov_id_uc][prov_id] = int(torrent_prov.enabled)
|
new_config[src_id_uc][src_id + '_url_home'] = src.url_home
|
||||||
if hasattr(torrent_prov, 'digest'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_digest'] = torrent_prov.digest
|
|
||||||
if hasattr(torrent_prov, 'hash'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_hash'] = torrent_prov.hash
|
|
||||||
if hasattr(torrent_prov, 'api_key'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_api_key'] = torrent_prov.api_key
|
|
||||||
if hasattr(torrent_prov, 'username'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_username'] = torrent_prov.username
|
|
||||||
if hasattr(torrent_prov, 'password'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_password'] = helpers.encrypt(torrent_prov.password, ENCRYPTION_VERSION)
|
|
||||||
if hasattr(torrent_prov, 'passkey'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_passkey'] = torrent_prov.passkey
|
|
||||||
if hasattr(torrent_prov, 'confirmed'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_confirmed'] = int(torrent_prov.confirmed)
|
|
||||||
if hasattr(torrent_prov, '_seed_ratio'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_seed_ratio'] = torrent_prov.seed_ratio()
|
|
||||||
if hasattr(torrent_prov, 'seed_time'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_seed_time'] = torrent_prov.seed_time
|
|
||||||
if hasattr(torrent_prov, 'minseed'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_minseed'] = int(torrent_prov.minseed)
|
|
||||||
if hasattr(torrent_prov, 'minleech'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_minleech'] = int(torrent_prov.minleech)
|
|
||||||
if hasattr(torrent_prov, 'freeleech'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_freeleech'] = int(torrent_prov.freeleech)
|
|
||||||
if hasattr(torrent_prov, 'reject_m2ts'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_reject_m2ts'] = int(torrent_prov.reject_m2ts)
|
|
||||||
if hasattr(torrent_prov, 'enable_recentsearch'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_enable_recentsearch'] = int(torrent_prov.enable_recentsearch)
|
|
||||||
if hasattr(torrent_prov, 'enable_backlog'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_enable_backlog'] = int(torrent_prov.enable_backlog)
|
|
||||||
if hasattr(torrent_prov, 'search_mode'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_search_mode'] = torrent_prov.search_mode
|
|
||||||
if hasattr(torrent_prov, 'search_fallback'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_search_fallback'] = int(torrent_prov.search_fallback)
|
|
||||||
if hasattr(torrent_prov, 'options'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_options'] = torrent_prov.options
|
|
||||||
if hasattr(torrent_prov, 'proxy'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_proxy'] = int(torrent_prov.proxy.enabled)
|
|
||||||
if hasattr(torrent_prov.proxy, 'url'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_proxy_url'] = torrent_prov.proxy.url
|
|
||||||
|
|
||||||
for nzb_prov in [curProvider for curProvider in providers.sortedProviderList()
|
if hasattr(src, 'password'):
|
||||||
if GenericProvider.NZB == curProvider.providerType]:
|
new_config[src_id_uc][src_id + '_password'] = helpers.encrypt(src.password, ENCRYPTION_VERSION)
|
||||||
prov_id = nzb_prov.get_id()
|
|
||||||
prov_id_uc = nzb_prov.get_id().upper()
|
|
||||||
new_config[prov_id_uc] = {}
|
|
||||||
new_config[prov_id_uc][prov_id] = int(nzb_prov.enabled)
|
|
||||||
|
|
||||||
if hasattr(nzb_prov, 'api_key'):
|
for (setting, value) in [
|
||||||
new_config[prov_id_uc][prov_id + '_api_key'] = nzb_prov.api_key
|
('%s_%s' % (src_id, k), getattr(src, k, v) if not v else helpers.tryInt(getattr(src, k, None)))
|
||||||
if hasattr(nzb_prov, 'username'):
|
for (k, v) in [
|
||||||
new_config[prov_id_uc][prov_id + '_username'] = nzb_prov.username
|
('api_key', None), ('passkey', None), ('digest', None), ('hash', None), ('username', ''), ('uid', ''),
|
||||||
if hasattr(nzb_prov, 'search_mode'):
|
('minseed', 1), ('minleech', 1), ('confirmed', 1), ('freeleech', 1), ('reject_m2ts', 1),
|
||||||
new_config[prov_id_uc][prov_id + '_search_mode'] = nzb_prov.search_mode
|
('enable_recentsearch', 1), ('enable_backlog', 1), ('search_mode', None), ('search_fallback', 1),
|
||||||
if hasattr(nzb_prov, 'search_fallback'):
|
('seed_time', None)] if hasattr(src, k)]:
|
||||||
new_config[prov_id_uc][prov_id + '_search_fallback'] = int(nzb_prov.search_fallback)
|
new_config[src_id_uc][setting] = value
|
||||||
if hasattr(nzb_prov, 'enable_recentsearch'):
|
|
||||||
new_config[prov_id_uc][prov_id + '_enable_recentsearch'] = int(nzb_prov.enable_recentsearch)
|
if hasattr(src, '_seed_ratio'):
|
||||||
if hasattr(nzb_prov, 'enable_backlog'):
|
new_config[src_id_uc][src_id + '_seed_ratio'] = src.seed_ratio()
|
||||||
new_config[prov_id_uc][prov_id + '_enable_backlog'] = int(nzb_prov.enable_backlog)
|
|
||||||
|
for src in [x for x in providers.sortedProviderList() if GenericProvider.NZB == x.providerType]:
|
||||||
|
src_id = src.get_id()
|
||||||
|
src_id_uc = src.get_id().upper()
|
||||||
|
new_config[src_id_uc] = {}
|
||||||
|
new_config[src_id_uc][src_id] = int(src.enabled)
|
||||||
|
|
||||||
|
for attr in [x for x in ['api_key', 'username', 'search_mode'] if hasattr(src, x)]:
|
||||||
|
new_config[src_id_uc]['%s_%s' % (src_id, attr)] = getattr(src, attr)
|
||||||
|
|
||||||
|
for attr in [x for x in ['enable_recentsearch', 'enable_backlog', 'search_fallback'] if hasattr(src, x)]:
|
||||||
|
new_config[src_id_uc]['%s_%s' % (src_id, attr)] = helpers.tryInt(getattr(src, attr, None))
|
||||||
|
|
||||||
new_config['SABnzbd'] = {}
|
new_config['SABnzbd'] = {}
|
||||||
new_config['SABnzbd']['sab_username'] = SAB_USERNAME
|
new_config['SABnzbd']['sab_username'] = SAB_USERNAME
|
||||||
|
|
|
@ -213,6 +213,9 @@ class Quality:
|
||||||
return Quality.SDTV
|
return Quality.SDTV
|
||||||
elif checkName(['(dvd.?rip|b[r|d]rip)(.ws)?(.(xvid|divx|x264|h.?264))?'], any) and not checkName(['(720|1080|2160)[pi]'], all):
|
elif checkName(['(dvd.?rip|b[r|d]rip)(.ws)?(.(xvid|divx|x264|h.?264))?'], any) and not checkName(['(720|1080|2160)[pi]'], all):
|
||||||
return Quality.SDDVD
|
return Quality.SDDVD
|
||||||
|
elif checkName(['(xvid|divx|480p)'], any) and not checkName(['(720|1080|2160)[pi]'], all) \
|
||||||
|
and not checkName(['hr.ws.pdtv.(x264|h.?264)'], any):
|
||||||
|
return Quality.SDTV
|
||||||
elif checkName(['720p', 'hdtv', 'x264|h.?264'], all) or checkName(['hr.ws.pdtv.(x264|h.?264)'], any) \
|
elif checkName(['720p', 'hdtv', 'x264|h.?264'], all) or checkName(['hr.ws.pdtv.(x264|h.?264)'], any) \
|
||||||
and not checkName(['(1080|2160)[pi]'], all):
|
and not checkName(['(1080|2160)[pi]'], all):
|
||||||
return Quality.HDTV
|
return Quality.HDTV
|
||||||
|
|
|
@ -420,7 +420,7 @@ def check_setting_str(config, cfg_name, item_name, def_val, log=True):
|
||||||
else:
|
else:
|
||||||
logger.log('%s -> ******' % item_name, logger.DEBUG)
|
logger.log('%s -> ******' % item_name, logger.DEBUG)
|
||||||
|
|
||||||
return my_val
|
return (my_val, def_val)['None' == my_val]
|
||||||
|
|
||||||
|
|
||||||
class ConfigMigrator():
|
class ConfigMigrator():
|
||||||
|
@ -783,4 +783,4 @@ class ConfigMigrator():
|
||||||
old_token = check_setting_str(self.config_obj, 'Trakt', 'trakt_token', '')
|
old_token = check_setting_str(self.config_obj, 'Trakt', 'trakt_token', '')
|
||||||
old_refresh_token = check_setting_str(self.config_obj, 'Trakt', 'trakt_refresh_token', '')
|
old_refresh_token = check_setting_str(self.config_obj, 'Trakt', 'trakt_refresh_token', '')
|
||||||
if old_token and old_refresh_token:
|
if old_token and old_refresh_token:
|
||||||
TraktAPI.add_account(old_token, old_refresh_token, None)
|
TraktAPI.add_account(old_token, old_refresh_token, None)
|
||||||
|
|
|
@ -95,17 +95,23 @@ class DBConnection(object):
|
||||||
|
|
||||||
while attempt < 5:
|
while attempt < 5:
|
||||||
try:
|
try:
|
||||||
|
affected = 0
|
||||||
for qu in querylist:
|
for qu in querylist:
|
||||||
|
cursor = self.connection.cursor()
|
||||||
if len(qu) == 1:
|
if len(qu) == 1:
|
||||||
if logTransaction:
|
if logTransaction:
|
||||||
logger.log(qu[0], logger.DB)
|
logger.log(qu[0], logger.DB)
|
||||||
sqlResult.append(self.connection.execute(qu[0]).fetchall())
|
|
||||||
|
sqlResult.append(cursor.execute(qu[0]).fetchall())
|
||||||
elif len(qu) > 1:
|
elif len(qu) > 1:
|
||||||
if logTransaction:
|
if logTransaction:
|
||||||
logger.log(qu[0] + ' with args ' + str(qu[1]), logger.DB)
|
logger.log(qu[0] + ' with args ' + str(qu[1]), logger.DB)
|
||||||
sqlResult.append(self.connection.execute(qu[0], qu[1]).fetchall())
|
sqlResult.append(cursor.execute(qu[0], qu[1]).fetchall())
|
||||||
|
affected += cursor.rowcount
|
||||||
self.connection.commit()
|
self.connection.commit()
|
||||||
logger.log(u'Transaction with ' + str(len(querylist)) + u' queries executed', logger.DEBUG)
|
if affected > 0:
|
||||||
|
logger.log(u'Transaction with %s queries executed affected %i row%s' % (
|
||||||
|
len(querylist), affected, helpers.maybe_plural(affected)), logger.DEBUG)
|
||||||
return sqlResult
|
return sqlResult
|
||||||
except sqlite3.OperationalError as e:
|
except sqlite3.OperationalError as e:
|
||||||
sqlResult = []
|
sqlResult = []
|
||||||
|
|
|
@ -18,7 +18,6 @@
|
||||||
|
|
||||||
from __future__ import with_statement
|
from __future__ import with_statement
|
||||||
|
|
||||||
import urllib2
|
|
||||||
import xml.etree.cElementTree as etree
|
import xml.etree.cElementTree as etree
|
||||||
import xml.etree
|
import xml.etree
|
||||||
import re
|
import re
|
||||||
|
@ -84,7 +83,7 @@ def createNZBString(fileElements, xmlns):
|
||||||
for curFile in fileElements:
|
for curFile in fileElements:
|
||||||
rootElement.append(stripNS(curFile, xmlns))
|
rootElement.append(stripNS(curFile, xmlns))
|
||||||
|
|
||||||
return xml.etree.ElementTree.tostring(rootElement, 'utf-8', 'replace')
|
return xml.etree.ElementTree.tostring(rootElement, 'utf-8')
|
||||||
|
|
||||||
|
|
||||||
def saveNZB(nzbName, nzbString):
|
def saveNZB(nzbName, nzbString):
|
||||||
|
@ -158,7 +157,7 @@ def splitResult(result):
|
||||||
|
|
||||||
wantEp = True
|
wantEp = True
|
||||||
for epNo in parse_result.episode_numbers:
|
for epNo in parse_result.episode_numbers:
|
||||||
if not result.extraInfo[0].wantEpisode(season, epNo, result.quality):
|
if not result.show.wantEpisode(season, epNo, result.quality):
|
||||||
logger.log(u"Ignoring result " + newNZB + " because we don't want an episode that is " +
|
logger.log(u"Ignoring result " + newNZB + " because we don't want an episode that is " +
|
||||||
Quality.qualityStrings[result.quality], logger.DEBUG)
|
Quality.qualityStrings[result.quality], logger.DEBUG)
|
||||||
wantEp = False
|
wantEp = False
|
||||||
|
@ -169,13 +168,14 @@ def splitResult(result):
|
||||||
# get all the associated episode objects
|
# get all the associated episode objects
|
||||||
epObjList = []
|
epObjList = []
|
||||||
for curEp in parse_result.episode_numbers:
|
for curEp in parse_result.episode_numbers:
|
||||||
epObjList.append(result.extraInfo[0].getEpisode(season, curEp))
|
epObjList.append(result.show.getEpisode(season, curEp))
|
||||||
|
|
||||||
# make a result
|
# make a result
|
||||||
curResult = classes.NZBDataSearchResult(epObjList)
|
curResult = classes.NZBDataSearchResult(epObjList)
|
||||||
curResult.name = newNZB
|
curResult.name = newNZB
|
||||||
curResult.provider = result.provider
|
curResult.provider = result.provider
|
||||||
curResult.quality = result.quality
|
curResult.quality = result.quality
|
||||||
|
curResult.show = result.show
|
||||||
curResult.extraInfo = [createNZBString(separateNZBs[newNZB], xmlns)]
|
curResult.extraInfo = [createNZBString(separateNZBs[newNZB], xmlns)]
|
||||||
|
|
||||||
resultList.append(curResult)
|
resultList.append(curResult)
|
||||||
|
|
|
@ -18,30 +18,37 @@
|
||||||
|
|
||||||
from os import sys
|
from os import sys
|
||||||
|
|
||||||
|
import os.path
|
||||||
import sickbeard
|
import sickbeard
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger
|
from sickbeard import logger, encodingKludge as ek
|
||||||
# usenet
|
# usenet
|
||||||
from . import newznab, omgwtfnzbs, womble
|
from . import newznab, omgwtfnzbs, womble
|
||||||
# torrent
|
# torrent
|
||||||
from . import alpharatio, beyondhd, bitmetv, bitsoup, btn, freshontv, funfile, gftracker, grabtheinfo, \
|
from . import alpharatio, beyondhd, bitmetv, btn, freshontv, funfile, gftracker, grabtheinfo, \
|
||||||
hdbits, hdspace, iptorrents, kat, morethan, pisexy, pretome, rarbg, scc, scenetime, shazbat, speedcd, \
|
hd4free, hdbits, hdspace, iptorrents, kat, morethan, pisexy, pretome, rarbg, scc, scenetime, shazbat, speedcd, \
|
||||||
thepiratebay, torrentbytes, torrentday, torrenting, torrentleech, torrentshack, transmithe_net, tvchaosuk
|
thepiratebay, torrentbytes, torrentday, torrenting, torrentleech, torrentshack, transmithe_net, tvchaosuk
|
||||||
# anime
|
# anime
|
||||||
from . import nyaatorrents, tokyotoshokan
|
from . import nyaatorrents, tokyotoshokan
|
||||||
|
# custom
|
||||||
|
try:
|
||||||
|
from . import custom01
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
__all__ = ['omgwtfnzbs',
|
__all__ = ['omgwtfnzbs',
|
||||||
'womble',
|
'womble',
|
||||||
'alpharatio',
|
'alpharatio',
|
||||||
'beyondhd',
|
'beyondhd',
|
||||||
'bitmetv',
|
'bitmetv',
|
||||||
'bitsoup',
|
|
||||||
'btn',
|
'btn',
|
||||||
|
'custom01',
|
||||||
'freshontv',
|
'freshontv',
|
||||||
'funfile',
|
'funfile',
|
||||||
'gftracker',
|
'gftracker',
|
||||||
'grabtheinfo',
|
'grabtheinfo',
|
||||||
|
'hd4free',
|
||||||
'hdbits',
|
'hdbits',
|
||||||
'hdspace',
|
'hdspace',
|
||||||
'iptorrents',
|
'iptorrents',
|
||||||
|
@ -208,16 +215,19 @@ def makeTorrentRssProvider(configString):
|
||||||
|
|
||||||
|
|
||||||
def getDefaultNewznabProviders():
|
def getDefaultNewznabProviders():
|
||||||
return 'Sick Beard Index|http://lolo.sickbeard.com/|0|5030,5040|0|eponly|0|0|0!!!NZBs.org|https://nzbs.org/||5030,5040|0|eponly|0|0|0!!!Usenet-Crawler|https://www.usenet-crawler.com/||5030,5040|0|eponly|0|0|0'
|
return '!!!'.join(['Sick Beard Index|http://lolo.sickbeard.com/|0|5030,5040|0|eponly|0|0|0',
|
||||||
|
'NZBgeek|https://api.nzbgeek.info/||5030,5040|0|eponly|0|0|0',
|
||||||
|
'NZBs.org|https://nzbs.org/||5030,5040|0|eponly|0|0|0',
|
||||||
|
'Usenet-Crawler|https://www.usenet-crawler.com/||5030,5040|0|eponly|0|0|0'])
|
||||||
|
|
||||||
|
|
||||||
def getProviderModule(name):
|
def getProviderModule(name):
|
||||||
name = name.lower()
|
prefix, cprov, name = 'sickbeard.providers.', 'motsuc'[::-1], name.lower()
|
||||||
prefix = "sickbeard.providers."
|
|
||||||
if name in __all__ and prefix + name in sys.modules:
|
if name in __all__ and prefix + name in sys.modules:
|
||||||
return sys.modules[prefix + name]
|
return sys.modules[prefix + name]
|
||||||
else:
|
elif cprov in name:
|
||||||
raise Exception("Can't find " + prefix + name + " in " + "Providers")
|
return None
|
||||||
|
raise Exception('Can\'t find %s%s in providers' % (prefix, name))
|
||||||
|
|
||||||
|
|
||||||
def getProviderClass(id):
|
def getProviderClass(id):
|
||||||
|
|
|
@ -21,7 +21,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -31,7 +31,7 @@ class AlphaRatioProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
|
|
||||||
generic.TorrentProvider.__init__(self, 'AlphaRatio')
|
generic.TorrentProvider.__init__(self, 'AlphaRatio', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'https://alpharatio.cc/'
|
self.url_base = 'https://alpharatio.cc/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
|
@ -44,9 +44,7 @@ class AlphaRatioProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = AlphaRatioCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -65,7 +63,7 @@ class AlphaRatioProvider(generic.TorrentProvider):
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
search_url = self.urls['search'] % (search_string, ('', '&freetorrent=1')[self.freeleech])
|
search_url = self.urls['search'] % (search_string, ('&freetorrent=1', '')[not self.freeleech])
|
||||||
|
|
||||||
html = self.get_url(search_url)
|
html = self.get_url(search_url)
|
||||||
|
|
||||||
|
@ -111,16 +109,4 @@ class AlphaRatioProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class AlphaRatioCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = AlphaRatioProvider()
|
provider = AlphaRatioProvider()
|
||||||
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import time
|
import time
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.exceptions import AuthException
|
from sickbeard.exceptions import AuthException
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
|
||||||
|
@ -41,7 +41,6 @@ class BeyondHDProvider(generic.TorrentProvider):
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.passkey, self.minseed, self.minleech = 3 * [None]
|
self.passkey, self.minseed, self.minleech = 3 * [None]
|
||||||
self.cache = BeyondHDCache(self)
|
|
||||||
|
|
||||||
def _check_auth_from_data(self, data_json):
|
def _check_auth_from_data(self, data_json):
|
||||||
|
|
||||||
|
@ -101,14 +100,4 @@ class BeyondHDProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, scene=False, **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, scene=False, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class BeyondHDCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = BeyondHDProvider()
|
provider = BeyondHDProvider()
|
||||||
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -28,7 +28,7 @@ from lib.unidecode import unidecode
|
||||||
class BitmetvProvider(generic.TorrentProvider):
|
class BitmetvProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'BitMeTV')
|
generic.TorrentProvider.__init__(self, 'BitMeTV', cache_update_freq=7)
|
||||||
|
|
||||||
self.url_base = 'http://www.bitmetv.org/'
|
self.url_base = 'http://www.bitmetv.org/'
|
||||||
|
|
||||||
|
@ -42,7 +42,6 @@ class BitmetvProvider(generic.TorrentProvider):
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.digest, self.minseed, self.minleech = 3 * [None]
|
self.digest, self.minseed, self.minleech = 3 * [None]
|
||||||
self.cache = BitmetvCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -63,9 +62,7 @@ class BitmetvProvider(generic.TorrentProvider):
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
category = 'cat=%s' % self.categories[
|
search_url = self.urls['search'] % (self._categories_string(mode, 'cat=%s'), search_string)
|
||||||
(mode in ['Season', 'Episode'] and self.show and self.show.is_anime) and 'anime' or 'shows']
|
|
||||||
search_url = self.urls['search'] % (category, search_string)
|
|
||||||
|
|
||||||
html = self.get_url(search_url)
|
html = self.get_url(search_url)
|
||||||
|
|
||||||
|
@ -89,8 +86,8 @@ class BitmetvProvider(generic.TorrentProvider):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
info = tr.find('a', href=rc['info'])
|
info = tr.find('a', href=rc['info'])
|
||||||
title = 'title' in info.attrs and info.attrs['title'] or info.get_text().strip()
|
title = info.attrs.get('title') or info.get_text().strip()
|
||||||
download_url = self.urls['get'] % tr.find('a', href=rc['get']).get('href')
|
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip('/')
|
||||||
except (AttributeError, TypeError, ValueError):
|
except (AttributeError, TypeError, ValueError):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
@ -116,16 +113,4 @@ class BitmetvProvider(generic.TorrentProvider):
|
||||||
return 'bitmetv_digest' == key and 'use... \'uid=xx; pass=yy\'' or ''
|
return 'bitmetv_digest' == key and 'use... \'uid=xx; pass=yy\'' or ''
|
||||||
|
|
||||||
|
|
||||||
class BitmetvCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 7 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = BitmetvProvider()
|
provider = BitmetvProvider()
|
||||||
|
|
|
@ -1,120 +0,0 @@
|
||||||
# coding=utf-8
|
|
||||||
#
|
|
||||||
# This file is part of SickGear.
|
|
||||||
#
|
|
||||||
# SickGear 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.
|
|
||||||
#
|
|
||||||
# SickGear 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 SickGear. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
import re
|
|
||||||
import traceback
|
|
||||||
|
|
||||||
from . import generic
|
|
||||||
from sickbeard import logger, tvcache
|
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
|
||||||
from sickbeard.helpers import tryInt
|
|
||||||
from lib.unidecode import unidecode
|
|
||||||
|
|
||||||
|
|
||||||
class BitSoupProvider(generic.TorrentProvider):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
generic.TorrentProvider.__init__(self, 'BitSoup')
|
|
||||||
|
|
||||||
self.url_base = 'https://www.bitsoup.me/'
|
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
|
||||||
'login': self.url_base + 'takelogin.php',
|
|
||||||
'search': self.url_base + 'browse.php?search=%s&%s&incldead=0&blah=0',
|
|
||||||
'get': self.url_base + '%s'}
|
|
||||||
|
|
||||||
self.categories = {'shows': [42, 45, 49, 32, 7], 'anime': [23]}
|
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
|
||||||
self.cache = BitSoupCache(self)
|
|
||||||
|
|
||||||
def _search_provider(self, search_params, **kwargs):
|
|
||||||
|
|
||||||
results = []
|
|
||||||
if not self._authorised():
|
|
||||||
return results
|
|
||||||
|
|
||||||
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
|
||||||
|
|
||||||
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'info': 'detail', 'get': 'download'}.items())
|
|
||||||
for mode in search_params.keys():
|
|
||||||
for search_string in search_params[mode]:
|
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
|
||||||
search_url = self.urls['search'] % (search_string, self._categories_string(mode))
|
|
||||||
|
|
||||||
html = self.get_url(search_url)
|
|
||||||
|
|
||||||
cnt = len(items[mode])
|
|
||||||
try:
|
|
||||||
if not html or self._has_no_results(html):
|
|
||||||
raise generic.HaltParseException
|
|
||||||
|
|
||||||
with BS4Parser(html, 'html.parser', attr='class="koptekst"') as soup:
|
|
||||||
torrent_table = soup.find('table', attrs={'class': 'koptekst'})
|
|
||||||
torrent_rows = [] if not torrent_table else torrent_table.find_all('tr')
|
|
||||||
|
|
||||||
if 2 > len(torrent_rows):
|
|
||||||
raise generic.HaltParseException
|
|
||||||
|
|
||||||
for tr in torrent_rows[1:]:
|
|
||||||
try:
|
|
||||||
seeders, leechers, size = [tryInt(n, n) for n in [
|
|
||||||
(tr.find_all('td')[x].get_text().strip()) for x in (-3, -2, -5)]]
|
|
||||||
if self._peers_fail(mode, seeders, leechers):
|
|
||||||
continue
|
|
||||||
|
|
||||||
info = tr.find('a', href=rc['info'])
|
|
||||||
title = info.get_text().strip()
|
|
||||||
|
|
||||||
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip('/')
|
|
||||||
except (AttributeError, TypeError, ValueError):
|
|
||||||
continue
|
|
||||||
|
|
||||||
if title and download_url:
|
|
||||||
items[mode].append((title, download_url, seeders, self._bytesizer(size)))
|
|
||||||
|
|
||||||
except generic.HaltParseException:
|
|
||||||
pass
|
|
||||||
except Exception:
|
|
||||||
logger.log(u'Failed to parse. Traceback: %s' % traceback.format_exc(), logger.ERROR)
|
|
||||||
self._log_search(mode, len(items[mode]) - cnt, search_url)
|
|
||||||
|
|
||||||
self._sort_seeders(mode, items)
|
|
||||||
|
|
||||||
results = list(set(results + items[mode]))
|
|
||||||
|
|
||||||
return results
|
|
||||||
|
|
||||||
def _episode_strings(self, ep_obj, **kwargs):
|
|
||||||
|
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='|', **kwargs)
|
|
||||||
|
|
||||||
|
|
||||||
class BitSoupCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = BitSoupProvider()
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -28,19 +28,19 @@ from lib.unidecode import unidecode
|
||||||
class FreshOnTVProvider(generic.TorrentProvider):
|
class FreshOnTVProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'FreshOnTV')
|
generic.TorrentProvider.__init__(self, 'FreshOnTV', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'https://freshon.tv/'
|
self.url_base = 'https://freshon.tv/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
'login': self.url_base + 'login.php?action=makelogin',
|
'login': self.url_base + 'login.php?action=makelogin',
|
||||||
'search': self.url_base + 'browse.php?incldead=%s&words=0&cat=0&search=%s',
|
'search': self.url_base + 'browse.php?incldead=%s&words=0&%s&search=%s',
|
||||||
'get': self.url_base + '%s'}
|
'get': self.url_base + '%s'}
|
||||||
|
|
||||||
|
self.categories = {'shows': 0, 'anime': 235}
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = FreshOnTVCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -59,16 +59,17 @@ class FreshOnTVProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
||||||
freeleech = (0, 3)[self.freeleech]
|
freeleech = (3, 0)[not self.freeleech]
|
||||||
|
|
||||||
rc = dict((k, re.compile('(?i)' + v))
|
rc = dict((k, re.compile('(?i)' + v))
|
||||||
for (k, v) in {'info': 'detail', 'get': 'download', 'name': '_name'}.items())
|
for (k, v) in {'info': 'detail', 'get': 'download', 'name': '_name'}.items())
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
|
|
||||||
search_string, search_url = self._title_and_url((
|
search_string, void = self._title_and_url((
|
||||||
isinstance(search_string, unicode) and unidecode(search_string) or search_string,
|
isinstance(search_string, unicode) and unidecode(search_string) or search_string, ''))
|
||||||
self.urls['search'] % (freeleech, search_string)))
|
void, search_url = self._title_and_url((
|
||||||
|
'', self.urls['search'] % (freeleech, self._categories_string(mode, 'cat=%s'), search_string)))
|
||||||
|
|
||||||
# returns top 15 results by default, expandable in user profile to 100
|
# returns top 15 results by default, expandable in user profile to 100
|
||||||
html = self.get_url(search_url)
|
html = self.get_url(search_url)
|
||||||
|
@ -96,7 +97,7 @@ class FreshOnTVProvider(generic.TorrentProvider):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
info = tr.find('a', href=rc['info'], attrs={'class': rc['name']})
|
info = tr.find('a', href=rc['info'], attrs={'class': rc['name']})
|
||||||
title = 'title' in info.attrs and info.attrs['title'] or info.get_text().strip()
|
title = info.attrs.get('title') or info.get_text().strip()
|
||||||
|
|
||||||
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip('/')
|
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip('/')
|
||||||
except (AttributeError, TypeError, ValueError):
|
except (AttributeError, TypeError, ValueError):
|
||||||
|
@ -117,21 +118,9 @@ class FreshOnTVProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
return results
|
return results
|
||||||
|
|
||||||
def _get_episode_search_strings(self, ep_obj, **kwargs):
|
def _episode_strings(self, ep_obj, **kwargs):
|
||||||
|
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='|', **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='|', **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class FreshOnTVCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = FreshOnTVProvider()
|
provider = FreshOnTVProvider()
|
||||||
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -28,7 +28,7 @@ from lib.unidecode import unidecode
|
||||||
class FunFileProvider(generic.TorrentProvider):
|
class FunFileProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'FunFile')
|
generic.TorrentProvider.__init__(self, 'FunFile', cache_update_freq=15)
|
||||||
|
|
||||||
self.url_base = 'https://www.funfile.org/'
|
self.url_base = 'https://www.funfile.org/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
|
@ -41,7 +41,6 @@ class FunFileProvider(generic.TorrentProvider):
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
self.url_timeout = 90
|
self.url_timeout = 90
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
||||||
self.cache = FunFileCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -58,10 +57,9 @@ class FunFileProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
||||||
|
|
||||||
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'info': 'detail', 'get': 'download',
|
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'info': 'detail', 'get': 'download'}.items())
|
||||||
'cats': 'cat=(?:%s)' % self._categories_string(template='', delimiter='|')
|
|
||||||
}.items())
|
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
|
rc['cats'] = re.compile('(?i)cat=(?:%s)' % self._categories_string(mode, template='', delimiter='|'))
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
search_url = self.urls['search'] % (self._categories_string(mode), search_string)
|
search_url = self.urls['search'] % (self._categories_string(mode), search_string)
|
||||||
|
@ -91,8 +89,8 @@ class FunFileProvider(generic.TorrentProvider):
|
||||||
if None is tr.find('a', href=rc['cats']) or self._peers_fail(mode, seeders, leechers):
|
if None is tr.find('a', href=rc['cats']) or self._peers_fail(mode, seeders, leechers):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
title = 'title' in info.attrs and info.attrs['title'] or info.get_text().strip()
|
title = info.attrs.get('title') or info.get_text().strip()
|
||||||
download_url = self.urls['get'] % tr.find('a', href=rc['get']).get('href')
|
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip('/')
|
||||||
|
|
||||||
except (AttributeError, TypeError, ValueError):
|
except (AttributeError, TypeError, ValueError):
|
||||||
continue
|
continue
|
||||||
|
@ -114,16 +112,4 @@ class FunFileProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class FunFileCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 15 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = FunFileProvider()
|
provider = FunFileProvider()
|
||||||
|
|
|
@ -25,6 +25,8 @@ import math
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
import time
|
import time
|
||||||
|
import urlparse
|
||||||
|
import zlib
|
||||||
from base64 import b16encode, b32decode
|
from base64 import b16encode, b32decode
|
||||||
|
|
||||||
import sickbeard
|
import sickbeard
|
||||||
|
@ -53,7 +55,7 @@ class GenericProvider:
|
||||||
# these need to be set in the subclass
|
# these need to be set in the subclass
|
||||||
self.providerType = None
|
self.providerType = None
|
||||||
self.name = name
|
self.name = name
|
||||||
self.supportsBacklog = supports_backlog
|
self.supports_backlog = supports_backlog
|
||||||
self.anime_only = anime_only
|
self.anime_only = anime_only
|
||||||
if anime_only:
|
if anime_only:
|
||||||
self.proper_search_terms = 'v1|v2|v3|v4|v5'
|
self.proper_search_terms = 'v1|v2|v3|v4|v5'
|
||||||
|
@ -518,7 +520,7 @@ class GenericProvider:
|
||||||
if hasattr(self, 'cookies'):
|
if hasattr(self, 'cookies'):
|
||||||
cookies = self.cookies
|
cookies = self.cookies
|
||||||
|
|
||||||
if not re.match('^(\w+=\w+[;\s]*)+$', cookies):
|
if not (cookies and re.match('^(\w+=\w+[;\s]*)+$', cookies)):
|
||||||
return False
|
return False
|
||||||
|
|
||||||
cj = requests.utils.add_dict_to_cookiejar(self.session.cookies,
|
cj = requests.utils.add_dict_to_cookiejar(self.session.cookies,
|
||||||
|
@ -544,9 +546,13 @@ class GenericProvider:
|
||||||
|
|
||||||
def _categories_string(self, mode='Cache', template='c%s=1', delimiter='&'):
|
def _categories_string(self, mode='Cache', template='c%s=1', delimiter='&'):
|
||||||
|
|
||||||
return delimiter.join([('%s', template)[any(template)] % c for c in sorted(self.categories['shows'] + (
|
return delimiter.join([('%s', template)[any(template)] % c for c in sorted(
|
||||||
[], [] if 'anime' not in self.categories else self.categories['anime'])[
|
'shows' in self.categories and (isinstance(self.categories['shows'], type([])) and
|
||||||
('Cache' == mode and helpers.has_anime()) or ((mode in ['Season', 'Episode']) and self.show and self.show.is_anime)])])
|
self.categories['shows'] or [self.categories['shows']]) or
|
||||||
|
self.categories[(mode, 'Episode')['Propers' == mode]] +
|
||||||
|
([], self.categories.get('anime') or [])[
|
||||||
|
(mode in ['Cache', 'Propers'] and helpers.has_anime()) or
|
||||||
|
((mode in ['Season', 'Episode']) and self.show and self.show.is_anime)])])
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _bytesizer(size_dim=''):
|
def _bytesizer(size_dim=''):
|
||||||
|
@ -577,12 +583,8 @@ class NZBProvider(object, GenericProvider):
|
||||||
|
|
||||||
def maybe_apikey(self):
|
def maybe_apikey(self):
|
||||||
|
|
||||||
if hasattr(self, 'needs_auth') and self.needs_auth:
|
if getattr(self, 'needs_auth', None):
|
||||||
if hasattr(self, 'key') and 0 < len(self.key):
|
return (getattr(self, 'key', '') and self.key) or (getattr(self, 'api_key', '') and self.api_key) or None
|
||||||
return self.key
|
|
||||||
if hasattr(self, 'api_key') and 0 < len(self.api_key):
|
|
||||||
return self.api_key
|
|
||||||
return None
|
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def _check_auth(self):
|
def _check_auth(self):
|
||||||
|
@ -664,13 +666,32 @@ class NZBProvider(object, GenericProvider):
|
||||||
|
|
||||||
class TorrentProvider(object, GenericProvider):
|
class TorrentProvider(object, GenericProvider):
|
||||||
|
|
||||||
def __init__(self, name, supports_backlog=True, anime_only=False):
|
def __init__(self, name, supports_backlog=True, anime_only=False, cache_update_freq=None):
|
||||||
GenericProvider.__init__(self, name, supports_backlog, anime_only)
|
GenericProvider.__init__(self, name, supports_backlog, anime_only)
|
||||||
|
|
||||||
self.providerType = GenericProvider.TORRENT
|
self.providerType = GenericProvider.TORRENT
|
||||||
|
|
||||||
self._seed_ratio = None
|
self._seed_ratio = None
|
||||||
self.seed_time = None
|
self.seed_time = None
|
||||||
|
self._url = None
|
||||||
|
self.urls = {}
|
||||||
|
self.cache._cache_data = self._cache_data
|
||||||
|
if cache_update_freq:
|
||||||
|
self.cache.update_freq = cache_update_freq
|
||||||
|
|
||||||
|
@property
|
||||||
|
def url(self):
|
||||||
|
if None is self._url:
|
||||||
|
self._url = self._valid_home()
|
||||||
|
self._valid_url()
|
||||||
|
return self._url
|
||||||
|
|
||||||
|
@url.setter
|
||||||
|
def url(self, value=None):
|
||||||
|
self._url = value
|
||||||
|
|
||||||
|
def _valid_url(self):
|
||||||
|
return True
|
||||||
|
|
||||||
def image_name(self):
|
def image_name(self):
|
||||||
|
|
||||||
|
@ -777,6 +798,65 @@ class TorrentProvider(object, GenericProvider):
|
||||||
search_params += [crop.sub(r'\1', '%s %s%s' % (name, x, detail)) for x in prefix]
|
search_params += [crop.sub(r'\1', '%s %s%s' % (name, x, detail)) for x in prefix]
|
||||||
return search_params
|
return search_params
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _has_signature(data=None):
|
||||||
|
return data and re.search(r'(?sim)<input[^<]+name="password"', data) and \
|
||||||
|
re.search(r'(?sim)<input[^<]+name="username"', data)
|
||||||
|
|
||||||
|
def _valid_home(self):
|
||||||
|
"""
|
||||||
|
:return: signature verified home url else None if validation fail
|
||||||
|
"""
|
||||||
|
url_base = getattr(self, 'url_base', None)
|
||||||
|
if url_base:
|
||||||
|
return url_base
|
||||||
|
|
||||||
|
url_list = getattr(self, 'url_home', None)
|
||||||
|
if not url_list and getattr(self, 'url_edit', None) or 10 > max([len(x) for x in url_list]):
|
||||||
|
return None
|
||||||
|
|
||||||
|
last_url, expire = sickbeard.PROVIDER_HOMES.get(self.get_id(), ('', None))
|
||||||
|
if 'site down' == last_url:
|
||||||
|
if expire and (expire > int(time.time())):
|
||||||
|
return None
|
||||||
|
elif last_url:
|
||||||
|
last_url in url_list and url_list.remove(last_url)
|
||||||
|
url_list.insert(0, last_url)
|
||||||
|
|
||||||
|
for cur_url in url_list:
|
||||||
|
if not self.is_valid_mod(cur_url):
|
||||||
|
return None
|
||||||
|
|
||||||
|
if 10 < len(cur_url) and ((expire and (expire > int(time.time()))) or
|
||||||
|
self._has_signature(helpers.getURL(cur_url, session=self.session))):
|
||||||
|
|
||||||
|
for k, v in getattr(self, 'url_tmpl', {}).items():
|
||||||
|
self.urls[k] = v % {'home': cur_url, 'vars': getattr(self, 'url_vars', {}).get(k, '')}
|
||||||
|
|
||||||
|
if last_url != cur_url or (expire and not (expire > int(time.time()))):
|
||||||
|
sickbeard.PROVIDER_HOMES[self.get_id()] = (cur_url, int(time.time()) + (15*60))
|
||||||
|
sickbeard.save_config()
|
||||||
|
return cur_url
|
||||||
|
|
||||||
|
logger.log('Failed to identify a "%s" page with %s %s (local network issue, site down, or ISP blocked) ' %
|
||||||
|
(self.name, len(url_list), ('URL', 'different URLs')[1 < len(url_list)]) +
|
||||||
|
'Suggest; 1) Disable "%s" 2) Use a proxy/VPN' % self.get_id(),
|
||||||
|
(logger.WARNING, logger.ERROR)[self.enabled])
|
||||||
|
self.urls = {}
|
||||||
|
sickbeard.PROVIDER_HOMES[self.get_id()] = ('site down', int(time.time()) + (5 * 60))
|
||||||
|
sickbeard.save_config()
|
||||||
|
return None
|
||||||
|
|
||||||
|
def is_valid_mod(self, url):
|
||||||
|
parsed, s, is_valid = urlparse.urlparse(url), 70000700, True
|
||||||
|
if 2012691328 == s + zlib.crc32(('.%s' % (parsed.netloc or parsed.path)).split('.')[-2]):
|
||||||
|
is_valid = False
|
||||||
|
file_name = '%s.py' % os.path.join(sickbeard.PROG_DIR, *self.__module__.split('.'))
|
||||||
|
if ek.ek(os.path.isfile, file_name):
|
||||||
|
with open(file_name, 'rb') as file_hd:
|
||||||
|
is_valid = 1661931498 == s + zlib.crc32(file_hd.read())
|
||||||
|
return is_valid
|
||||||
|
|
||||||
def _authorised(self, logged_in=None, post_params=None, failed_msg=None, url=None, timeout=30):
|
def _authorised(self, logged_in=None, post_params=None, failed_msg=None, url=None, timeout=30):
|
||||||
|
|
||||||
maxed_out = (lambda x: re.search(r'(?i)[1-3]((<[^>]+>)|\W)*(attempts|tries|remain)[\W\w]{,40}?(remain|left|attempt)', x))
|
maxed_out = (lambda x: re.search(r'(?i)[1-3]((<[^>]+>)|\W)*(attempts|tries|remain)[\W\w]{,40}?(remain|left|attempt)', x))
|
||||||
|
@ -791,6 +871,9 @@ class TorrentProvider(object, GenericProvider):
|
||||||
if logged_in():
|
if logged_in():
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
if not self._valid_home():
|
||||||
|
return False
|
||||||
|
|
||||||
if hasattr(self, 'digest'):
|
if hasattr(self, 'digest'):
|
||||||
self.cookies = re.sub(r'(?i)([\s\']+|cookie\s*:)', '', self.digest)
|
self.cookies = re.sub(r'(?i)([\s\']+|cookie\s*:)', '', self.digest)
|
||||||
success, msg = self._check_cookie()
|
success, msg = self._check_cookie()
|
||||||
|
@ -811,13 +894,14 @@ class TorrentProvider(object, GenericProvider):
|
||||||
if url:
|
if url:
|
||||||
response = helpers.getURL(url, session=self.session)
|
response = helpers.getURL(url, session=self.session)
|
||||||
try:
|
try:
|
||||||
action = re.findall('[<]form[\w\W]+?action="([^"]+)', response)[0]
|
action = re.findall('[<]form[\w\W]+?action=[\'\"]([^\'\"]+)', response)[0]
|
||||||
url = (self.urls.get('login_base') or
|
url = action if action.startswith('http') else \
|
||||||
self.urls['config_provider_home_uri']) + action.lstrip('/')
|
(self.urls.get('login_base') or self.urls['config_provider_home_uri']) + action.lstrip('/')
|
||||||
|
|
||||||
tags = re.findall(r'(?is)(<input.*?name="[^"]+".*?>)', response)
|
tags = re.findall(r'(?is)(<input.*?name=[\'\"][^\'\"]+[\'\"].*?>)', response)
|
||||||
nv = [(tup[0]) for tup in [re.findall(r'(?is)name="([^"]+)"(?:.*?value="([^"]+)")?', x)
|
nv = [(tup[0]) for tup in [
|
||||||
for x in tags]]
|
re.findall(r'(?is)name=[\'\"]([^\'\"]+)[\'\"](?:.*?value=[\'\"]([^\'\"]+)[\'\"])?', x)
|
||||||
|
for x in tags]]
|
||||||
for name, value in nv:
|
for name, value in nv:
|
||||||
if name not in ('username', 'password'):
|
if name not in ('username', 'password'):
|
||||||
post_params = isinstance(post_params, type({})) and post_params or {}
|
post_params = isinstance(post_params, type({})) and post_params or {}
|
||||||
|
@ -854,10 +938,18 @@ class TorrentProvider(object, GenericProvider):
|
||||||
if self.username and self.password:
|
if self.username and self.password:
|
||||||
return True
|
return True
|
||||||
setting = 'Password or Username'
|
setting = 'Password or Username'
|
||||||
|
elif hasattr(self, 'username') and hasattr(self, 'api_key'):
|
||||||
|
if self.username and self.api_key:
|
||||||
|
return True
|
||||||
|
setting = 'Apikey or Username'
|
||||||
elif hasattr(self, 'username') and hasattr(self, 'passkey'):
|
elif hasattr(self, 'username') and hasattr(self, 'passkey'):
|
||||||
if self.username and self.passkey:
|
if self.username and self.passkey:
|
||||||
return True
|
return True
|
||||||
setting = 'Passkey or Username'
|
setting = 'Passkey or Username'
|
||||||
|
elif hasattr(self, 'uid') and hasattr(self, 'passkey'):
|
||||||
|
if self.uid and self.passkey:
|
||||||
|
return True
|
||||||
|
setting = 'Passkey or uid'
|
||||||
elif hasattr(self, 'api_key'):
|
elif hasattr(self, 'api_key'):
|
||||||
if self.api_key:
|
if self.api_key:
|
||||||
return True
|
return True
|
||||||
|
@ -899,7 +991,7 @@ class TorrentProvider(object, GenericProvider):
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _has_no_results(*html):
|
def _has_no_results(*html):
|
||||||
return re.search(r'(?i)<(?:b|h\d|strong)[^>]*>(?:' +
|
return re.search(r'(?i)<(?:b|div|h\d|span|strong)[^>]*>(?:' +
|
||||||
'your\ssearch\sdid\snot\smatch|' +
|
'your\ssearch\sdid\snot\smatch|' +
|
||||||
'nothing\sfound|' +
|
'nothing\sfound|' +
|
||||||
'no\storrents\sfound|' +
|
'no\storrents\sfound|' +
|
||||||
|
@ -907,7 +999,6 @@ class TorrentProvider(object, GenericProvider):
|
||||||
'.*?no\shits\.\sTry\sadding' +
|
'.*?no\shits\.\sTry\sadding' +
|
||||||
')', html[0])
|
')', html[0])
|
||||||
|
|
||||||
def cache_data(self, *args, **kwargs):
|
def _cache_data(self):
|
||||||
|
|
||||||
search_params = {'Cache': ['']}
|
return self._search_provider({'Cache': ['']})
|
||||||
return self._search_provider(search_params)
|
|
||||||
|
|
|
@ -20,7 +20,7 @@ import time
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -29,7 +29,7 @@ from lib.unidecode import unidecode
|
||||||
class GFTrackerProvider(generic.TorrentProvider):
|
class GFTrackerProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'GFTracker')
|
generic.TorrentProvider.__init__(self, 'GFTracker', cache_update_freq=17)
|
||||||
|
|
||||||
self.url_base = 'https://thegft.org/'
|
self.url_base = 'https://thegft.org/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
|
@ -44,7 +44,6 @@ class GFTrackerProvider(generic.TorrentProvider):
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
||||||
self.cache = GFTrackerCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -122,16 +121,4 @@ class GFTrackerProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, scene=False, **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, scene=False, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class GFTrackerCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 17 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = GFTrackerProvider()
|
provider = GFTrackerProvider()
|
||||||
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -28,7 +28,7 @@ from lib.unidecode import unidecode
|
||||||
class GrabTheInfoProvider(generic.TorrentProvider):
|
class GrabTheInfoProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'GrabTheInfo')
|
generic.TorrentProvider.__init__(self, 'GrabTheInfo', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'http://grabthe.info/'
|
self.url_base = 'http://grabthe.info/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
|
@ -41,9 +41,7 @@ class GrabTheInfoProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = GrabTheInfoCache(self)
|
|
||||||
|
|
||||||
def _search_provider(self, search_params, **kwargs):
|
def _search_provider(self, search_params, **kwargs):
|
||||||
|
|
||||||
|
@ -57,7 +55,7 @@ class GrabTheInfoProvider(generic.TorrentProvider):
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
search_url = self.urls['browse'] % (self._categories_string(), ('0', '3')[self.freeleech],
|
search_url = self.urls['browse'] % (self._categories_string(), ('3', '0')[not self.freeleech],
|
||||||
(self.urls['search'] % search_string, '')['Cache' == mode])
|
(self.urls['search'] % search_string, '')['Cache' == mode])
|
||||||
|
|
||||||
html = self.get_url(search_url)
|
html = self.get_url(search_url)
|
||||||
|
@ -120,16 +118,4 @@ class GrabTheInfoProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='|', **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='|', **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class GrabTheInfoCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = GrabTheInfoProvider()
|
provider = GrabTheInfoProvider()
|
||||||
|
|
81
sickbeard/providers/hd4free.py
Normal file
|
@ -0,0 +1,81 @@
|
||||||
|
# coding=utf-8
|
||||||
|
#
|
||||||
|
# This file is part of SickGear.
|
||||||
|
#
|
||||||
|
# SickGear 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.
|
||||||
|
#
|
||||||
|
# SickGear 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 SickGear. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import time
|
||||||
|
|
||||||
|
from . import generic
|
||||||
|
from sickbeard.helpers import tryInt
|
||||||
|
|
||||||
|
|
||||||
|
class HD4FreeProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
generic.TorrentProvider.__init__(self, 'HD4Free')
|
||||||
|
|
||||||
|
self.url_base = 'https://hd4free.xyz/'
|
||||||
|
|
||||||
|
self.urls = {'search': self.url_base + 'searchapi.php',
|
||||||
|
'get': self.url_base + 'download.php?torrent=%s&torrent_pass=%s'}
|
||||||
|
|
||||||
|
self.url = self.url_base
|
||||||
|
|
||||||
|
self.username, self.api_key, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
|
|
||||||
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
return self._check_auth()
|
||||||
|
|
||||||
|
def _search_provider(self, search_params, age=0, **kwargs):
|
||||||
|
|
||||||
|
results = []
|
||||||
|
if not self._authorised():
|
||||||
|
return results
|
||||||
|
|
||||||
|
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
||||||
|
|
||||||
|
params = {'username': self.username, 'apikey': self.api_key,
|
||||||
|
'tv': 'true', 'fl': ('true', None)[not self.freeleech]}
|
||||||
|
for mode in search_params.keys():
|
||||||
|
for search_string in search_params[mode]:
|
||||||
|
params['search'] = '+'.join(search_string.split())
|
||||||
|
data_json = self.get_url(self.urls['search'], params=params, json=True)
|
||||||
|
|
||||||
|
cnt = len(items[mode])
|
||||||
|
for k, item in data_json.items():
|
||||||
|
if 'error' == k or not item.get('total_results'):
|
||||||
|
break
|
||||||
|
seeders, leechers, size = [tryInt(n, n) for n in [
|
||||||
|
item.get(x) for x in 'seeders', 'leechers', 'size']]
|
||||||
|
if self._peers_fail(mode, seeders, leechers):
|
||||||
|
continue
|
||||||
|
title = item.get('release_name')
|
||||||
|
download_url = (self.urls['get'] % (item.get('torrentid'), item.get('torrentpass')), None)[
|
||||||
|
not (item.get('torrentid') and item.get('torrentpass'))]
|
||||||
|
if title and download_url:
|
||||||
|
items[mode].append((title, download_url, seeders, self._bytesizer('%smb' % size)))
|
||||||
|
|
||||||
|
self._log_search(mode, len(items[mode]) - cnt, self.session.response['url'])
|
||||||
|
time.sleep(1.1)
|
||||||
|
|
||||||
|
self._sort_seeders(mode, items)
|
||||||
|
|
||||||
|
results = list(set(results + items[mode]))
|
||||||
|
|
||||||
|
return results
|
||||||
|
|
||||||
|
|
||||||
|
provider = HD4FreeProvider()
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import urllib
|
import urllib
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.exceptions import AuthException
|
from sickbeard.exceptions import AuthException
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from sickbeard.indexers import indexer_config
|
from sickbeard.indexers import indexer_config
|
||||||
|
@ -33,7 +33,7 @@ except ImportError:
|
||||||
class HDBitsProvider(generic.TorrentProvider):
|
class HDBitsProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'HDBits')
|
generic.TorrentProvider.__init__(self, 'HDBits', cache_update_freq=15)
|
||||||
|
|
||||||
# api_spec: https://hdbits.org/wiki/API
|
# api_spec: https://hdbits.org/wiki/API
|
||||||
self.url_base = 'https://hdbits.org/'
|
self.url_base = 'https://hdbits.org/'
|
||||||
|
@ -46,9 +46,7 @@ class HDBitsProvider(generic.TorrentProvider):
|
||||||
self.proper_search_terms = [' proper ', ' repack ']
|
self.proper_search_terms = [' proper ', ' repack ']
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.passkey, self.minseed, self.minleech = 4 * [None]
|
self.username, self.passkey, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = HDBitsCache(self)
|
|
||||||
|
|
||||||
def check_auth_from_data(self, parsed_json):
|
def check_auth_from_data(self, parsed_json):
|
||||||
|
|
||||||
|
@ -148,16 +146,4 @@ class HDBitsProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class HDBitsCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 15 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = HDBitsProvider()
|
provider = HDBitsProvider()
|
||||||
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
|
||||||
|
@ -27,7 +27,7 @@ from lib.unidecode import unidecode
|
||||||
class HDSpaceProvider(generic.TorrentProvider):
|
class HDSpaceProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'HDSpace')
|
generic.TorrentProvider.__init__(self, 'HDSpace', cache_update_freq=17)
|
||||||
|
|
||||||
self.url_base = 'https://hd-space.org/'
|
self.url_base = 'https://hd-space.org/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
|
@ -40,9 +40,7 @@ class HDSpaceProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = HDSpaceCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -124,16 +122,4 @@ class HDSpaceProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, scene=False, **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, scene=False, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class HDSpaceCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 17 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = HDSpaceProvider()
|
provider = HDSpaceProvider()
|
||||||
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
|
||||||
|
@ -29,24 +29,24 @@ class IPTorrentsProvider(generic.TorrentProvider):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'IPTorrents')
|
generic.TorrentProvider.__init__(self, 'IPTorrents')
|
||||||
|
|
||||||
self.url_base = 'https://iptorrents.eu/'
|
self.url_home = ['https://iptorrents.%s/' % u for u in 'eu', 'com', 'ru']
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
|
||||||
'login': self.url_base + 'torrents/',
|
self.url_vars = {'login': 'getrss.php', 'search': 't?%s;q=%s;qf=ti%s%s#torrents', 'get': '%s'}
|
||||||
'search': self.url_base + 't?%s;q=%s;qf=ti%s%s#torrents',
|
self.url_tmpl = {'config_provider_home_uri': '%(home)s', 'login': '%(home)s%(vars)s',
|
||||||
'get': self.url_base + '%s'}
|
'search': '%(home)s%(vars)s', 'get': '%(home)s%(vars)s'}
|
||||||
|
|
||||||
self.categories = {'shows': [4, 5, 22, 23, 24, 25, 26, 55, 65, 66, 73, 78, 79], 'anime': [60]}
|
self.categories = {'shows': [4, 5, 22, 23, 24, 25, 26, 55, 65, 66, 73, 78, 79], 'anime': [60]}
|
||||||
|
|
||||||
self.proper_search_terms = None
|
self.proper_search_terms = None
|
||||||
self.url = self.urls['config_provider_home_uri']
|
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.digest, self.freeleech, self.minseed, self.minleech = 4 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = IPTorrentsCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
return super(IPTorrentsProvider, self)._authorised(post_params={'php': ''})
|
return super(IPTorrentsProvider, self)._authorised(
|
||||||
|
logged_in=(lambda x=None: (None is x or 'RSS Link' in x) and self.has_all_cookies() and
|
||||||
|
self.session.cookies['uid'] in self.digest and self.session.cookies['pass'] in self.digest),
|
||||||
|
failed_msg=(lambda x=None: u'Invalid cookie details for %s. Check settings'))
|
||||||
|
|
||||||
def _search_provider(self, search_params, **kwargs):
|
def _search_provider(self, search_params, **kwargs):
|
||||||
|
|
||||||
|
@ -61,8 +61,9 @@ class IPTorrentsProvider(generic.TorrentProvider):
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
# URL with 50 tv-show results, or max 150 if adjusted in IPTorrents profile
|
# URL with 50 tv-show results, or max 150 if adjusted in IPTorrents profile
|
||||||
search_url = self.urls['search'] % (self._categories_string(mode, '%s', ';'), search_string,
|
search_url = self.urls['search'] % (
|
||||||
('', ';free')[self.freeleech], (';o=seeders', '')['Cache' == mode])
|
self._categories_string(mode, '%s', ';'), search_string,
|
||||||
|
(';free', '')[not self.freeleech], (';o=seeders', '')['Cache' == mode])
|
||||||
|
|
||||||
html = self.get_url(search_url)
|
html = self.get_url(search_url)
|
||||||
|
|
||||||
|
@ -108,15 +109,9 @@ class IPTorrentsProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
class IPTorrentsCache(tvcache.TVCache):
|
def ui_string(key):
|
||||||
|
return 'iptorrents_digest' == key and 'use... \'uid=xx; pass=yy\'' or ''
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = IPTorrentsProvider()
|
provider = IPTorrentsProvider()
|
||||||
|
|
|
@ -23,7 +23,7 @@ import traceback
|
||||||
import urllib
|
import urllib
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import config, logger, show_name_helpers, tvcache
|
from sickbeard import config, logger, show_name_helpers
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import (has_anime, tryInt)
|
from sickbeard.helpers import (has_anime, tryInt)
|
||||||
from sickbeard.common import Quality, mediaExtensions
|
from sickbeard.common import Quality, mediaExtensions
|
||||||
|
@ -34,20 +34,22 @@ from lib.unidecode import unidecode
|
||||||
class KATProvider(generic.TorrentProvider):
|
class KATProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'KickAssTorrents')
|
generic.TorrentProvider.__init__(self, 'KickAssTorrents', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'https://kat.ph/'
|
self.url_home = ['https://%s/' % u for u in 'kat.ph', 'kat.cr', 'kickass.unblocked.red', 'katproxy.com']
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
|
||||||
'base': [self.url_base, 'http://katproxy.com/'],
|
self.url_vars = {'search': 'usearch/%s/?field=time_add&sorder=desc', 'get': '%s'}
|
||||||
'search': 'usearch/%s/',
|
self.url_tmpl = {'config_provider_home_uri': '%(home)s',
|
||||||
'sorted': '?field=time_add&sorder=desc'}
|
'search': '%(home)s%(vars)s', 'get': '%(home)s%(vars)s'}
|
||||||
|
|
||||||
self.proper_search_terms = None
|
self.proper_search_terms = None
|
||||||
self.url = self.urls['config_provider_home_uri']
|
|
||||||
|
|
||||||
self.minseed, self.minleech = 2 * [None]
|
self.minseed, self.minleech = 2 * [None]
|
||||||
self.confirmed = False
|
self.confirmed = False
|
||||||
self.cache = KATCache(self)
|
|
||||||
|
@staticmethod
|
||||||
|
def _has_signature(data=None):
|
||||||
|
return data and (re.search(r'(?sim)(<title>KAT)', data[15:1024:]) or 'kastatic' in data)
|
||||||
|
|
||||||
def _find_season_quality(self, title, torrent_link, ep_number):
|
def _find_season_quality(self, title, torrent_link, ep_number):
|
||||||
""" Return the modified title of a Season Torrent with the quality found inspecting torrent file list """
|
""" Return the modified title of a Season Torrent with the quality found inspecting torrent file list """
|
||||||
|
@ -135,7 +137,7 @@ class KATProvider(generic.TorrentProvider):
|
||||||
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
||||||
|
|
||||||
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'link': 'normal', 'get': '^magnet', 'verif': 'verif'}.items())
|
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'link': 'normal', 'get': '^magnet', 'verif': 'verif'}.items())
|
||||||
url = 0
|
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
search_show = mode in ['Season', 'Episode']
|
search_show = mode in ['Season', 'Episode']
|
||||||
if not search_show and has_anime():
|
if not search_show and has_anime():
|
||||||
|
@ -145,19 +147,17 @@ class KATProvider(generic.TorrentProvider):
|
||||||
for enum, search_string in enumerate(search_params[mode]):
|
for enum, search_string in enumerate(search_params[mode]):
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
|
|
||||||
self.url = self.urls['base'][url]
|
search_url = self.urls['search'] % urllib.quote('%scategory:%s' % (
|
||||||
search_url = self.url + (self.urls['search'] % urllib.quote('%scategory:%s' % (
|
|
||||||
('', '%s ' % search_string)['Cache' != mode],
|
('', '%s ' % search_string)['Cache' != mode],
|
||||||
('tv', 'anime')[(search_show and bool(self.show and self.show.is_anime)) or bool(enum)])))
|
('tv', 'anime')[(search_show and bool(self.show and self.show.is_anime)) or bool(enum)]))
|
||||||
|
|
||||||
self.session.headers.update({'Referer': search_url})
|
self.session.headers.update({'Referer': search_url})
|
||||||
html = self.get_url(search_url + self.urls['sorted'])
|
html = self.get_url(search_url)
|
||||||
|
|
||||||
cnt = len(items[mode])
|
cnt = len(items[mode])
|
||||||
try:
|
try:
|
||||||
if not html or 'kastatic' not in html or self._has_no_results(html) or re.search(r'(?is)<(?:h\d)[^>]*>.*?(?:did\snot\smatch)', html):
|
if not html or self._has_no_results(html) or \
|
||||||
if html and 'kastatic' not in html:
|
re.search(r'(?is)<(?:h\d)[^>]*>.*?(?:did\snot\smatch)', html):
|
||||||
url += (1, 0)[url == len(self.urls['base'])]
|
|
||||||
raise generic.HaltParseException
|
raise generic.HaltParseException
|
||||||
|
|
||||||
with BS4Parser(html, features=['html5lib', 'permissive']) as soup:
|
with BS4Parser(html, features=['html5lib', 'permissive']) as soup:
|
||||||
|
@ -183,11 +183,13 @@ class KATProvider(generic.TorrentProvider):
|
||||||
except (AttributeError, TypeError, ValueError):
|
except (AttributeError, TypeError, ValueError):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if self.confirmed and not (tr.find('a', title=rc['verif']) or tr.find('i', title=rc['verif'])):
|
if self.confirmed and not (tr.find('a', title=rc['verif']) or
|
||||||
|
tr.find('i', title=rc['verif'])):
|
||||||
logger.log(u'Skipping untrusted non-verified result: %s' % title, logger.DEBUG)
|
logger.log(u'Skipping untrusted non-verified result: %s' % title, logger.DEBUG)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# Check number video files = episode in season and find the real Quality for full season torrent analyzing files in torrent
|
# Check number video files = episode in season and find the real Quality for
|
||||||
|
# full season torrent analyzing files in torrent
|
||||||
if 'Season' == mode and 'sponly' == search_mode:
|
if 'Season' == mode and 'sponly' == search_mode:
|
||||||
ep_number = int(epcount / len(set(show_name_helpers.allPossibleShowNames(self.show))))
|
ep_number = int(epcount / len(set(show_name_helpers.allPossibleShowNames(self.show))))
|
||||||
title = self._find_season_quality(title, link, ep_number)
|
title = self._find_season_quality(title, link, ep_number)
|
||||||
|
@ -208,16 +210,4 @@ class KATProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class KATCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = KATProvider()
|
provider = KATProvider()
|
||||||
|
|
|
@ -21,7 +21,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -30,19 +30,19 @@ from lib.unidecode import unidecode
|
||||||
class MoreThanProvider(generic.TorrentProvider):
|
class MoreThanProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'MoreThan')
|
generic.TorrentProvider.__init__(self, 'MoreThan', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'https://www.morethan.tv/'
|
self.url_base = 'https://www.morethan.tv/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
'login': self.url_base + 'login.php',
|
'login': self.url_base + 'login.php',
|
||||||
'search': self.url_base + 'torrents.php?searchstr=%s&' + '&'.join([
|
'search': self.url_base + 'torrents.php?searchstr=%s&' + '&'.join([
|
||||||
'tags_type=1', 'order_by=time', '&order_way=desc', 'filter_cat[2]=1', 'action=basic', 'searchsubmit=1']),
|
'tags_type=1', 'order_by=time', 'order_way=desc',
|
||||||
|
'filter_cat[2]=1', 'action=basic', 'searchsubmit=1']),
|
||||||
'get': self.url_base + '%s'}
|
'get': self.url_base + '%s'}
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
||||||
self.cache = MoreThanCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -95,7 +95,7 @@ class MoreThanProvider(generic.TorrentProvider):
|
||||||
title = '%s %s' % (tr.find('div', attrs={'class': rc['name']}).get_text().strip(),
|
title = '%s %s' % (tr.find('div', attrs={'class': rc['name']}).get_text().strip(),
|
||||||
title)
|
title)
|
||||||
|
|
||||||
link = str(tr.find('a', title=rc['get'])['href']).replace('&', '&').lstrip('/')
|
link = str(tr.find('a', href=rc['get'])['href']).replace('&', '&').lstrip('/')
|
||||||
download_url = self.urls['get'] % link
|
download_url = self.urls['get'] % link
|
||||||
except (AttributeError, TypeError, ValueError):
|
except (AttributeError, TypeError, ValueError):
|
||||||
continue
|
continue
|
||||||
|
@ -117,16 +117,4 @@ class MoreThanProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class MoreThanCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = MoreThanProvider()
|
provider = MoreThanProvider()
|
||||||
|
|
|
@ -27,14 +27,14 @@ from sickbeard.exceptions import AuthException
|
||||||
|
|
||||||
class NewznabProvider(generic.NZBProvider):
|
class NewznabProvider(generic.NZBProvider):
|
||||||
|
|
||||||
def __init__(self, name, url, key='', cat_ids='5030,5040', search_mode='eponly',
|
def __init__(self, name, url, key='', cat_ids=None, search_mode=None,
|
||||||
search_fallback=False, enable_recentsearch=False, enable_backlog=False):
|
search_fallback=False, enable_recentsearch=False, enable_backlog=False):
|
||||||
generic.NZBProvider.__init__(self, name, True, False)
|
generic.NZBProvider.__init__(self, name, True, False)
|
||||||
|
|
||||||
self.url = url
|
self.url = url
|
||||||
self.key = key
|
self.key = key
|
||||||
self.cat_ids = cat_ids
|
self.cat_ids = cat_ids or '5030,5040'
|
||||||
self.search_mode = search_mode
|
self.search_mode = search_mode or 'eponly'
|
||||||
self.search_fallback = search_fallback
|
self.search_fallback = search_fallback
|
||||||
self.enable_recentsearch = enable_recentsearch
|
self.enable_recentsearch = enable_recentsearch
|
||||||
self.enable_backlog = enable_backlog
|
self.enable_backlog = enable_backlog
|
||||||
|
@ -81,11 +81,11 @@ class NewznabProvider(generic.NZBProvider):
|
||||||
if isinstance(api_key, basestring):
|
if isinstance(api_key, basestring):
|
||||||
params['apikey'] = api_key
|
params['apikey'] = api_key
|
||||||
|
|
||||||
categories = self.get_url('%s/api' % self.url, params=params, timeout=10)
|
url = '%s/api?%s' % (self.url.strip('/'), '&'.join(['%s=%s' % (k, v) for k, v in params.items()]))
|
||||||
|
categories = self.get_url(url, timeout=10)
|
||||||
if not categories:
|
if not categories:
|
||||||
logger.log(u'Error getting html for [%s]' % self.session.response['url'], logger.DEBUG)
|
logger.log(u'Error getting html for [%s]' % url, logger.DEBUG)
|
||||||
return (False, return_categories, 'Error getting html for [%s]' %
|
return False, return_categories, 'Error getting html for [%s]' % url
|
||||||
('%s/api?%s' % (self.url, '&'.join('%s=%s' % (x, y) for x, y in params.items()))))
|
|
||||||
|
|
||||||
xml_categories = helpers.parse_xml(categories)
|
xml_categories = helpers.parse_xml(categories)
|
||||||
if not xml_categories:
|
if not xml_categories:
|
||||||
|
@ -114,16 +114,20 @@ class NewznabProvider(generic.NZBProvider):
|
||||||
base_params = {}
|
base_params = {}
|
||||||
|
|
||||||
# season
|
# season
|
||||||
|
ep_detail = None
|
||||||
if ep_obj.show.air_by_date or ep_obj.show.is_sports:
|
if ep_obj.show.air_by_date or ep_obj.show.is_sports:
|
||||||
date_str = str(ep_obj.airdate).split('-')[0]
|
airdate = str(ep_obj.airdate).split('-')[0]
|
||||||
base_params['season'] = date_str
|
base_params['season'] = airdate
|
||||||
base_params['q'] = date_str.replace('-', '.')
|
base_params['q'] = airdate
|
||||||
|
if ep_obj.show.air_by_date:
|
||||||
|
ep_detail = '+"%s"' % airdate
|
||||||
elif ep_obj.show.is_anime:
|
elif ep_obj.show.is_anime:
|
||||||
base_params['season'] = '%d' % ep_obj.scene_absolute_number
|
base_params['season'] = '%d' % ep_obj.scene_absolute_number
|
||||||
else:
|
else:
|
||||||
base_params['season'] = str((ep_obj.season, ep_obj.scene_season)[bool(ep_obj.show.is_scene)])
|
base_params['season'] = str((ep_obj.season, ep_obj.scene_season)[bool(ep_obj.show.is_scene)])
|
||||||
|
ep_detail = 'S%02d' % helpers.tryInt(base_params['season'], 1)
|
||||||
|
|
||||||
# search
|
# id search
|
||||||
ids = helpers.mapIndexersToShow(ep_obj.show)
|
ids = helpers.mapIndexersToShow(ep_obj.show)
|
||||||
if ids[1]: # or ids[2]:
|
if ids[1]: # or ids[2]:
|
||||||
params = base_params.copy()
|
params = base_params.copy()
|
||||||
|
@ -136,7 +140,7 @@ class NewznabProvider(generic.NZBProvider):
|
||||||
use_id = True
|
use_id = True
|
||||||
use_id and search_params.append(params)
|
use_id and search_params.append(params)
|
||||||
|
|
||||||
# add new query strings for exceptions
|
# query search and exceptions
|
||||||
name_exceptions = list(
|
name_exceptions = list(
|
||||||
set([helpers.sanitizeSceneName(a) for a in
|
set([helpers.sanitizeSceneName(a) for a in
|
||||||
scene_exceptions.get_scene_exceptions(ep_obj.show.indexerid) + [ep_obj.show.name]]))
|
scene_exceptions.get_scene_exceptions(ep_obj.show.indexerid) + [ep_obj.show.name]]))
|
||||||
|
@ -144,7 +148,14 @@ class NewznabProvider(generic.NZBProvider):
|
||||||
params = base_params.copy()
|
params = base_params.copy()
|
||||||
if 'q' in params:
|
if 'q' in params:
|
||||||
params['q'] = '%s.%s' % (cur_exception, params['q'])
|
params['q'] = '%s.%s' % (cur_exception, params['q'])
|
||||||
search_params.append(params)
|
search_params.append(params)
|
||||||
|
|
||||||
|
if ep_detail:
|
||||||
|
params = base_params.copy()
|
||||||
|
params['q'] = '%s.%s' % (cur_exception, ep_detail)
|
||||||
|
'season' in params and params.pop('season')
|
||||||
|
'ep' in params and params.pop('ep')
|
||||||
|
search_params.append(params)
|
||||||
|
|
||||||
return [{'Season': search_params}]
|
return [{'Season': search_params}]
|
||||||
|
|
||||||
|
@ -156,18 +167,25 @@ class NewznabProvider(generic.NZBProvider):
|
||||||
if not ep_obj:
|
if not ep_obj:
|
||||||
return [base_params]
|
return [base_params]
|
||||||
|
|
||||||
|
ep_detail = None
|
||||||
if ep_obj.show.air_by_date or ep_obj.show.is_sports:
|
if ep_obj.show.air_by_date or ep_obj.show.is_sports:
|
||||||
date_str = str(ep_obj.airdate)
|
airdate = str(ep_obj.airdate).split('-')
|
||||||
base_params['season'] = date_str.partition('-')[0]
|
base_params['season'] = airdate[0]
|
||||||
base_params['ep'] = date_str.partition('-')[2].replace('-', '/')
|
if ep_obj.show.air_by_date:
|
||||||
|
base_params['ep'] = '/'.join(airdate[1:])
|
||||||
|
ep_detail = '+"%s.%s"' % (base_params['season'], '.'.join(airdate[1:]))
|
||||||
elif ep_obj.show.is_anime:
|
elif ep_obj.show.is_anime:
|
||||||
base_params['ep'] = '%i' % int(
|
base_params['ep'] = '%i' % (helpers.tryInt(ep_obj.scene_absolute_number) or
|
||||||
ep_obj.scene_absolute_number if int(ep_obj.scene_absolute_number) > 0 else ep_obj.scene_episode)
|
helpers.tryInt(ep_obj.scene_episode))
|
||||||
|
ep_detail = '%02d' % base_params['ep']
|
||||||
else:
|
else:
|
||||||
base_params['season'], base_params['ep'] = (
|
base_params['season'], base_params['ep'] = (
|
||||||
(ep_obj.season, ep_obj.episode), (ep_obj.scene_season, ep_obj.scene_episode))[ep_obj.show.is_scene]
|
(ep_obj.season, ep_obj.episode), (ep_obj.scene_season, ep_obj.scene_episode))[ep_obj.show.is_scene]
|
||||||
|
ep_detail = sickbeard.config.naming_ep_type[2] % {
|
||||||
|
'seasonnumber': helpers.tryInt(base_params['season'], 1),
|
||||||
|
'episodenumber': helpers.tryInt(base_params['ep'], 1)}
|
||||||
|
|
||||||
# search
|
# id search
|
||||||
ids = helpers.mapIndexersToShow(ep_obj.show)
|
ids = helpers.mapIndexersToShow(ep_obj.show)
|
||||||
if ids[1]: # or ids[2]:
|
if ids[1]: # or ids[2]:
|
||||||
params = base_params.copy()
|
params = base_params.copy()
|
||||||
|
@ -181,7 +199,7 @@ class NewznabProvider(generic.NZBProvider):
|
||||||
use_id = True
|
use_id = True
|
||||||
use_id and search_params.append(params)
|
use_id and search_params.append(params)
|
||||||
|
|
||||||
# add new query strings for exceptions
|
# query search and exceptions
|
||||||
name_exceptions = list(
|
name_exceptions = list(
|
||||||
set([helpers.sanitizeSceneName(a) for a in
|
set([helpers.sanitizeSceneName(a) for a in
|
||||||
scene_exceptions.get_scene_exceptions(ep_obj.show.indexerid) + [ep_obj.show.name]]))
|
scene_exceptions.get_scene_exceptions(ep_obj.show.indexerid) + [ep_obj.show.name]]))
|
||||||
|
@ -191,15 +209,11 @@ class NewznabProvider(generic.NZBProvider):
|
||||||
params['q'] = cur_exception
|
params['q'] = cur_exception
|
||||||
search_params.append(params)
|
search_params.append(params)
|
||||||
|
|
||||||
if ep_obj.show.is_anime:
|
if ep_detail:
|
||||||
# Experimental, add a search string without search explicitly for the episode!
|
|
||||||
# Remove the ?ep=e46 parameter and use the episode number to the query parameter.
|
|
||||||
# Can be useful for newznab indexers that do not have the episodes 100% parsed.
|
|
||||||
# Start with only applying the search string to anime shows
|
|
||||||
params = base_params.copy()
|
params = base_params.copy()
|
||||||
params['q'] = '%s.%02d' % (cur_exception, int(params['ep']))
|
params['q'] = '%s.%s' % (cur_exception, ep_detail)
|
||||||
if 'ep' in params:
|
'season' in params and params.pop('season')
|
||||||
params.pop('ep')
|
'ep' in params and params.pop('ep')
|
||||||
search_params.append(params)
|
search_params.append(params)
|
||||||
|
|
||||||
return [{'Episode': search_params}]
|
return [{'Episode': search_params}]
|
||||||
|
@ -229,8 +243,10 @@ class NewznabProvider(generic.NZBProvider):
|
||||||
|
|
||||||
# category ids
|
# category ids
|
||||||
cat = []
|
cat = []
|
||||||
cat_anime = ('5070', '6070')['nzbs_org' == self.get_id()]
|
cat_sport = ['5060']
|
||||||
cat_sport = '5060'
|
cat_anime = []
|
||||||
|
if 'nzbgeek' != self.get_id():
|
||||||
|
cat_anime = (['5070'], ['6070'])['nzbs_org' == self.get_id()]
|
||||||
if 'Episode' == mode or 'Season' == mode:
|
if 'Episode' == mode or 'Season' == mode:
|
||||||
if not ('rid' in params or 'tvdbid' in params or 'q' in params or not self.supports_tvdbid()):
|
if not ('rid' in params or 'tvdbid' in params or 'q' in params or not self.supports_tvdbid()):
|
||||||
logger.log('Error no rid, tvdbid, or search term available for search.')
|
logger.log('Error no rid, tvdbid, or search term available for search.')
|
||||||
|
@ -238,18 +254,24 @@ class NewznabProvider(generic.NZBProvider):
|
||||||
|
|
||||||
if self.show:
|
if self.show:
|
||||||
if self.show.is_sports:
|
if self.show.is_sports:
|
||||||
cat = [cat_sport]
|
cat = cat_sport
|
||||||
elif self.show.is_anime:
|
elif self.show.is_anime:
|
||||||
cat = [cat_anime]
|
cat = cat_anime
|
||||||
else:
|
else:
|
||||||
cat = [cat_sport, cat_anime]
|
cat = cat_sport + cat_anime
|
||||||
|
|
||||||
if self.cat_ids or len(cat):
|
if self.cat_ids or len(cat):
|
||||||
base_params['cat'] = ','.join(sorted(set(self.cat_ids.split(',') + cat)))
|
base_params['cat'] = ','.join(sorted(set(self.cat_ids.split(',') + cat)))
|
||||||
|
|
||||||
request_params = base_params.copy()
|
request_params = base_params.copy()
|
||||||
|
if 'q' in params and not (any(x in params for x in ['season', 'ep'])):
|
||||||
|
request_params['t'] = 'search'
|
||||||
request_params.update(params)
|
request_params.update(params)
|
||||||
|
|
||||||
|
# workaround a strange glitch
|
||||||
|
if sum(ord(i) for i in self.get_id()) in [383] and 5 == 14 - request_params['maxage']:
|
||||||
|
request_params['maxage'] += 1
|
||||||
|
|
||||||
offset = 0
|
offset = 0
|
||||||
batch_count = not 0
|
batch_count = not 0
|
||||||
|
|
||||||
|
@ -336,18 +358,17 @@ class NewznabCache(tvcache.TVCache):
|
||||||
|
|
||||||
result = []
|
result = []
|
||||||
|
|
||||||
if True or self.shouldUpdate():
|
if 4489 != sickbeard.RECENTSEARCH_FREQUENCY or self.should_update():
|
||||||
try:
|
try:
|
||||||
self._checkAuth()
|
self._checkAuth()
|
||||||
|
items = self.provider.cache_data()
|
||||||
except Exception:
|
except Exception:
|
||||||
return result
|
items = None
|
||||||
|
|
||||||
items = self.provider.cache_data()
|
|
||||||
if items:
|
if items:
|
||||||
|
|
||||||
self._clearCache()
|
self._clearCache()
|
||||||
self.setLastUpdate()
|
|
||||||
|
|
||||||
|
# parse data
|
||||||
cl = []
|
cl = []
|
||||||
for item in items:
|
for item in items:
|
||||||
ci = self._parseItem(item)
|
ci = self._parseItem(item)
|
||||||
|
@ -358,6 +379,9 @@ class NewznabCache(tvcache.TVCache):
|
||||||
my_db = self.get_db()
|
my_db = self.get_db()
|
||||||
my_db.mass_action(cl)
|
my_db.mass_action(cl)
|
||||||
|
|
||||||
|
# set updated as time the attempt to fetch data is
|
||||||
|
self.setLastUpdate()
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
# overwrite method with that parses the rageid from the newznab feed
|
# overwrite method with that parses the rageid from the newznab feed
|
||||||
|
|
|
@ -17,7 +17,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -37,7 +37,6 @@ class PiSexyProvider(generic.TorrentProvider):
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
||||||
self.cache = PiSexyCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -108,14 +107,4 @@ class PiSexyProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class PiSexyCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = PiSexyProvider()
|
provider = PiSexyProvider()
|
||||||
|
|
|
@ -16,7 +16,6 @@
|
||||||
# along with SickGear. If not, see <http://www.gnu.org/licenses/>.
|
# along with SickGear. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import tvcache
|
|
||||||
from sickbeard.rssfeeds import RSSFeeds
|
from sickbeard.rssfeeds import RSSFeeds
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
|
||||||
|
@ -24,7 +23,7 @@ from lib.unidecode import unidecode
|
||||||
class PreToMeProvider(generic.TorrentProvider):
|
class PreToMeProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'PreToMe')
|
generic.TorrentProvider.__init__(self, 'PreToMe', cache_update_freq=6)
|
||||||
|
|
||||||
self.url_base = 'https://pretome.info/'
|
self.url_base = 'https://pretome.info/'
|
||||||
|
|
||||||
|
@ -35,7 +34,6 @@ class PreToMeProvider(generic.TorrentProvider):
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.passkey = None
|
self.passkey = None
|
||||||
self.cache = PreToMeCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -72,16 +70,4 @@ class PreToMeProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class PreToMeCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 6 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = PreToMeProvider()
|
provider = PreToMeProvider()
|
||||||
|
|
|
@ -21,7 +21,7 @@ import datetime
|
||||||
import time
|
import time
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import helpers, logger, tvcache
|
from sickbeard import helpers, logger
|
||||||
from sickbeard.indexers.indexer_config import INDEXER_TVDB
|
from sickbeard.indexers.indexer_config import INDEXER_TVDB
|
||||||
|
|
||||||
|
|
||||||
|
@ -51,7 +51,6 @@ class RarbgProvider(generic.TorrentProvider):
|
||||||
self.minseed, self.minleech, self.token, self.token_expiry = 4 * [None]
|
self.minseed, self.minleech, self.token, self.token_expiry = 4 * [None]
|
||||||
self.confirmed = False
|
self.confirmed = False
|
||||||
self.request_throttle = datetime.datetime.now()
|
self.request_throttle = datetime.datetime.now()
|
||||||
self.cache = RarbgCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, reset=False, **kwargs):
|
def _authorised(self, reset=False, **kwargs):
|
||||||
|
|
||||||
|
@ -178,14 +177,4 @@ class RarbgProvider(generic.TorrentProvider):
|
||||||
return search_params
|
return search_params
|
||||||
|
|
||||||
|
|
||||||
class RarbgCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = RarbgProvider()
|
provider = RarbgProvider()
|
||||||
|
|
|
@ -20,7 +20,7 @@ import time
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -31,18 +31,16 @@ class SCCProvider(generic.TorrentProvider):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'SceneAccess')
|
generic.TorrentProvider.__init__(self, 'SceneAccess')
|
||||||
|
|
||||||
self.url_base = 'https://sceneaccess.eu/'
|
self.url_home = ['https://sceneaccess.%s/' % u for u in 'eu', 'org']
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
|
||||||
'login': self.url_base + 'login',
|
|
||||||
'search': self.url_base + 'browse?search=%s&method=1&c27=27&c17=17&c11=11',
|
|
||||||
'nonscene': self.url_base + 'nonscene?search=%s&method=1&c44=44&c45=44',
|
|
||||||
'archive': self.url_base + 'archive?search=%s&method=1&c26=26',
|
|
||||||
'get': self.url_base + '%s'}
|
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url_vars = {
|
||||||
|
'login': 'login', 'search': 'browse?search=%s&method=1&c27=27&c17=17&c11=11', 'get': '%s',
|
||||||
|
'nonscene': 'nonscene?search=%s&method=1&c44=44&c45=44', 'archive': 'archive?search=%s&method=1&c26=26'}
|
||||||
|
self.url_tmpl = {
|
||||||
|
'config_provider_home_uri': '%(home)s', 'login': '%(home)s%(vars)s', 'search': '%(home)s%(vars)s',
|
||||||
|
'get': '%(home)s%(vars)s', 'nonscene': '%(home)s%(vars)s', 'archive': '%(home)s%(vars)s'}
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
||||||
self.cache = SCCCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -121,16 +119,4 @@ class SCCProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='.', **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='.', **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class SCCCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = SCCProvider()
|
provider = SCCProvider()
|
||||||
|
|
|
@ -20,7 +20,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -29,7 +29,7 @@ from lib.unidecode import unidecode
|
||||||
class SceneTimeProvider(generic.TorrentProvider):
|
class SceneTimeProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'SceneTime')
|
generic.TorrentProvider.__init__(self, 'SceneTime', cache_update_freq=15)
|
||||||
|
|
||||||
self.url_base = 'https://www.scenetime.com/'
|
self.url_base = 'https://www.scenetime.com/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
|
@ -42,9 +42,7 @@ class SceneTimeProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = SceneTimeCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -58,15 +56,16 @@ class SceneTimeProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
||||||
|
|
||||||
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'info': 'detail', 'get': '.*id=(\d+).*', 'fl': '\[freeleech\]',
|
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {
|
||||||
'cats': 'cat=(?:%s)' % self._categories_string(template='', delimiter='|')
|
'info': 'detail', 'get': '.*id=(\d+).*', 'fl': '\[freeleech\]',
|
||||||
}.items())
|
'cats': 'cat=(?:%s)' % self._categories_string(template='', delimiter='|')}.items())
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
|
|
||||||
post_data = self.urls['params'].copy()
|
post_data = self.urls['params'].copy()
|
||||||
post_data.update(ast.literal_eval('{%s}' % self._categories_string(template='"c%s": "1"', delimiter=',')))
|
post_data.update(ast.literal_eval(
|
||||||
|
'{%s}' % self._categories_string(template='"c%s": "1"', delimiter=',')))
|
||||||
if 'Cache' != mode:
|
if 'Cache' != mode:
|
||||||
search_string = '+'.join(search_string.split())
|
search_string = '+'.join(search_string.split())
|
||||||
post_data['search'] = search_string
|
post_data['search'] = search_string
|
||||||
|
@ -99,10 +98,11 @@ class SceneTimeProvider(generic.TorrentProvider):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
info = tr.find('a', href=rc['info'])
|
info = tr.find('a', href=rc['info'])
|
||||||
title = 'title' in info.attrs and info.attrs['title'] or info.get_text().strip()
|
title = info.attrs.get('title') or info.get_text().strip()
|
||||||
|
|
||||||
download_url = self.urls['get'] % {'id': re.sub(rc['get'], r'\1', str(info.attrs['href'])),
|
download_url = self.urls['get'] % {
|
||||||
'title': str(title).replace(' ', '.')}
|
'id': re.sub(rc['get'], r'\1', str(info.attrs['href'])),
|
||||||
|
'title': str(title).replace(' ', '.')}
|
||||||
except (AttributeError, TypeError, ValueError):
|
except (AttributeError, TypeError, ValueError):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
@ -124,16 +124,4 @@ class SceneTimeProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class SceneTimeCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 15 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = SceneTimeProvider()
|
provider = SceneTimeProvider()
|
||||||
|
|
|
@ -22,7 +22,7 @@ import time
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import helpers, logger, tvcache
|
from sickbeard import helpers, logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -32,7 +32,7 @@ class ShazbatProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
|
|
||||||
generic.TorrentProvider.__init__(self, 'Shazbat')
|
generic.TorrentProvider.__init__(self, 'Shazbat', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'https://www.shazbat.tv/'
|
self.url_base = 'https://www.shazbat.tv/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
|
@ -46,8 +46,6 @@ class ShazbatProvider(generic.TorrentProvider):
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = ShazbatCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -147,16 +145,4 @@ class ShazbatProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, detail_only=True, scene=False, **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, detail_only=True, scene=False, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class ShazbatCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = ShazbatProvider()
|
provider = ShazbatProvider()
|
||||||
|
|
|
@ -19,34 +19,32 @@ import re
|
||||||
import time
|
import time
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import tvcache
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
|
|
||||||
|
|
||||||
class SpeedCDProvider(generic.TorrentProvider):
|
class SpeedCDProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'SpeedCD')
|
generic.TorrentProvider.__init__(self, 'SpeedCD', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'http://speed.cd/'
|
self.url_base = 'https://speed.cd/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
'login_action': self.url_base + 'login.php',
|
'login_action': self.url_base + 'login.php',
|
||||||
'search': self.url_base + 'V3/API/API.php',
|
'search': self.url_base + 'V3/API/API.php',
|
||||||
'get': self.url_base + 'download.php?torrent=%s'}
|
'get': self.url_base + '%s'}
|
||||||
|
|
||||||
self.categories = {'Season': {'c41': 1, 'c53': 1},
|
self.categories = {'Season': [41, 53], 'Episode': [2, 49, 50, 55], 'anime': [30]}
|
||||||
'Episode': {'c2': 1, 'c49': 1, 'c50': 1, 'c55': 1},
|
self.categories['Cache'] = self.categories['Season'] + self.categories['Episode']
|
||||||
'Cache': {'c41': 1, 'c2': 1, 'c49': 1, 'c50': 1, 'c53': 1, 'c55': 1}}
|
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = SpeedCDCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
return super(SpeedCDProvider, self)._authorised(logged_in=(lambda x=None: self.has_all_cookies('inSpeed_speedian')))
|
return super(SpeedCDProvider, self)._authorised(
|
||||||
|
logged_in=(lambda x=None: self.has_all_cookies('inSpeed_speedian')))
|
||||||
|
|
||||||
def _search_provider(self, search_params, **kwargs):
|
def _search_provider(self, search_params, **kwargs):
|
||||||
|
|
||||||
|
@ -56,37 +54,49 @@ class SpeedCDProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
||||||
|
|
||||||
remove_tag = re.compile(r'<[^>]*>')
|
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'get': 'download', 'fl': '\[freeleech\]'}.items())
|
||||||
|
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
search_mode = (mode, 'Episode')['Propers' == mode]
|
rc['cats'] = re.compile('(?i)cat=(?:%s)' % self._categories_string(mode, template='', delimiter='|'))
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = '+'.join(search_string.split())
|
search_string = '+'.join(search_string.split())
|
||||||
post_data = dict({'/browse.php?': None, 'cata': 'yes', 'jxt': 4, 'jxw': 'b', 'search': search_string},
|
post_data = dict((x.split('=') for x in self._categories_string(mode).split('&')), search=search_string,
|
||||||
**self.categories[search_mode])
|
jxt=2, jxw='b', freeleech=('on', None)[not self.freeleech])
|
||||||
if self.freeleech:
|
|
||||||
post_data['freeleech'] = 'on'
|
|
||||||
|
|
||||||
data_json = self.get_url(self.urls['search'], post_data=post_data, json=True)
|
data_json = self.get_url(self.urls['search'], post_data=post_data, json=True)
|
||||||
|
|
||||||
cnt = len(items[mode])
|
cnt = len(items[mode])
|
||||||
try:
|
try:
|
||||||
if not data_json:
|
html = data_json.get('Fs')[0].get('Cn')[0].get('d')
|
||||||
|
if not html or self._has_no_results(html):
|
||||||
raise generic.HaltParseException
|
raise generic.HaltParseException
|
||||||
torrents = data_json.get('Fs', [])[0].get('Cn', {}).get('torrents', [])
|
|
||||||
|
|
||||||
for item in torrents:
|
with BS4Parser(html, features=['html5lib', 'permissive']) as soup:
|
||||||
|
torrent_table = soup.find('table', attrs={'cellspacing': 0})
|
||||||
|
torrent_rows = [] if not torrent_table else torrent_table.find_all('tr')
|
||||||
|
|
||||||
if self.freeleech and not item.get('free'):
|
if 2 > len(torrent_rows):
|
||||||
continue
|
raise generic.HaltParseException
|
||||||
|
|
||||||
seeders, leechers, size = [tryInt(n, n) for n in [item.get(x) for x in 'seed', 'leech', 'size']]
|
for tr in torrent_rows[1:]:
|
||||||
if self._peers_fail(mode, seeders, leechers):
|
try:
|
||||||
continue
|
seeders, leechers, size = [tryInt(n, n) for n in [
|
||||||
|
tr.find_all('td')[x].get_text().strip() for x in (-2, -1, -3)]]
|
||||||
|
if None is tr.find('a', href=rc['cats']) \
|
||||||
|
or self.freeleech and None is rc['fl'].search(tr.find_all('td')[1].get_text()) \
|
||||||
|
or self._peers_fail(mode, seeders, leechers):
|
||||||
|
continue
|
||||||
|
|
||||||
title = remove_tag.sub('', item.get('name'))
|
info = tr.find('a', 'torrent')
|
||||||
download_url = self.urls['get'] % item.get('id')
|
title = info.attrs.get('title') or info.get_text().strip()
|
||||||
if title and download_url:
|
|
||||||
items[mode].append((title, download_url, seeders, self._bytesizer(size)))
|
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip('/')
|
||||||
|
|
||||||
|
except (AttributeError, TypeError, ValueError):
|
||||||
|
continue
|
||||||
|
|
||||||
|
if title and download_url:
|
||||||
|
items[mode].append((title, download_url, seeders, self._bytesizer(size)))
|
||||||
|
|
||||||
except Exception:
|
except Exception:
|
||||||
time.sleep(1.1)
|
time.sleep(1.1)
|
||||||
|
@ -105,16 +115,4 @@ class SpeedCDProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='.', **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='.', **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class SpeedCDCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = SpeedCDProvider()
|
provider = SpeedCDProvider()
|
||||||
|
|
|
@ -23,7 +23,7 @@ import traceback
|
||||||
import urllib
|
import urllib
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import config, logger, tvcache, show_name_helpers
|
from sickbeard import config, logger, show_name_helpers
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.common import Quality, mediaExtensions
|
from sickbeard.common import Quality, mediaExtensions
|
||||||
from sickbeard.name_parser.parser import NameParser, InvalidNameException, InvalidShowException
|
from sickbeard.name_parser.parser import NameParser, InvalidNameException, InvalidShowException
|
||||||
|
@ -33,40 +33,32 @@ from lib.unidecode import unidecode
|
||||||
class ThePirateBayProvider(generic.TorrentProvider):
|
class ThePirateBayProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'The Pirate Bay')
|
generic.TorrentProvider.__init__(self, 'The Pirate Bay', cache_update_freq=20)
|
||||||
|
|
||||||
self.urls = {'config_provider_home_uri': ['https://thepiratebay.se/', 'https://thepiratebay.gd/',
|
self.url_home = ['https://thepiratebay.%s/' % u for u in 'se', 'org']
|
||||||
'https://thepiratebay.mn/', 'https://thepiratebay.vg/',
|
|
||||||
'https://thepiratebay.la/'],
|
self.url_vars = {'search': 'search/%s/0/7/200', 'browse': 'tv/latest/'}
|
||||||
'search': 'search/%s/0/7/200',
|
self.url_tmpl = {'config_provider_home_uri': '%(home)s', 'search': '%(home)s%(vars)s',
|
||||||
'browse': 'tv/latest/'} # order by seed
|
'browse': '%(home)s%(vars)s'}
|
||||||
|
|
||||||
self.proper_search_terms = None
|
self.proper_search_terms = None
|
||||||
self.url = self.urls['config_provider_home_uri'][0]
|
|
||||||
|
|
||||||
self.minseed, self.minleech = 2 * [None]
|
self.minseed, self.minleech = 2 * [None]
|
||||||
self.confirmed = False
|
self.confirmed = False
|
||||||
self.cache = ThePirateBayCache(self)
|
|
||||||
|
@staticmethod
|
||||||
|
def _has_signature(data=None):
|
||||||
|
return data and re.search(r'Pirate\sBay', data[33:7632:])
|
||||||
|
|
||||||
def _find_season_quality(self, title, torrent_id, ep_number):
|
def _find_season_quality(self, title, torrent_id, ep_number):
|
||||||
""" Return the modified title of a Season Torrent with the quality found inspecting torrent file list """
|
""" Return the modified title of a Season Torrent with the quality found inspecting torrent file list """
|
||||||
|
|
||||||
|
if not self.url:
|
||||||
|
return False
|
||||||
|
|
||||||
quality = Quality.UNKNOWN
|
quality = Quality.UNKNOWN
|
||||||
file_name = None
|
file_name = None
|
||||||
data = None
|
data = self.get_url('%sajax_details_filelist.php?id=%s' % (self.url, torrent_id))
|
||||||
has_signature = False
|
|
||||||
details_url = '/ajax_details_filelist.php?id=%s' % torrent_id
|
|
||||||
for idx, url in enumerate(self.urls['config_provider_home_uri']):
|
|
||||||
data = self.get_url(url + details_url)
|
|
||||||
if data and re.search(r'<title>The\sPirate\sBay', data[33:200:]):
|
|
||||||
has_signature = True
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
data = None
|
|
||||||
|
|
||||||
if not has_signature:
|
|
||||||
logger.log(u'Failed to identify a page from ThePirateBay at %s attempted urls (tpb blocked? general network issue or site dead)' % len(self.urls['config_provider_home_uri']), logger.ERROR)
|
|
||||||
|
|
||||||
if not data:
|
if not data:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
@ -138,30 +130,22 @@ class ThePirateBayProvider(generic.TorrentProvider):
|
||||||
def _search_provider(self, search_params, search_mode='eponly', epcount=0, **kwargs):
|
def _search_provider(self, search_params, search_mode='eponly', epcount=0, **kwargs):
|
||||||
|
|
||||||
results = []
|
results = []
|
||||||
|
if not self.url:
|
||||||
|
return results
|
||||||
|
|
||||||
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
||||||
|
|
||||||
rc = dict((k, re.compile('(?i)' + v))
|
rc = dict((k, re.compile('(?i)' + v))
|
||||||
for (k, v) in {'info': 'detail', 'get': 'download[^"]+magnet', 'tid': r'.*/(\d{5,}).*',
|
for (k, v) in {'info': 'detail', 'get': 'download[^"]+magnet', 'tid': r'.*/(\d{5,}).*',
|
||||||
'verify': '(?:helper|moderator|trusted|vip)'}.items())
|
'verify': '(?:helper|moderator|trusted|vip)'}.items())
|
||||||
has_signature = False
|
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
|
|
||||||
log_url = '%s %s' % (self.name, search_string) # placebo value
|
search_url = self.urls['browse'] if 'Cache' == mode \
|
||||||
for idx, search_url in enumerate(self.urls['config_provider_home_uri']):
|
else self.urls['search'] % (urllib.quote(search_string))
|
||||||
search_url += self.urls['browse'] if 'Cache' == mode\
|
html = self.get_url(search_url)
|
||||||
else self.urls['search'] % (urllib.quote(search_string))
|
|
||||||
|
|
||||||
log_url = u'(%s/%s): %s' % (idx + 1, len(self.urls['config_provider_home_uri']), search_url)
|
|
||||||
|
|
||||||
html = self.get_url(search_url)
|
|
||||||
|
|
||||||
if html and re.search(r'Pirate\sBay', html[33:7632:]):
|
|
||||||
has_signature = True
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
html = None
|
|
||||||
|
|
||||||
cnt = len(items[mode])
|
cnt = len(items[mode])
|
||||||
try:
|
try:
|
||||||
|
@ -213,28 +197,13 @@ class ThePirateBayProvider(generic.TorrentProvider):
|
||||||
pass
|
pass
|
||||||
except Exception:
|
except Exception:
|
||||||
logger.log(u'Failed to parse. Traceback: %s' % traceback.format_exc(), logger.ERROR)
|
logger.log(u'Failed to parse. Traceback: %s' % traceback.format_exc(), logger.ERROR)
|
||||||
self._log_search(mode, len(items[mode]) - cnt, log_url)
|
self._log_search(mode, len(items[mode]) - cnt, search_url)
|
||||||
|
|
||||||
self._sort_seeders(mode, items)
|
self._sort_seeders(mode, items)
|
||||||
|
|
||||||
results = list(set(results + items[mode]))
|
results = list(set(results + items[mode]))
|
||||||
|
|
||||||
if not has_signature:
|
|
||||||
logger.log(u'Failed to identify a page from ThePirateBay at %s attempted urls (tpb blocked? general network issue or site dead)' % len(self.urls['config_provider_home_uri']), logger.ERROR)
|
|
||||||
|
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class ThePirateBayCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = ThePirateBayProvider()
|
provider = ThePirateBayProvider()
|
||||||
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -28,21 +28,18 @@ from lib.unidecode import unidecode
|
||||||
class TorrentBytesProvider(generic.TorrentProvider):
|
class TorrentBytesProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'TorrentBytes')
|
generic.TorrentProvider.__init__(self, 'TorrentBytes', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'https://www.torrentbytes.net/'
|
self.url_home = ['https://www.torrentbytes.net/']
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
|
||||||
'login': self.url_base + 'takelogin.php',
|
|
||||||
'search': self.url_base + 'browse.php?search=%s&%s',
|
|
||||||
'get': self.url_base + '%s'}
|
|
||||||
|
|
||||||
self.categories = {'shows': [41, 33, 38, 32, 37]}
|
self.url_vars = {'login': 'takelogin.php', 'search': 'browse.php?search=%s&%s', 'get': '%s'}
|
||||||
|
self.url_tmpl = {'config_provider_home_uri': '%(home)s', 'login': '%(home)s%(vars)s',
|
||||||
|
'search': '%(home)s%(vars)s', 'get': '%(home)s%(vars)s'}
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.categories = {'Season': [41, 32], 'Episode': [33, 37, 38]}
|
||||||
|
self.categories['Cache'] = self.categories['Season'] + self.categories['Episode']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = TorrentBytesCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -56,12 +53,12 @@ class TorrentBytesProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
||||||
|
|
||||||
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'info': 'detail', 'get': 'download', 'fl': '\[\W*F\W?L\W*\]'
|
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'info': 'detail', 'get': 'download',
|
||||||
}.items())
|
'fl': '\[\W*F\W?L\W*\]'}.items())
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
search_url = self.urls['search'] % (search_string, self._categories_string())
|
search_url = self.urls['search'] % (search_string, self._categories_string(mode))
|
||||||
|
|
||||||
html = self.get_url(search_url, timeout=90)
|
html = self.get_url(search_url, timeout=90)
|
||||||
|
|
||||||
|
@ -82,11 +79,11 @@ class TorrentBytesProvider(generic.TorrentProvider):
|
||||||
info = tr.find('a', href=rc['info'])
|
info = tr.find('a', href=rc['info'])
|
||||||
seeders, leechers, size = [tryInt(n, n) for n in [
|
seeders, leechers, size = [tryInt(n, n) for n in [
|
||||||
tr.find_all('td')[x].get_text().strip() for x in (-2, -1, -4)]]
|
tr.find_all('td')[x].get_text().strip() for x in (-2, -1, -4)]]
|
||||||
if self.freeleech and (len(info.contents) < 2 or not rc['fl'].search(info.contents[1].string.strip())) \
|
if self.freeleech and (len(info.contents) < 2 or not rc['fl'].search(
|
||||||
or self._peers_fail(mode, seeders, leechers):
|
info.contents[1].string.strip())) or self._peers_fail(mode, seeders, leechers):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
title = 'title' in info.attrs and info.attrs['title'] or info.contents[0]
|
title = info.attrs.get('title') or info.contents[0]
|
||||||
title = (isinstance(title, list) and title[0] or title).strip()
|
title = (isinstance(title, list) and title[0] or title).strip()
|
||||||
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip('/')
|
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip('/')
|
||||||
except (AttributeError, TypeError, ValueError):
|
except (AttributeError, TypeError, ValueError):
|
||||||
|
@ -109,16 +106,4 @@ class TorrentBytesProvider(generic.TorrentProvider):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
class TorrentBytesCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = TorrentBytesProvider()
|
provider = TorrentBytesProvider()
|
||||||
|
|
|
@ -19,8 +19,7 @@ import re
|
||||||
import time
|
import time
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import tvcache
|
from sickbeard.helpers import tryInt
|
||||||
from sickbeard.helpers import (has_anime, tryInt)
|
|
||||||
|
|
||||||
|
|
||||||
class TorrentDayProvider(generic.TorrentProvider):
|
class TorrentDayProvider(generic.TorrentProvider):
|
||||||
|
@ -28,22 +27,19 @@ class TorrentDayProvider(generic.TorrentProvider):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'TorrentDay')
|
generic.TorrentProvider.__init__(self, 'TorrentDay')
|
||||||
|
|
||||||
self.url_base = 'https://torrentday.eu/'
|
self.url_home = ['https://%s/' % u for u in 'torrentday.eu', 'secure.torrentday.com', 'tdonline.org',
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
'torrentday.it', 'www.td.af', 'www.torrentday.com']
|
||||||
'login': self.url_base + 'torrents/',
|
|
||||||
'search': self.url_base + 'V3/API/API.php',
|
|
||||||
'get': self.url_base + 'download.php/%s/%s'}
|
|
||||||
|
|
||||||
self.categories = {'Season': {'c31': 1, 'c33': 1, 'c14': 1},
|
self.url_vars = {'login': 'torrents/', 'search': 'V3/API/API.php', 'get': 'download.php/%s/%s'}
|
||||||
'Episode': {'c32': 1, 'c26': 1, 'c7': 1, 'c2': 1},
|
self.url_tmpl = {'config_provider_home_uri': '%(home)s', 'login': '%(home)s%(vars)s',
|
||||||
'Cache': {'c31': 1, 'c33': 1, 'c14': 1, 'c32': 1, 'c26': 1, 'c7': 1, 'c2': 1}}
|
'search': '%(home)s%(vars)s', 'get': '%(home)s%(vars)s'}
|
||||||
|
|
||||||
|
self.categories = {'Season': [31, 33, 14], 'Episode': [24, 32, 26, 7, 2], 'Anime': [29]}
|
||||||
|
self.categories['Cache'] = self.categories['Season'] + self.categories['Episode']
|
||||||
|
|
||||||
self.proper_search_terms = None
|
self.proper_search_terms = None
|
||||||
self.url = self.urls['config_provider_home_uri']
|
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = TorrentDayCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -66,11 +62,8 @@ class TorrentDayProvider(generic.TorrentProvider):
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = '+'.join(search_string.split())
|
search_string = '+'.join(search_string.split())
|
||||||
post_data = dict({'/browse.php?': None, 'cata': 'yes', 'jxt': 8, 'jxw': 'b', 'search': search_string},
|
post_data = dict((x.split('=') for x in self._categories_string(mode).split('&')),
|
||||||
**self.categories[(mode, 'Episode')['Propers' == mode]])
|
search=search_string, cata='yes', jxt=8, jxw='b')
|
||||||
if ('Cache' == mode and has_anime()) or (
|
|
||||||
mode in ['Season', 'Episode'] and self.show and self.show.is_anime):
|
|
||||||
post_data.update({'c29': 1})
|
|
||||||
|
|
||||||
if self.freeleech:
|
if self.freeleech:
|
||||||
post_data.update({'free': 'on'})
|
post_data.update({'free': 'on'})
|
||||||
|
@ -112,14 +105,4 @@ class TorrentDayProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='.', date_or=True, **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='.', date_or=True, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class TorrentDayCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = TorrentDayProvider()
|
provider = TorrentDayProvider()
|
||||||
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -30,25 +30,20 @@ class TorrentingProvider(generic.TorrentProvider):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'Torrenting')
|
generic.TorrentProvider.__init__(self, 'Torrenting')
|
||||||
|
|
||||||
self.url_base = 'https://www.torrenting.com/'
|
self.url_home = ['https://%s/' % u for u in 'www.torrenting.com', 'ttonline.us']
|
||||||
|
|
||||||
self.api = 'https://ttonline.us/'
|
self.url_vars = {'login': 'rss.php', 'search': 'browse.php?%s&search=%s', 'get': '%s'}
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.url_tmpl = {'config_provider_home_uri': '%(home)s', 'login': '%(home)s%(vars)s',
|
||||||
'login': self.api + 'secure.php',
|
'search': '%(home)s%(vars)s', 'get': '%(home)s%(vars)s'}
|
||||||
'search': self.api + 'browse.php?%s&search=%s',
|
|
||||||
'get': self.api + '%s'}
|
|
||||||
|
|
||||||
self.categories = {'shows': [4, 5]}
|
self.categories = {'shows': [4, 5]}
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
|
||||||
|
|
||||||
self.digest, self.minseed, self.minleech = 3 * [None]
|
self.digest, self.minseed, self.minleech = 3 * [None]
|
||||||
self.cache = TorrentingCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
return super(TorrentingProvider, self)._authorised(
|
return super(TorrentingProvider, self)._authorised(
|
||||||
logged_in=(lambda x=None: (None is x or 'Other Links' in x) and self.has_all_cookies() and
|
logged_in=(lambda x=None: (None is x or 'RSS link' in x) and self.has_all_cookies() and
|
||||||
self.session.cookies['uid'] in self.digest and self.session.cookies['pass'] in self.digest),
|
self.session.cookies['uid'] in self.digest and self.session.cookies['pass'] in self.digest),
|
||||||
failed_msg=(lambda x=None: u'Invalid cookie details for %s. Check settings'))
|
failed_msg=(lambda x=None: u'Invalid cookie details for %s. Check settings'))
|
||||||
|
|
||||||
|
@ -60,9 +55,9 @@ class TorrentingProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
items = {'Cache': [], 'Season': [], 'Episode': [], 'Propers': []}
|
||||||
|
|
||||||
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {'info': 'detail', 'get': 'download',
|
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {
|
||||||
'cats': 'cat=(?:%s)' % self._categories_string(template='', delimiter='|')
|
'info': 'detail', 'cats': 'cat=(?:%s)' % self._categories_string(template='', delimiter='|'),
|
||||||
}.items())
|
'get': 'download'}.items())
|
||||||
for mode in search_params.keys():
|
for mode in search_params.keys():
|
||||||
for search_string in search_params[mode]:
|
for search_string in search_params[mode]:
|
||||||
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
search_string = isinstance(search_string, unicode) and unidecode(search_string) or search_string
|
||||||
|
@ -90,8 +85,9 @@ class TorrentingProvider(generic.TorrentProvider):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
info = tr.find('a', href=rc['info'])
|
info = tr.find('a', href=rc['info'])
|
||||||
title = 'title' in info.attrs and info.attrs['title'] or info.get_text().strip()
|
title = info.attrs.get('title') or info.get_text().strip()
|
||||||
download_url = self.urls['get'] % tr.find('a', href=rc['get']).get('href')
|
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip('/')
|
||||||
|
|
||||||
except (AttributeError, TypeError, ValueError):
|
except (AttributeError, TypeError, ValueError):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
@ -117,14 +113,4 @@ class TorrentingProvider(generic.TorrentProvider):
|
||||||
return 'torrenting_digest' == key and 'use... \'uid=xx; pass=yy\'' or ''
|
return 'torrenting_digest' == key and 'use... \'uid=xx; pass=yy\'' or ''
|
||||||
|
|
||||||
|
|
||||||
class TorrentingCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = TorrentingProvider()
|
provider = TorrentingProvider()
|
||||||
|
|
|
@ -19,14 +19,14 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
|
||||||
|
|
||||||
class TorrentLeechProvider(generic.TorrentProvider):
|
class TorrentLeechProvider(generic.TorrentProvider):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'TorrentLeech')
|
generic.TorrentProvider.__init__(self, 'TorrentLeech', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'https://torrentleech.org/'
|
self.url_base = 'https://torrentleech.org/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
|
@ -40,7 +40,6 @@ class TorrentLeechProvider(generic.TorrentProvider):
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
||||||
self.cache = TorrentLeechCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -111,14 +110,5 @@ class TorrentLeechProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='|', **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='|', **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class TorrentLeechCache(tvcache.TVCache):
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = TorrentLeechProvider()
|
provider = TorrentLeechProvider()
|
||||||
|
|
|
@ -21,7 +21,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
@ -30,13 +30,14 @@ from lib.unidecode import unidecode
|
||||||
class TorrentShackProvider(generic.TorrentProvider):
|
class TorrentShackProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'TorrentShack')
|
generic.TorrentProvider.__init__(self, 'TorrentShack', cache_update_freq=20)
|
||||||
|
|
||||||
self.url_base = 'https://torrentshack.me/'
|
self.url_base = 'https://torrentshack.me/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
'login': self.url_base + 'login.php?lang=',
|
'login': self.url_base + 'login.php?lang=',
|
||||||
'search': self.url_base + 'torrents.php?searchstr=%s&%s&' + '&'.join(
|
'search': self.url_base + 'torrents.php?searchstr=%s&%s&' + '&'.join(
|
||||||
['release_type=both', 'searchtags=', 'tags_type=0', 'order_by=s3', 'order_way=desc', 'torrent_preset=all']),
|
['release_type=both', 'searchtags=', 'tags_type=0',
|
||||||
|
'order_by=s3', 'order_way=desc', 'torrent_preset=all']),
|
||||||
'get': self.url_base + '%s'}
|
'get': self.url_base + '%s'}
|
||||||
|
|
||||||
self.categories = {'shows': [600, 620, 700, 981, 980], 'anime': [850]}
|
self.categories = {'shows': [600, 620, 700, 981, 980], 'anime': [850]}
|
||||||
|
@ -44,7 +45,6 @@ class TorrentShackProvider(generic.TorrentProvider):
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
||||||
self.cache = TorrentShackCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -117,16 +117,4 @@ class TorrentShackProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='.', **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, sep_date='.', **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class TorrentShackCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 20 # cache update frequency
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = TorrentShackProvider()
|
provider = TorrentShackProvider()
|
||||||
|
|
|
@ -19,7 +19,8 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import helpers, logger, tvcache
|
from sickbeard import common, helpers, logger
|
||||||
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from lib.unidecode import unidecode
|
from lib.unidecode import unidecode
|
||||||
|
|
||||||
|
@ -27,7 +28,7 @@ from lib.unidecode import unidecode
|
||||||
class TransmithenetProvider(generic.TorrentProvider):
|
class TransmithenetProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'Transmithe.net')
|
generic.TorrentProvider.__init__(self, 'Transmithe.net', cache_update_freq=17)
|
||||||
|
|
||||||
self.url_base = 'https://transmithe.net/'
|
self.url_base = 'https://transmithe.net/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
|
@ -39,10 +40,9 @@ class TransmithenetProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
self.url = self.urls['config_provider_home_uri']
|
self.url = self.urls['config_provider_home_uri']
|
||||||
self.user_authkey, self.user_passkey = 2 * [None]
|
self.user_authkey, self.user_passkey = 2 * [None]
|
||||||
|
self.chk_td = True
|
||||||
|
|
||||||
self.username, self.password, self.minseed, self.minleech = 4 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.freeleech = False
|
|
||||||
self.cache = TransmithenetCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -88,11 +88,13 @@ class TransmithenetProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
try:
|
try:
|
||||||
title_parts = group_name.split('[')
|
title_parts = group_name.split('[')
|
||||||
maybe_res = re.findall('((?:72|108)0\w)', title_parts[1])
|
maybe_res = re.findall('((?:72|108|216)0\w)', title_parts[1])
|
||||||
|
maybe_ext = re.findall('(?i)(%s)' % '|'.join(common.mediaExtensions), title_parts[1])
|
||||||
detail = title_parts[1].split('/')
|
detail = title_parts[1].split('/')
|
||||||
detail[1] = detail[1].strip().lower().replace('mkv', 'x264')
|
detail[1] = detail[1].strip().lower().replace('mkv', 'x264')
|
||||||
title = '%s.%s' % (title_parts[0].strip(), '.'.join(
|
title = '%s.%s' % (BS4Parser(title_parts[0].strip(), 'html.parser').soup.string, '.'.join(
|
||||||
(len(maybe_res) and [maybe_res[0]] or []) + [detail[0].strip(), detail[1]]))
|
(maybe_res and [maybe_res[0]] or []) +
|
||||||
|
[detail[0].strip(), detail[1], maybe_ext and maybe_ext[0].lower() or 'mkv']))
|
||||||
except (IndexError, KeyError):
|
except (IndexError, KeyError):
|
||||||
title = group_name
|
title = group_name
|
||||||
download_url = self.urls['get'] % (self.user_authkey, self.user_passkey, torrent_id)
|
download_url = self.urls['get'] % (self.user_authkey, self.user_passkey, torrent_id)
|
||||||
|
@ -119,16 +121,4 @@ class TransmithenetProvider(generic.TorrentProvider):
|
||||||
return generic.TorrentProvider._episode_strings(self, ep_obj, scene=False, **kwargs)
|
return generic.TorrentProvider._episode_strings(self, ep_obj, scene=False, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class TransmithenetCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
self.update_freq = 17
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = TransmithenetProvider()
|
provider = TransmithenetProvider()
|
||||||
|
|
|
@ -19,7 +19,7 @@ import re
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import logger
|
||||||
from sickbeard.bs4_parser import BS4Parser
|
from sickbeard.bs4_parser import BS4Parser
|
||||||
from sickbeard.helpers import tryInt
|
from sickbeard.helpers import tryInt
|
||||||
from sickbeard.config import naming_ep_type
|
from sickbeard.config import naming_ep_type
|
||||||
|
@ -32,9 +32,9 @@ class TVChaosUKProvider(generic.TorrentProvider):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
generic.TorrentProvider.__init__(self, 'TVChaosUK')
|
generic.TorrentProvider.__init__(self, 'TVChaosUK')
|
||||||
|
|
||||||
self.url_base = 'https://tvchaosuk.com/'
|
self.url_base = 'https://www.tvchaosuk.com/'
|
||||||
self.urls = {'config_provider_home_uri': self.url_base,
|
self.urls = {'config_provider_home_uri': self.url_base,
|
||||||
'login': self.url_base + 'takelogin.php',
|
'login_action': self.url_base + 'login.php',
|
||||||
'search': self.url_base + 'browse.php',
|
'search': self.url_base + 'browse.php',
|
||||||
'get': self.url_base + '%s'}
|
'get': self.url_base + '%s'}
|
||||||
|
|
||||||
|
@ -42,7 +42,6 @@ class TVChaosUKProvider(generic.TorrentProvider):
|
||||||
|
|
||||||
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
self.username, self.password, self.freeleech, self.minseed, self.minleech = 5 * [None]
|
||||||
self.search_fallback = True
|
self.search_fallback = True
|
||||||
self.cache = TVChaosUKCache(self)
|
|
||||||
|
|
||||||
def _authorised(self, **kwargs):
|
def _authorised(self, **kwargs):
|
||||||
|
|
||||||
|
@ -92,8 +91,9 @@ class TVChaosUKProvider(generic.TorrentProvider):
|
||||||
info = tr.find('a', href=rc['info'])
|
info = tr.find('a', href=rc['info'])
|
||||||
title = (tr.find('div', attrs={'class': 'tooltip-content'}).get_text() or info.get_text()).strip()
|
title = (tr.find('div', attrs={'class': 'tooltip-content'}).get_text() or info.get_text()).strip()
|
||||||
title = re.findall('(?m)(^[^\r\n]+)', title)[0]
|
title = re.findall('(?m)(^[^\r\n]+)', title)[0]
|
||||||
download_url = self.urls['get'] % str(tr.find('a', href=rc['get'])['href']).lstrip(
|
download_url = str(tr.find('a', href=rc['get'])['href'])
|
||||||
'/').replace(self.urls['config_provider_home_uri'], '')
|
if not download_url.startswith('http'):
|
||||||
|
download_url = self.urls['get'] % download_url.lstrip('/')
|
||||||
except Exception:
|
except Exception:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
@ -134,6 +134,7 @@ class TVChaosUKProvider(generic.TorrentProvider):
|
||||||
add_pad = re.findall('((?:19|20)\d\d\-\d\d\-\d\d)([\w\W])', title)
|
add_pad = re.findall('((?:19|20)\d\d\-\d\d\-\d\d)([\w\W])', title)
|
||||||
if len(add_pad) and add_pad[0][1] not in [' ', '.']:
|
if len(add_pad) and add_pad[0][1] not in [' ', '.']:
|
||||||
title = title.replace(''.join(add_pad[0]), '%s %s' % (add_pad[0][0], add_pad[0][1]))
|
title = title.replace(''.join(add_pad[0]), '%s %s' % (add_pad[0][0], add_pad[0][1]))
|
||||||
|
title = re.sub(r'(?sim)(.*?)(?:Episode|Season).\d+.(.*)', r'\1\2', title)
|
||||||
|
|
||||||
if title and download_url:
|
if title and download_url:
|
||||||
items[mode].append((title, download_url, seeders, self._bytesizer(size)))
|
items[mode].append((title, download_url, seeders, self._bytesizer(size)))
|
||||||
|
@ -176,14 +177,4 @@ class TVChaosUKProvider(generic.TorrentProvider):
|
||||||
return 'tvchaosuk_tip' == key and 'has missing quality data so you must add quality Custom/Unknown to any wanted show' or ''
|
return 'tvchaosuk_tip' == key and 'has missing quality data so you must add quality Custom/Unknown to any wanted show' or ''
|
||||||
|
|
||||||
|
|
||||||
class TVChaosUKCache(tvcache.TVCache):
|
|
||||||
|
|
||||||
def __init__(self, this_provider):
|
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
|
||||||
|
|
||||||
def _cache_data(self):
|
|
||||||
|
|
||||||
return self.provider.cache_data()
|
|
||||||
|
|
||||||
|
|
||||||
provider = TVChaosUKProvider()
|
provider = TVChaosUKProvider()
|
||||||
|
|
|
@ -17,7 +17,8 @@
|
||||||
# along with SickGear. If not, see <http://www.gnu.org/licenses/>.
|
# along with SickGear. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
from . import generic
|
from . import generic
|
||||||
from sickbeard import logger, tvcache
|
from sickbeard import tvcache
|
||||||
|
import time
|
||||||
|
|
||||||
|
|
||||||
class WombleProvider(generic.NZBProvider):
|
class WombleProvider(generic.NZBProvider):
|
||||||
|
@ -34,46 +35,23 @@ class WombleCache(tvcache.TVCache):
|
||||||
def __init__(self, this_provider):
|
def __init__(self, this_provider):
|
||||||
tvcache.TVCache.__init__(self, this_provider)
|
tvcache.TVCache.__init__(self, this_provider)
|
||||||
|
|
||||||
self.update_freq = 15 # cache update frequency
|
self.update_freq = 6 # cache update frequency
|
||||||
|
|
||||||
def updateCache(self):
|
def _cache_data(self):
|
||||||
|
|
||||||
# delete anything older then 7 days
|
result = []
|
||||||
self._clearCache()
|
for section in ['sd', 'hd', 'x264', 'dvd']:
|
||||||
|
url = '%srss/?sec=tv-%s&fr=false' % (self.provider.url, section)
|
||||||
if not self.shouldUpdate():
|
|
||||||
return
|
|
||||||
|
|
||||||
cl = []
|
|
||||||
data = None
|
|
||||||
for url in [self.provider.url + 'rss/?sec=tv-x264&fr=false',
|
|
||||||
self.provider.url + 'rss/?sec=tv-sd&fr=false',
|
|
||||||
self.provider.url + 'rss/?sec=tv-dvd&fr=false',
|
|
||||||
self.provider.url + 'rss/?sec=tv-hd&fr=false']:
|
|
||||||
logger.log(u'Womble\'s Index cache update URL: ' + url, logger.DEBUG)
|
|
||||||
data = self.getRSSFeed(url)
|
data = self.getRSSFeed(url)
|
||||||
|
time.sleep(1.1)
|
||||||
|
cnt = len(result)
|
||||||
|
for entry in (data and data.get('entries', []) or []):
|
||||||
|
if entry.get('title') and entry.get('link', '').startswith('http'):
|
||||||
|
result.append((entry['title'], entry['link'], None, None))
|
||||||
|
|
||||||
# As long as we got something from the provider we count it as an update
|
self.provider.log_result(count=len(result) - cnt, url=url)
|
||||||
if not data:
|
|
||||||
return []
|
|
||||||
|
|
||||||
# By now we know we've got data and no auth errors, all we need to do is put it in the database
|
return result
|
||||||
for item in data.entries:
|
|
||||||
title, url = self._title_and_url(item)
|
|
||||||
ci = self._parseItem(title, url)
|
|
||||||
if None is not ci:
|
|
||||||
cl.append(ci)
|
|
||||||
|
|
||||||
if 0 < len(cl):
|
|
||||||
my_db = self.get_db()
|
|
||||||
my_db.mass_action(cl)
|
|
||||||
|
|
||||||
# set last updated
|
|
||||||
if data:
|
|
||||||
self.setLastUpdate()
|
|
||||||
|
|
||||||
def _checkAuth(self, *data):
|
|
||||||
return 'Invalid Link' != data[0]
|
|
||||||
|
|
||||||
|
|
||||||
provider = WombleProvider()
|
provider = WombleProvider()
|
||||||
|
|
|
@ -179,6 +179,21 @@ def snatch_episode(result, end_status=SNATCHED):
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def pass_show_wordlist_checks(name, show):
|
||||||
|
re_extras = dict(re_prefix='.*', re_suffix='.*')
|
||||||
|
result = show_name_helpers.contains_any(name, show.rls_ignore_words, **re_extras)
|
||||||
|
if None is not result and result:
|
||||||
|
logger.log(u'Ignored: %s for containing ignore word' % name)
|
||||||
|
return False
|
||||||
|
|
||||||
|
result = show_name_helpers.contains_any(name, show.rls_require_words, **re_extras)
|
||||||
|
if None is not result and not result:
|
||||||
|
logger.log(u'Ignored: %s for not containing any required word match' % name)
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
def pick_best_result(results, show, quality_list=None):
|
def pick_best_result(results, show, quality_list=None):
|
||||||
logger.log(u'Picking the best result out of %s' % [x.name for x in results], logger.DEBUG)
|
logger.log(u'Picking the best result out of %s' % [x.name for x in results], logger.DEBUG)
|
||||||
|
|
||||||
|
@ -195,15 +210,7 @@ def pick_best_result(results, show, quality_list=None):
|
||||||
logger.log(u'%s is an unwanted quality, rejecting it' % cur_result.name, logger.DEBUG)
|
logger.log(u'%s is an unwanted quality, rejecting it' % cur_result.name, logger.DEBUG)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
re_extras = dict(re_prefix='.*', re_suffix='.*')
|
if not pass_show_wordlist_checks(cur_result.name, show):
|
||||||
result = show_name_helpers.contains_any(cur_result.name, show.rls_ignore_words, **re_extras)
|
|
||||||
if None is not result and result:
|
|
||||||
logger.log(u'Ignored: %s for containing ignore word' % cur_result.name)
|
|
||||||
continue
|
|
||||||
|
|
||||||
result = show_name_helpers.contains_any(cur_result.name, show.rls_require_words, **re_extras)
|
|
||||||
if None is not result and not result:
|
|
||||||
logger.log(u'Ignored: %s for not containing any required word match' % cur_result.name)
|
|
||||||
continue
|
continue
|
||||||
|
|
||||||
cur_size = getattr(cur_result, 'size', None)
|
cur_size = getattr(cur_result, 'size', None)
|
||||||
|
@ -427,8 +434,11 @@ def search_for_needed_episodes(episodes):
|
||||||
|
|
||||||
threading.currentThread().name = orig_thread_name
|
threading.currentThread().name = orig_thread_name
|
||||||
|
|
||||||
if not search_done:
|
if not len(providers):
|
||||||
logger.log(u'No NZB/Torrent provider enabled to do recent searches. Please check provider options.', logger.ERROR)
|
logger.log('No NZB/Torrent sources enabled in Search Provider options to do recent searches', logger.WARNING)
|
||||||
|
elif not search_done:
|
||||||
|
logger.log('Failed recent search of %s enabled provider%s. More info in debug log.' % (
|
||||||
|
len(providers), helpers.maybe_plural(len(providers))), logger.ERROR)
|
||||||
|
|
||||||
return found_results.values()
|
return found_results.values()
|
||||||
|
|
||||||
|
@ -672,12 +682,39 @@ def search_providers(show, episodes, manual_search=False):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# filter out possible bad torrents from providers
|
# filter out possible bad torrents from providers
|
||||||
if 'torrent' == best_result.resultType and 'blackhole' != sickbeard.TORRENT_METHOD:
|
if 'torrent' == best_result.resultType:
|
||||||
best_result.content = None
|
if best_result.url.startswith('magnet'):
|
||||||
if not best_result.url.startswith('magnet'):
|
if 'blackhole' != sickbeard.TORRENT_METHOD:
|
||||||
best_result.content = best_result.provider.get_url(best_result.url)
|
best_result.content = None
|
||||||
if not best_result.content:
|
else:
|
||||||
|
td = best_result.provider.get_url(best_result.url)
|
||||||
|
if not td:
|
||||||
continue
|
continue
|
||||||
|
if getattr(best_result.provider, 'chk_td', None):
|
||||||
|
name = None
|
||||||
|
try:
|
||||||
|
hdr = re.findall('(\w+(\d+):)', td[0:6])[0]
|
||||||
|
x, v = len(hdr[0]), int(hdr[1])
|
||||||
|
for item in range(0, 12):
|
||||||
|
y = x + v
|
||||||
|
name = 'name' == td[x: y]
|
||||||
|
w = re.findall('((?:i\d+e|d|l)?(\d+):)', td[y: y + 32])[0]
|
||||||
|
x, v = y + len(w[0]), int(w[1])
|
||||||
|
if name:
|
||||||
|
name = td[x: x + v]
|
||||||
|
break
|
||||||
|
except:
|
||||||
|
continue
|
||||||
|
if name:
|
||||||
|
if not pass_show_wordlist_checks(name, show):
|
||||||
|
continue
|
||||||
|
if not show_name_helpers.pass_wordlist_checks(name):
|
||||||
|
logger.log(u'Ignored: %s (debug log has detail)' % name)
|
||||||
|
continue
|
||||||
|
best_result.name = name
|
||||||
|
|
||||||
|
if 'blackhole' != sickbeard.TORRENT_METHOD:
|
||||||
|
best_result.content = td
|
||||||
|
|
||||||
# add result if its not a duplicate and
|
# add result if its not a duplicate and
|
||||||
found = False
|
found = False
|
||||||
|
@ -702,8 +739,10 @@ def search_providers(show, episodes, manual_search=False):
|
||||||
if len(episodes) == wanted_ep_count:
|
if len(episodes) == wanted_ep_count:
|
||||||
break
|
break
|
||||||
|
|
||||||
if not search_done:
|
if not len(provider_list):
|
||||||
logger.log(u'No NZB/Torrent providers found or enabled in the SickGear config for backlog searches. Please check your settings.',
|
logger.log('No NZB/Torrent sources enabled in Search Provider options to do backlog searches', logger.WARNING)
|
||||||
logger.ERROR)
|
elif not search_done:
|
||||||
|
logger.log('Failed backlog search of %s enabled provider%s. More info in debug log.' % (
|
||||||
|
len(provider_list), helpers.maybe_plural(len(provider_list))), logger.ERROR)
|
||||||
|
|
||||||
return final_results
|
return final_results
|
||||||
|
|
|
@ -18,7 +18,6 @@
|
||||||
|
|
||||||
from __future__ import with_statement
|
from __future__ import with_statement
|
||||||
|
|
||||||
import time
|
|
||||||
import traceback
|
import traceback
|
||||||
import threading
|
import threading
|
||||||
import datetime
|
import datetime
|
||||||
|
@ -288,21 +287,29 @@ class RecentSearchQueueItem(generic_queue.QueueItem):
|
||||||
orig_thread_name = threading.currentThread().name
|
orig_thread_name = threading.currentThread().name
|
||||||
threads = []
|
threads = []
|
||||||
|
|
||||||
logger.log('Updating provider caches with recent upload data')
|
|
||||||
|
|
||||||
providers = [x for x in sickbeard.providers.sortedProviderList() if x.is_active() and x.enable_recentsearch]
|
providers = [x for x in sickbeard.providers.sortedProviderList() if x.is_active() and x.enable_recentsearch]
|
||||||
for cur_provider in providers:
|
for cur_provider in providers:
|
||||||
# spawn separate threads for each provider so we don't need to wait for providers with slow network operation
|
if not cur_provider.cache.should_update():
|
||||||
|
continue
|
||||||
|
|
||||||
|
if not threads:
|
||||||
|
logger.log('Updating provider caches with recent upload data')
|
||||||
|
|
||||||
|
# spawn a thread for each provider to save time waiting for slow response providers
|
||||||
threads.append(threading.Thread(target=cur_provider.cache.updateCache,
|
threads.append(threading.Thread(target=cur_provider.cache.updateCache,
|
||||||
name='%s :: [%s]' % (orig_thread_name, cur_provider.name)))
|
name='%s :: [%s]' % (orig_thread_name, cur_provider.name)))
|
||||||
# start the thread we just created
|
# start the thread we just created
|
||||||
threads[-1].start()
|
threads[-1].start()
|
||||||
|
|
||||||
# wait for all threads to finish
|
if not len(providers):
|
||||||
for t in threads:
|
logger.log('No NZB/Torrent sources enabled in Search Provider options for cache update', logger.WARNING)
|
||||||
t.join()
|
|
||||||
|
|
||||||
logger.log('Finished updating provider caches')
|
if threads:
|
||||||
|
# wait for all threads to finish
|
||||||
|
for t in threads:
|
||||||
|
t.join()
|
||||||
|
|
||||||
|
logger.log('Finished updating provider caches')
|
||||||
|
|
||||||
|
|
||||||
class ProperSearchQueueItem(generic_queue.QueueItem):
|
class ProperSearchQueueItem(generic_queue.QueueItem):
|
||||||
|
@ -427,7 +434,7 @@ class FailedQueueItem(generic_queue.QueueItem):
|
||||||
history.logFailed(epObj, release, provider)
|
history.logFailed(epObj, release, provider)
|
||||||
|
|
||||||
failed_history.revertEpisode(epObj)
|
failed_history.revertEpisode(epObj)
|
||||||
logger.log(u'Beginning failed download search for: []' % epObj.prettyName())
|
logger.log(u'Beginning failed download search for: [%s]' % epObj.prettyName())
|
||||||
|
|
||||||
search_result = search.search_providers(self.show, self.segment, True)
|
search_result = search.search_providers(self.show, self.segment, True)
|
||||||
|
|
||||||
|
|
|
@ -32,6 +32,7 @@ from name_parser.parser import NameParser, InvalidNameException, InvalidShowExce
|
||||||
from sickbeard.rssfeeds import RSSFeeds
|
from sickbeard.rssfeeds import RSSFeeds
|
||||||
import itertools
|
import itertools
|
||||||
|
|
||||||
|
|
||||||
class CacheDBConnection(db.DBConnection):
|
class CacheDBConnection(db.DBConnection):
|
||||||
def __init__(self, providerName):
|
def __init__(self, providerName):
|
||||||
db.DBConnection.__init__(self, 'cache.db')
|
db.DBConnection.__init__(self, 'cache.db')
|
||||||
|
@ -44,6 +45,7 @@ class CacheDBConnection(db.DBConnection):
|
||||||
if str(e) != 'table lastUpdate already exists':
|
if str(e) != 'table lastUpdate already exists':
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
|
||||||
class TVCache:
|
class TVCache:
|
||||||
def __init__(self, provider):
|
def __init__(self, provider):
|
||||||
|
|
||||||
|
@ -56,7 +58,7 @@ class TVCache:
|
||||||
return CacheDBConnection(self.providerID)
|
return CacheDBConnection(self.providerID)
|
||||||
|
|
||||||
def _clearCache(self):
|
def _clearCache(self):
|
||||||
if self.shouldClearCache():
|
if self.should_clear_cache():
|
||||||
myDB = self.get_db()
|
myDB = self.get_db()
|
||||||
myDB.action('DELETE FROM provider_cache WHERE provider = ?', [self.providerID])
|
myDB.action('DELETE FROM provider_cache WHERE provider = ?', [self.providerID])
|
||||||
|
|
||||||
|
@ -81,21 +83,16 @@ class TVCache:
|
||||||
logger.log(u'Authentication error: ' + ex(e), logger.ERROR)
|
logger.log(u'Authentication error: ' + ex(e), logger.ERROR)
|
||||||
return []
|
return []
|
||||||
|
|
||||||
if self.shouldUpdate():
|
if self.should_update():
|
||||||
# as long as the http request worked we count this as an update
|
|
||||||
data = self._cache_data()
|
data = self._cache_data()
|
||||||
if not data:
|
|
||||||
return []
|
|
||||||
|
|
||||||
# clear cache
|
# clear cache
|
||||||
self._clearCache()
|
if data:
|
||||||
|
self._clearCache()
|
||||||
# set updated
|
|
||||||
self.setLastUpdate()
|
|
||||||
|
|
||||||
# parse data
|
# parse data
|
||||||
cl = []
|
cl = []
|
||||||
for item in data:
|
for item in data or []:
|
||||||
title, url = self._title_and_url(item)
|
title, url = self._title_and_url(item)
|
||||||
ci = self._parseItem(title, url)
|
ci = self._parseItem(title, url)
|
||||||
if ci is not None:
|
if ci is not None:
|
||||||
|
@ -105,6 +102,9 @@ class TVCache:
|
||||||
myDB = self.get_db()
|
myDB = self.get_db()
|
||||||
myDB.mass_action(cl)
|
myDB.mass_action(cl)
|
||||||
|
|
||||||
|
# set updated as time the attempt to fetch data is
|
||||||
|
self.setLastUpdate()
|
||||||
|
|
||||||
return []
|
return []
|
||||||
|
|
||||||
def getRSSFeed(self, url, **kwargs):
|
def getRSSFeed(self, url, **kwargs):
|
||||||
|
@ -180,21 +180,13 @@ class TVCache:
|
||||||
lastUpdate = property(_getLastUpdate)
|
lastUpdate = property(_getLastUpdate)
|
||||||
lastSearch = property(_getLastSearch)
|
lastSearch = property(_getLastSearch)
|
||||||
|
|
||||||
def shouldUpdate(self):
|
def should_update(self):
|
||||||
# if we've updated recently then skip the update
|
# if we've updated recently then skip the update
|
||||||
if datetime.datetime.today() - self.lastUpdate < datetime.timedelta(minutes=self.update_freq):
|
return datetime.datetime.today() - self.lastUpdate >= datetime.timedelta(minutes=self.update_freq)
|
||||||
logger.log(u'Last update was too soon, using old cache: today()-' + str(self.lastUpdate) + '<' + str(
|
|
||||||
datetime.timedelta(minutes=self.update_freq)), logger.DEBUG)
|
|
||||||
return False
|
|
||||||
|
|
||||||
return True
|
def should_clear_cache(self):
|
||||||
|
|
||||||
def shouldClearCache(self):
|
|
||||||
# if recent search hasn't used our previous results yet then don't clear the cache
|
# if recent search hasn't used our previous results yet then don't clear the cache
|
||||||
if self.lastUpdate > self.lastSearch:
|
return self.lastSearch >= self.lastUpdate
|
||||||
return False
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
def add_cache_entry(self, name, url, parse_result=None, indexer_id=0):
|
def add_cache_entry(self, name, url, parse_result=None, indexer_id=0):
|
||||||
|
|
||||||
|
|
|
@ -4662,10 +4662,7 @@ class ConfigProviders(Config):
|
||||||
|
|
||||||
providerDict[name].key = key
|
providerDict[name].key = key
|
||||||
# a 0 in the key spot indicates that no key is needed
|
# a 0 in the key spot indicates that no key is needed
|
||||||
if key == '0':
|
providerDict[name].needs_auth = '0' != key
|
||||||
providerDict[name].needs_auth = False
|
|
||||||
else:
|
|
||||||
providerDict[name].needs_auth = True
|
|
||||||
|
|
||||||
return providerDict[name].get_id() + '|' + providerDict[name].config_str()
|
return providerDict[name].get_id() + '|' + providerDict[name].config_str()
|
||||||
|
|
||||||
|
@ -4767,16 +4764,11 @@ class ConfigProviders(Config):
|
||||||
def saveProviders(self, newznab_string='', torrentrss_string='', provider_order=None, **kwargs):
|
def saveProviders(self, newznab_string='', torrentrss_string='', provider_order=None, **kwargs):
|
||||||
|
|
||||||
results = []
|
results = []
|
||||||
|
|
||||||
provider_str_list = provider_order.split()
|
|
||||||
provider_list = []
|
provider_list = []
|
||||||
|
|
||||||
newznabProviderDict = dict(
|
# add all the newznab info we have into our list
|
||||||
zip([x.get_id() for x in sickbeard.newznabProviderList], sickbeard.newznabProviderList))
|
newznab_sources = dict(zip([x.get_id() for x in sickbeard.newznabProviderList], sickbeard.newznabProviderList))
|
||||||
|
active_ids = []
|
||||||
finishedNames = []
|
|
||||||
|
|
||||||
# add all the newznab info we got into our list
|
|
||||||
if newznab_string:
|
if newznab_string:
|
||||||
for curNewznabProviderStr in newznab_string.split('!!!'):
|
for curNewznabProviderStr in newznab_string.split('!!!'):
|
||||||
|
|
||||||
|
@ -4789,282 +4781,157 @@ class ConfigProviders(Config):
|
||||||
if starify(cur_key, True):
|
if starify(cur_key, True):
|
||||||
cur_key = ''
|
cur_key = ''
|
||||||
|
|
||||||
newProvider = newznab.NewznabProvider(cur_name, cur_url, key=cur_key)
|
new_provider = newznab.NewznabProvider(cur_name, cur_url, key=cur_key)
|
||||||
|
|
||||||
cur_id = newProvider.get_id()
|
cur_id = new_provider.get_id()
|
||||||
|
|
||||||
# if it already exists then update it
|
# if it already exists then update it
|
||||||
if cur_id in newznabProviderDict:
|
if cur_id in newznab_sources:
|
||||||
newznabProviderDict[cur_id].name = cur_name
|
nzb_src = newznab_sources[cur_id]
|
||||||
newznabProviderDict[cur_id].url = cur_url
|
|
||||||
|
nzb_src.name, nzb_src.url, nzb_src.cat_ids = cur_name, cur_url, cur_cat
|
||||||
|
|
||||||
if cur_key:
|
if cur_key:
|
||||||
newznabProviderDict[cur_id].key = cur_key
|
nzb_src.key = cur_key
|
||||||
newznabProviderDict[cur_id].cat_ids = cur_cat
|
|
||||||
# a 0 in the key spot indicates that no key is needed
|
# a 0 in the key spot indicates that no key is needed
|
||||||
if cur_key == '0':
|
nzb_src.needs_auth = '0' != cur_key
|
||||||
newznabProviderDict[cur_id].needs_auth = False
|
|
||||||
else:
|
|
||||||
newznabProviderDict[cur_id].needs_auth = True
|
|
||||||
|
|
||||||
try:
|
attr = 'search_mode'
|
||||||
newznabProviderDict[cur_id].search_mode = str(kwargs[cur_id + '_search_mode']).strip()
|
if cur_id + '_' + attr in kwargs:
|
||||||
except:
|
setattr(nzb_src, attr, str(kwargs.get(cur_id + '_' + attr)).strip())
|
||||||
pass
|
|
||||||
|
|
||||||
try:
|
for attr in ['search_fallback', 'enable_recentsearch', 'enable_backlog']:
|
||||||
newznabProviderDict[cur_id].search_fallback = config.checkbox_to_value(
|
setattr(nzb_src, attr, config.checkbox_to_value(kwargs.get(cur_id + '_' + attr)))
|
||||||
kwargs[cur_id + '_search_fallback'])
|
|
||||||
except:
|
|
||||||
newznabProviderDict[cur_id].search_fallback = 0
|
|
||||||
|
|
||||||
try:
|
|
||||||
newznabProviderDict[cur_id].enable_recentsearch = config.checkbox_to_value(
|
|
||||||
kwargs[cur_id + '_enable_recentsearch'])
|
|
||||||
except:
|
|
||||||
newznabProviderDict[cur_id].enable_recentsearch = 0
|
|
||||||
|
|
||||||
try:
|
|
||||||
newznabProviderDict[cur_id].enable_backlog = config.checkbox_to_value(
|
|
||||||
kwargs[cur_id + '_enable_backlog'])
|
|
||||||
except:
|
|
||||||
newznabProviderDict[cur_id].enable_backlog = 0
|
|
||||||
else:
|
else:
|
||||||
sickbeard.newznabProviderList.append(newProvider)
|
sickbeard.newznabProviderList.append(new_provider)
|
||||||
|
|
||||||
finishedNames.append(cur_id)
|
active_ids.append(cur_id)
|
||||||
|
|
||||||
# delete anything that is missing
|
# delete anything that is missing
|
||||||
for curProvider in sickbeard.newznabProviderList:
|
for source in [x for x in sickbeard.newznabProviderList if x.get_id() not in active_ids]:
|
||||||
if curProvider.get_id() not in finishedNames:
|
sickbeard.newznabProviderList.remove(source)
|
||||||
sickbeard.newznabProviderList.remove(curProvider)
|
|
||||||
|
|
||||||
torrentRssProviderDict = dict(
|
|
||||||
zip([x.get_id() for x in sickbeard.torrentRssProviderList], sickbeard.torrentRssProviderList))
|
|
||||||
finishedNames = []
|
|
||||||
|
|
||||||
|
# add all the torrent RSS info we have into our list
|
||||||
|
torrent_rss_sources = dict(zip([x.get_id() for x in sickbeard.torrentRssProviderList],
|
||||||
|
sickbeard.torrentRssProviderList))
|
||||||
|
active_ids = []
|
||||||
if torrentrss_string:
|
if torrentrss_string:
|
||||||
for curTorrentRssProviderStr in torrentrss_string.split('!!!'):
|
for curTorrentRssProviderStr in torrentrss_string.split('!!!'):
|
||||||
|
|
||||||
if not curTorrentRssProviderStr:
|
if not curTorrentRssProviderStr:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
curName, curURL, curCookies = curTorrentRssProviderStr.split('|')
|
cur_name, cur_url, cur_cookies = curTorrentRssProviderStr.split('|')
|
||||||
curURL = config.clean_url(curURL, False)
|
cur_url = config.clean_url(cur_url, False)
|
||||||
|
|
||||||
if starify(curCookies, True):
|
if starify(cur_cookies, True):
|
||||||
curCookies = ''
|
cur_cookies = ''
|
||||||
|
|
||||||
newProvider = rsstorrent.TorrentRssProvider(curName, curURL, curCookies)
|
new_provider = rsstorrent.TorrentRssProvider(cur_name, cur_url, cur_cookies)
|
||||||
|
|
||||||
curID = newProvider.get_id()
|
cur_id = new_provider.get_id()
|
||||||
|
|
||||||
# if it already exists then update it
|
# if it already exists then update it
|
||||||
if curID in torrentRssProviderDict:
|
if cur_id in torrent_rss_sources:
|
||||||
torrentRssProviderDict[curID].name = curName
|
torrent_rss_sources[cur_id].name = cur_name
|
||||||
torrentRssProviderDict[curID].url = curURL
|
torrent_rss_sources[cur_id].url = cur_url
|
||||||
if curCookies:
|
if cur_cookies:
|
||||||
torrentRssProviderDict[curID].cookies = curCookies
|
torrent_rss_sources[cur_id].cookies = cur_cookies
|
||||||
else:
|
else:
|
||||||
sickbeard.torrentRssProviderList.append(newProvider)
|
sickbeard.torrentRssProviderList.append(new_provider)
|
||||||
|
|
||||||
finishedNames.append(curID)
|
active_ids.append(cur_id)
|
||||||
|
|
||||||
# delete anything that is missing
|
# delete anything that is missing
|
||||||
for curProvider in sickbeard.torrentRssProviderList:
|
for source in [x for x in sickbeard.torrentRssProviderList if x.get_id() not in active_ids]:
|
||||||
if curProvider.get_id() not in finishedNames:
|
sickbeard.torrentRssProviderList.remove(source)
|
||||||
sickbeard.torrentRssProviderList.remove(curProvider)
|
|
||||||
|
|
||||||
# do the enable/disable
|
# enable/disable states of source providers
|
||||||
for curProviderStr in provider_str_list:
|
provider_str_list = provider_order.split()
|
||||||
curProvider, curEnabled = curProviderStr.split(':')
|
sources = dict(zip([x.get_id() for x in sickbeard.providers.sortedProviderList()],
|
||||||
curEnabled = config.to_int(curEnabled)
|
sickbeard.providers.sortedProviderList()))
|
||||||
|
for cur_src_str in provider_str_list:
|
||||||
|
src_name, src_enabled = cur_src_str.split(':')
|
||||||
|
|
||||||
curProvObj = [x for x in sickbeard.providers.sortedProviderList() if
|
provider_list.append(src_name)
|
||||||
x.get_id() == curProvider and hasattr(x, 'enabled')]
|
src_enabled = bool(config.to_int(src_enabled))
|
||||||
if curProvObj:
|
|
||||||
curProvObj[0].enabled = bool(curEnabled)
|
|
||||||
|
|
||||||
provider_list.append(curProvider)
|
if src_name in sources and hasattr(sources[src_name], 'enabled'):
|
||||||
if curProvider in newznabProviderDict:
|
sources[src_name].enabled = src_enabled
|
||||||
newznabProviderDict[curProvider].enabled = bool(curEnabled)
|
|
||||||
elif curProvider in torrentRssProviderDict:
|
|
||||||
torrentRssProviderDict[curProvider].enabled = bool(curEnabled)
|
|
||||||
|
|
||||||
# dynamically load provider settings
|
if src_name in newznab_sources:
|
||||||
for curTorrentProvider in [curProvider for curProvider in sickbeard.providers.sortedProviderList() if
|
newznab_sources[src_name].enabled = src_enabled
|
||||||
curProvider.providerType == sickbeard.GenericProvider.TORRENT]:
|
elif src_name in torrent_rss_sources:
|
||||||
|
torrent_rss_sources[src_name].enabled = src_enabled
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, '_seed_ratio'):
|
# update torrent source settings
|
||||||
try:
|
for torrent_src in [src for src in sickbeard.providers.sortedProviderList()
|
||||||
curTorrentProvider._seed_ratio = str(kwargs[curTorrentProvider.get_id() + '_ratio']).strip()
|
if sickbeard.GenericProvider.TORRENT == src.providerType]:
|
||||||
except:
|
src_id_prefix = torrent_src.get_id() + '_'
|
||||||
curTorrentProvider._seed_ratio = None
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'seed_time') and curTorrentProvider.get_id() + '_seed_time' in kwargs:
|
attr = 'url_edit'
|
||||||
curTorrentProvider.seed_time = config.to_int(str(kwargs[curTorrentProvider.get_id() + '_seed_time']).strip(), 0)
|
if getattr(torrent_src, attr, None):
|
||||||
|
url_edit = ','.join(set(['%s' % url.strip() for url in kwargs.get(
|
||||||
|
src_id_prefix + attr, '').split(',')]))
|
||||||
|
torrent_src.url_home = ([url_edit], [])[not url_edit]
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'minseed'):
|
for attr in [x for x in ['username', 'uid'] if hasattr(torrent_src, x)]:
|
||||||
try:
|
setattr(torrent_src, attr, str(kwargs.get(src_id_prefix + attr, '')).strip())
|
||||||
curTorrentProvider.minseed = int(str(kwargs[curTorrentProvider.get_id() + '_minseed']).strip())
|
|
||||||
except:
|
|
||||||
curTorrentProvider.minseed = 0
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'minleech'):
|
for attr in [x for x in ['password', 'api_key', 'passkey', 'digest', 'hash'] if hasattr(torrent_src, x)]:
|
||||||
try:
|
key = str(kwargs.get(src_id_prefix + attr, '')).strip()
|
||||||
curTorrentProvider.minleech = int(str(kwargs[curTorrentProvider.get_id() + '_minleech']).strip())
|
if 'password' == attr:
|
||||||
except:
|
set('*') != set(key) and setattr(torrent_src, attr, key)
|
||||||
curTorrentProvider.minleech = 0
|
elif not starify(key, True):
|
||||||
|
setattr(torrent_src, attr, key)
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'digest'):
|
attr = 'ratio'
|
||||||
try:
|
if hasattr(torrent_src, '_seed_' + attr):
|
||||||
key = str(kwargs[curTorrentProvider.get_id() + '_digest']).strip()
|
setattr(torrent_src, '_seed_' + attr, kwargs.get(src_id_prefix + attr, '').strip() or None)
|
||||||
if not starify(key, True):
|
|
||||||
curTorrentProvider.digest = key
|
|
||||||
except:
|
|
||||||
curTorrentProvider.digest = None
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'hash'):
|
for attr in [x for x in ['minseed', 'minleech'] if hasattr(torrent_src, x)]:
|
||||||
try:
|
setattr(torrent_src, attr, config.to_int(str(kwargs.get(src_id_prefix + attr)).strip()))
|
||||||
key = str(kwargs[curTorrentProvider.get_id() + '_hash']).strip()
|
|
||||||
if not starify(key, True):
|
|
||||||
curTorrentProvider.hash = key
|
|
||||||
except:
|
|
||||||
curTorrentProvider.hash = None
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'api_key'):
|
for attr in [x for x in ['confirmed', 'freeleech', 'reject_m2ts', 'enable_recentsearch',
|
||||||
try:
|
'enable_backlog', 'search_fallback'] if hasattr(torrent_src, x)]:
|
||||||
key = str(kwargs[curTorrentProvider.get_id() + '_api_key']).strip()
|
setattr(torrent_src, attr, config.checkbox_to_value(kwargs.get(src_id_prefix + attr)))
|
||||||
if not starify(key, True):
|
|
||||||
curTorrentProvider.api_key = key
|
|
||||||
except:
|
|
||||||
curTorrentProvider.api_key = None
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'username'):
|
attr = 'seed_time'
|
||||||
try:
|
if hasattr(torrent_src, attr) and src_id_prefix + attr in kwargs:
|
||||||
curTorrentProvider.username = str(kwargs[curTorrentProvider.get_id() + '_username']).strip()
|
setattr(torrent_src, attr, config.to_int(str(kwargs.get(src_id_prefix + attr)).strip()))
|
||||||
except:
|
|
||||||
curTorrentProvider.username = None
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'password'):
|
attr = 'search_mode'
|
||||||
try:
|
if hasattr(torrent_src, attr):
|
||||||
key = str(kwargs[curTorrentProvider.get_id() + '_password']).strip()
|
setattr(torrent_src, attr, str(kwargs.get(src_id_prefix + attr, '')).strip() or 'eponly')
|
||||||
if set('*') != set(key):
|
|
||||||
curTorrentProvider.password = key
|
|
||||||
except:
|
|
||||||
curTorrentProvider.password = None
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'passkey'):
|
# update nzb source settings
|
||||||
try:
|
for nzb_src in [src for src in sickbeard.providers.sortedProviderList() if
|
||||||
key = str(kwargs[curTorrentProvider.get_id() + '_passkey']).strip()
|
sickbeard.GenericProvider.NZB == src.providerType]:
|
||||||
if not starify(key, True):
|
src_id_prefix = nzb_src.get_id() + '_'
|
||||||
curTorrentProvider.passkey = key
|
|
||||||
except:
|
|
||||||
curTorrentProvider.passkey = None
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'confirmed'):
|
attr = 'api_key'
|
||||||
try:
|
if hasattr(nzb_src, attr):
|
||||||
curTorrentProvider.confirmed = config.checkbox_to_value(
|
key = str(kwargs.get(src_id_prefix + attr, '')).strip()
|
||||||
kwargs[curTorrentProvider.get_id() + '_confirmed'])
|
if not starify(key, True):
|
||||||
except:
|
setattr(nzb_src, attr, key)
|
||||||
curTorrentProvider.confirmed = 0
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'proxy'):
|
attr = 'username'
|
||||||
try:
|
if hasattr(nzb_src, attr):
|
||||||
curTorrentProvider.proxy.enabled = config.checkbox_to_value(
|
setattr(nzb_src, attr, str(kwargs.get(src_id_prefix + attr, '')).strip() or None)
|
||||||
kwargs[curTorrentProvider.get_id() + '_proxy'])
|
|
||||||
except:
|
|
||||||
curTorrentProvider.proxy.enabled = 0
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider.proxy, 'url'):
|
attr = 'search_mode'
|
||||||
try:
|
if hasattr(nzb_src, attr):
|
||||||
curTorrentProvider.proxy.url = str(kwargs[curTorrentProvider.get_id() + '_proxy_url']).strip()
|
setattr(nzb_src, attr, str(kwargs.get(src_id_prefix + attr, '')).strip() or 'eponly')
|
||||||
except:
|
|
||||||
curTorrentProvider.proxy.url = None
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'freeleech'):
|
attr = 'enable_recentsearch'
|
||||||
try:
|
if hasattr(nzb_src, attr):
|
||||||
curTorrentProvider.freeleech = config.checkbox_to_value(
|
setattr(nzb_src, attr, config.checkbox_to_value(kwargs.get(src_id_prefix + attr)) or
|
||||||
kwargs[curTorrentProvider.get_id() + '_freeleech'])
|
not getattr(nzb_src, 'supports_backlog', True))
|
||||||
except:
|
|
||||||
curTorrentProvider.freeleech = 0
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'reject_m2ts'):
|
for attr in [x for x in ['search_fallback', 'enable_backlog'] if hasattr(nzb_src, x)]:
|
||||||
try:
|
setattr(nzb_src, attr, config.checkbox_to_value(kwargs.get(src_id_prefix + attr)))
|
||||||
curTorrentProvider.reject_m2ts = config.checkbox_to_value(
|
|
||||||
kwargs[curTorrentProvider.get_id() + '_reject_m2ts'])
|
|
||||||
except:
|
|
||||||
curTorrentProvider.reject_m2ts = 0
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'search_mode'):
|
|
||||||
try:
|
|
||||||
curTorrentProvider.search_mode = str(kwargs[curTorrentProvider.get_id() + '_search_mode']).strip()
|
|
||||||
except:
|
|
||||||
curTorrentProvider.search_mode = 'eponly'
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'search_fallback'):
|
|
||||||
try:
|
|
||||||
curTorrentProvider.search_fallback = config.checkbox_to_value(
|
|
||||||
kwargs[curTorrentProvider.get_id() + '_search_fallback'])
|
|
||||||
except:
|
|
||||||
curTorrentProvider.search_fallback = 0 # these exceptions are catching unselected checkboxes
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'enable_recentsearch'):
|
|
||||||
try:
|
|
||||||
curTorrentProvider.enable_recentsearch = config.checkbox_to_value(
|
|
||||||
kwargs[curTorrentProvider.get_id() + '_enable_recentsearch'])
|
|
||||||
except:
|
|
||||||
curTorrentProvider.enable_recentsearch = 0 # these exceptions are actually catching unselected checkboxes
|
|
||||||
|
|
||||||
if hasattr(curTorrentProvider, 'enable_backlog'):
|
|
||||||
try:
|
|
||||||
curTorrentProvider.enable_backlog = config.checkbox_to_value(
|
|
||||||
kwargs[curTorrentProvider.get_id() + '_enable_backlog'])
|
|
||||||
except:
|
|
||||||
curTorrentProvider.enable_backlog = 0 # these exceptions are actually catching unselected checkboxes
|
|
||||||
|
|
||||||
for curNzbProvider in [curProvider for curProvider in sickbeard.providers.sortedProviderList() if
|
|
||||||
curProvider.providerType == sickbeard.GenericProvider.NZB]:
|
|
||||||
|
|
||||||
if hasattr(curNzbProvider, 'api_key'):
|
|
||||||
try:
|
|
||||||
key = str(kwargs[curNzbProvider.get_id() + '_api_key']).strip()
|
|
||||||
if not starify(key, True):
|
|
||||||
curNzbProvider.api_key = key
|
|
||||||
except:
|
|
||||||
curNzbProvider.api_key = None
|
|
||||||
|
|
||||||
if hasattr(curNzbProvider, 'username'):
|
|
||||||
try:
|
|
||||||
curNzbProvider.username = str(kwargs[curNzbProvider.get_id() + '_username']).strip()
|
|
||||||
except:
|
|
||||||
curNzbProvider.username = None
|
|
||||||
|
|
||||||
if hasattr(curNzbProvider, 'search_mode'):
|
|
||||||
try:
|
|
||||||
curNzbProvider.search_mode = str(kwargs[curNzbProvider.get_id() + '_search_mode']).strip()
|
|
||||||
except:
|
|
||||||
curNzbProvider.search_mode = 'eponly'
|
|
||||||
|
|
||||||
if hasattr(curNzbProvider, 'search_fallback'):
|
|
||||||
try:
|
|
||||||
curNzbProvider.search_fallback = config.checkbox_to_value(
|
|
||||||
kwargs[curNzbProvider.get_id() + '_search_fallback'])
|
|
||||||
except:
|
|
||||||
curNzbProvider.search_fallback = 0 # these exceptions are actually catching unselected checkboxes
|
|
||||||
|
|
||||||
if hasattr(curNzbProvider, 'enable_recentsearch'):
|
|
||||||
try:
|
|
||||||
curNzbProvider.enable_recentsearch = config.checkbox_to_value(
|
|
||||||
kwargs[curNzbProvider.get_id() + '_enable_recentsearch'])
|
|
||||||
except:
|
|
||||||
curNzbProvider.enable_recentsearch = 0 # these exceptions are actually catching unselected checkboxes
|
|
||||||
|
|
||||||
if hasattr(curNzbProvider, 'enable_backlog'):
|
|
||||||
try:
|
|
||||||
curNzbProvider.enable_backlog = config.checkbox_to_value(
|
|
||||||
kwargs[curNzbProvider.get_id() + '_enable_backlog'])
|
|
||||||
except:
|
|
||||||
curNzbProvider.enable_backlog = 0 # these exceptions are actually catching unselected checkboxes
|
|
||||||
|
|
||||||
sickbeard.NEWZNAB_DATA = '!!!'.join([x.config_str() for x in sickbeard.newznabProviderList])
|
sickbeard.NEWZNAB_DATA = '!!!'.join([x.config_str() for x in sickbeard.newznabProviderList])
|
||||||
sickbeard.PROVIDER_ORDER = provider_list
|
sickbeard.PROVIDER_ORDER = provider_list
|
||||||
|
@ -5073,11 +4940,10 @@ class ConfigProviders(Config):
|
||||||
|
|
||||||
sickbeard.save_config()
|
sickbeard.save_config()
|
||||||
|
|
||||||
if len(results) > 0:
|
if 0 < len(results):
|
||||||
for x in results:
|
for x in results:
|
||||||
logger.log(x, logger.ERROR)
|
logger.log(x, logger.ERROR)
|
||||||
ui.notifications.error('Error(s) Saving Configuration',
|
ui.notifications.error('Error(s) Saving Configuration', '<br />\n'.join(results))
|
||||||
'<br />\n'.join(results))
|
|
||||||
else:
|
else:
|
||||||
ui.notifications.message('Configuration Saved', ek.ek(os.path.join, sickbeard.CONFIG_FILE))
|
ui.notifications.message('Configuration Saved', ek.ek(os.path.join, sickbeard.CONFIG_FILE))
|
||||||
|
|
||||||
|
|