2014-03-10 05:18:05 +00:00
#
# This file is part of Sick Beard.
#
# Sick Beard 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.
#
# Sick Beard 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 Sick Beard. If not, see <http://www.gnu.org/licenses/>.
#
# Created on Sep 20, 2012
# @author: Dermot Buckley <dermot@buckley.ie>
# @copyright: Dermot Buckley
#
import time
import traceback
try :
import json
except ImportError :
from lib import simplejson as json
2014-03-20 05:33:34 +00:00
import sickbeard
2014-03-10 05:18:05 +00:00
from sickbeard import logger
from sickbeard import db
2014-03-20 05:33:34 +00:00
from sickbeard import helpers
2014-03-10 05:18:05 +00:00
from sickbeard . exceptions import ex
from lib import requests
MAX_XEM_AGE_SECS = 86400 # 1 day
def get_scene_numbering ( indexer_id , season , episode , fallback_to_xem = True ) :
"""
Returns a tuple , ( season , episode ) , with the scene numbering ( if there is one ) ,
otherwise returns the xem numbering ( if fallback_to_xem is set ) , otherwise
2014-03-11 02:32:02 +00:00
returns the TVDB and TVRAGE numbering .
2014-03-10 05:18:05 +00:00
( so the return values will always be set )
@param indexer_id : int
@param season : int
@param episode : int
@param fallback_to_xem : bool If set ( the default ) , check xem for matches if there is no local scene numbering
@return : ( int , int ) a tuple with ( season , episode )
"""
if indexer_id is None or season is None or episode is None :
return ( season , episode )
2014-03-20 05:33:34 +00:00
2014-03-10 05:18:05 +00:00
result = find_scene_numbering ( indexer_id , season , episode )
if result :
return result
else :
if fallback_to_xem :
xem_result = find_xem_numbering ( indexer_id , season , episode )
if xem_result :
return xem_result
return ( season , episode )
def find_scene_numbering ( indexer_id , season , episode ) :
"""
Same as get_scene_numbering ( ) , but returns None if scene numbering is not set
"""
if indexer_id is None or season is None or episode is None :
return ( season , episode )
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return ( season , episode )
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
myDB = db . DBConnection ( )
2014-03-20 05:33:34 +00:00
rows = myDB . select ( " SELECT scene_season, scene_episode FROM scene_numbering WHERE indexer = ? and indexer_id = ? and season = ? and episode = ? " , [ indexer , indexer_id , season , episode ] )
2014-03-10 05:18:05 +00:00
if rows :
return ( int ( rows [ 0 ] [ " scene_season " ] ) , int ( rows [ 0 ] [ " scene_episode " ] ) )
else :
return ( season , episode )
def get_indexer_numbering ( indexer_id , sceneSeason , sceneEpisode , fallback_to_xem = True ) :
"""
2014-03-11 02:32:02 +00:00
Returns a tuple , ( season , episode ) with the TVDB and TVRAGE numbering for ( sceneSeason , sceneEpisode )
2014-03-10 05:18:05 +00:00
( this works like the reverse of get_scene_numbering )
"""
if indexer_id is None or sceneSeason is None or sceneEpisode is None :
return ( sceneSeason , sceneEpisode )
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return ( sceneSeason , sceneEpisode )
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
myDB = db . DBConnection ( )
2014-03-20 05:33:34 +00:00
rows = myDB . select ( " SELECT season, episode FROM scene_numbering WHERE indexer = ? and indexer_id = ? and scene_season = ? and scene_episode = ? " , [ indexer , indexer_id , sceneSeason , sceneEpisode ] )
2014-03-10 05:18:05 +00:00
if rows :
return ( int ( rows [ 0 ] [ " season " ] ) , int ( rows [ 0 ] [ " episode " ] ) )
else :
if fallback_to_xem :
return get_indexer_numbering_for_xem ( indexer_id , sceneSeason , sceneEpisode )
return ( sceneSeason , sceneEpisode )
def get_scene_numbering_for_show ( indexer_id ) :
"""
Returns a dict of ( season , episode ) : ( sceneSeason , sceneEpisode ) mappings
for an entire show . Both the keys and values of the dict are tuples .
Will be empty if there are no scene numbers set
"""
if indexer_id is None :
return { }
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return { }
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
myDB = db . DBConnection ( )
rows = myDB . select ( ''' SELECT season, episode, scene_season, scene_episode
2014-03-20 05:33:34 +00:00
FROM scene_numbering WHERE indexer = ? and indexer_id = ?
ORDER BY season , episode ''' , [indexer, indexer_id])
2014-03-10 05:18:05 +00:00
result = { }
for row in rows :
result [ ( int ( row [ ' season ' ] ) , int ( row [ ' episode ' ] ) ) ] = ( int ( row [ ' scene_season ' ] ) , int ( row [ ' scene_episode ' ] ) )
return result
def set_scene_numbering ( indexer_id , season , episode , sceneSeason = None , sceneEpisode = None ) :
"""
Set scene numbering for a season / episode .
To clear the scene numbering , leave both sceneSeason and sceneEpisode as None .
"""
if indexer_id is None or season is None or episode is None :
return
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
myDB = db . DBConnection ( )
# sanity
#if sceneSeason == None: sceneSeason = season
#if sceneEpisode == None: sceneEpisode = episode
# delete any existing record first
2014-03-20 05:33:34 +00:00
myDB . action ( ' DELETE FROM scene_numbering where indexer = ? and indexer_id = ? and season = ? and episode = ? ' , [ indexer , indexer_id , season , episode ] )
2014-03-10 05:18:05 +00:00
# now, if the new numbering is not the default, we save a new record
if sceneSeason is not None and sceneEpisode is not None :
2014-03-20 05:33:34 +00:00
myDB . action ( " INSERT INTO scene_numbering (indexer, indexer_id, season, episode, scene_season, scene_episode) VALUES (?,?,?,?,?,?) " , [ indexer , indexer_id , season , episode , sceneSeason , sceneEpisode ] )
2014-03-10 05:18:05 +00:00
def find_xem_numbering ( indexer_id , season , episode ) :
"""
Returns the scene numbering , as retrieved from xem .
Refreshes / Loads as needed .
@param indexer_id : int
@param season : int
@param episode : int
@return : ( int , int ) a tuple of scene_season , scene_episode , or None if there is no special mapping .
"""
if indexer_id is None or season is None or episode is None :
return None
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return None
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
if _xem_refresh_needed ( indexer_id ) :
_xem_refresh ( indexer_id )
cacheDB = db . DBConnection ( ' cache.db ' )
2014-03-20 05:33:34 +00:00
rows = cacheDB . select ( " SELECT scene_season, scene_episode FROM xem_numbering WHERE indexer = ? and indexer_id = ? and season = ? and episode = ? " , [ indexer , indexer_id , season , episode ] )
2014-03-10 05:18:05 +00:00
if rows :
return ( int ( rows [ 0 ] [ " scene_season " ] ) , int ( rows [ 0 ] [ " scene_episode " ] ) )
else :
return None
def get_indexer_numbering_for_xem ( indexer_id , sceneSeason , sceneEpisode ) :
"""
Reverse of find_xem_numbering : lookup a tvdb season and episode using scene numbering
@param indexer_id : int
@param sceneSeason : int
@param sceneEpisode : int
@return : ( int , int ) a tuple of ( season , episode )
"""
if indexer_id is None or sceneSeason is None or sceneEpisode is None :
return None
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return None
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
if _xem_refresh_needed ( indexer_id ) :
_xem_refresh ( indexer_id )
cacheDB = db . DBConnection ( ' cache.db ' )
2014-03-20 05:33:34 +00:00
rows = cacheDB . select ( " SELECT season, episode FROM xem_numbering WHERE indexer = ? and indexer_id = ? and scene_season = ? and scene_episode = ? " , [ indexer , indexer_id , sceneSeason , sceneEpisode ] )
2014-03-10 05:18:05 +00:00
if rows :
return ( int ( rows [ 0 ] [ " season " ] ) , int ( rows [ 0 ] [ " episode " ] ) )
else :
return ( sceneSeason , sceneEpisode )
def _xem_refresh_needed ( indexer_id ) :
"""
Is a refresh needed on a show ?
@param indexer_id : int
@return : bool
"""
if indexer_id is None :
return False
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return False
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
cacheDB = db . DBConnection ( ' cache.db ' )
2014-03-20 05:33:34 +00:00
rows = cacheDB . select ( " SELECT last_refreshed FROM xem_refresh WHERE indexer = ? and indexer_id = ? " , [ indexer , indexer_id ] )
2014-03-10 05:18:05 +00:00
if rows :
return time . time ( ) > ( int ( rows [ 0 ] [ ' last_refreshed ' ] ) + MAX_XEM_AGE_SECS )
else :
return True
def _xem_refresh ( indexer_id ) :
"""
Refresh data from xem for a tv show
@param indexer_id : int
"""
if indexer_id is None :
return
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
try :
2014-03-20 05:33:34 +00:00
logger . log ( u ' Looking up XEM scene mapping for show %s on %s ' % ( indexer_id , indexer , ) , logger . DEBUG )
data = None
if ' Tvdb ' in indexer :
data = requests . get ( ' http://thexem.de/map/all?id= %s &origin=tvdb&destination=scene ' % ( indexer_id , ) ) . json ( )
elif ' TVRage ' in indexer :
data = requests . get ( ' http://thexem.de/map/all?id= %s &origin=rage&destination=scene ' % ( indexer_id , ) ) . json ( )
2014-03-10 05:18:05 +00:00
if data is None or data == ' ' :
2014-03-20 05:33:34 +00:00
logger . log ( u ' No XEN data for show " %s on %s " , trying TVTumbler ' % ( indexer_id , indexer , ) , logger . MESSAGE )
if ' Tvdb ' in indexer :
data = requests . get ( ' http://show-api.tvtumbler.com/api/thexem/all?id= %s &origin=tvdb&destination=scene ' % ( indexer_id , ) ) . json ( )
elif ' TVRage ' in indexer :
data = requests . get ( ' http://show-api.tvtumbler.com/api/thexem/all?id= %s &origin=rage&destination=scene ' % ( indexer_id , ) ) . json ( )
2014-03-10 05:18:05 +00:00
if data is None or data == ' ' :
2014-03-20 05:33:34 +00:00
logger . log ( u ' TVTumbler also failed for show " %s on %s " . giving up. ' % ( indexer_id , indexer , ) , logger . MESSAGE )
2014-03-10 05:18:05 +00:00
return None
2014-03-20 05:33:34 +00:00
2014-03-10 05:18:05 +00:00
result = data
if result :
cacheDB = db . DBConnection ( ' cache.db ' )
2014-03-20 05:33:34 +00:00
cacheDB . action ( " INSERT OR REPLACE INTO xem_refresh (indexer, indexer_id, last_refreshed) VALUES (?,?,?) " , [ indexer , indexer_id , time . time ( ) ] )
2014-03-10 05:18:05 +00:00
if ' success ' in result [ ' result ' ] :
2014-03-20 05:33:34 +00:00
cacheDB . action ( " DELETE FROM xem_numbering where indexer = ? and indexer_id = ? " , [ indexer , indexer_id ] )
2014-03-10 05:18:05 +00:00
for entry in result [ ' data ' ] :
if ' scene ' in entry :
2014-03-20 05:33:34 +00:00
if ' Tvdb ' in indexer :
cacheDB . action ( " INSERT INTO xem_numbering (indexer, indexer_id, season, episode, scene_season, scene_episode) VALUES (?,?,?,?,?,?) " ,
[ indexer , indexer_id , entry [ ' tvdb ' ] [ ' season ' ] , entry [ ' tvdb ' ] [ ' episode ' ] , entry [ ' scene ' ] [ ' season ' ] , entry [ ' scene ' ] [ ' episode ' ] ] )
elif ' TVRage ' in indexer :
cacheDB . action ( " INSERT INTO xem_numbering (indexer, indexer_id, season, episode, scene_season, scene_episode) VALUES (?,?,?,?,?,?) " ,
[ indexer , indexer_id , entry [ ' rage ' ] [ ' season ' ] , entry [ ' rage ' ] [ ' episode ' ] , entry [ ' scene ' ] [ ' season ' ] , entry [ ' scene ' ] [ ' episode ' ] ] )
2014-03-10 05:18:05 +00:00
if ' scene_2 ' in entry : # for doubles
2014-03-20 05:33:34 +00:00
if ' Tvdb ' in indexer :
cacheDB . action ( " INSERT INTO xem_numbering (indexer, indexer_id, season, episode, scene_season, scene_episode) VALUES (?,?,?,?,?,?) " ,
[ indexer , indexer_id , entry [ ' tvdb ' ] [ ' season ' ] , entry [ ' tvdb ' ] [ ' episode ' ] , entry [ ' scene_2 ' ] [ ' season ' ] , entry [ ' scene_2 ' ] [ ' episode ' ] ] )
elif ' TVRage ' in indexer :
cacheDB . action ( " INSERT INTO xem_numbering (indexer, indexer_id, season, episode, scene_season, scene_episode) VALUES (?,?,?,?,?,?) " ,
[ indexer , indexer_id , entry [ ' rage ' ] [ ' season ' ] , entry [ ' rage ' ] [ ' episode ' ] , entry [ ' scene_2 ' ] [ ' season ' ] , entry [ ' scene_2 ' ] [ ' episode ' ] ] )
2014-03-10 05:18:05 +00:00
else :
2014-03-20 05:33:34 +00:00
logger . log ( u ' Failed to get XEM scene data for show %s from %s because " %s " ' % ( indexer_id , indexer , result [ ' message ' ] ) , logger . MESSAGE )
2014-03-10 05:18:05 +00:00
else :
2014-03-20 05:33:34 +00:00
logger . log ( u " Empty lookup result - no XEM data for show %s on %s " % ( indexer_id , indexer , ) , logger . MESSAGE )
2014-03-10 05:18:05 +00:00
except Exception , e :
2014-03-20 05:33:34 +00:00
logger . log ( u " Exception while refreshing XEM data for show " + str ( indexer_id ) + " on " + indexer + " : " + ex ( e ) , logger . WARNING )
2014-03-10 05:18:05 +00:00
logger . log ( traceback . format_exc ( ) , logger . DEBUG )
return None
def get_xem_numbering_for_show ( indexer_id ) :
"""
Returns a dict of ( season , episode ) : ( sceneSeason , sceneEpisode ) mappings
for an entire show . Both the keys and values of the dict are tuples .
Will be empty if there are no scene numbers set in xem
"""
if indexer_id is None :
return { }
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return { }
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
if _xem_refresh_needed ( indexer_id ) :
_xem_refresh ( indexer_id )
cacheDB = db . DBConnection ( ' cache.db ' )
rows = cacheDB . select ( ''' SELECT season, episode, scene_season, scene_episode
2014-03-20 05:33:34 +00:00
FROM xem_numbering WHERE indexer = ? and indexer_id = ?
ORDER BY season , episode ''' , [indexer, indexer_id])
2014-03-10 05:18:05 +00:00
result = { }
for row in rows :
result [ ( int ( row [ ' season ' ] ) , int ( row [ ' episode ' ] ) ) ] = ( int ( row [ ' scene_season ' ] ) , int ( row [ ' scene_episode ' ] ) )
return result
def get_xem_numbering_for_season ( indexer_id , season ) :
"""
Returns a dict of ( season , episode ) : ( sceneSeason , sceneEpisode ) mappings
for an entire show . Both the keys and values of the dict are tuples .
Will be empty if there are no scene numbers set
"""
if indexer_id is None or season is None :
return { }
2014-03-20 05:33:34 +00:00
showObj = helpers . findCertainShow ( sickbeard . showList , indexer_id )
if showObj is None : return { }
indexer = showObj . indexer
2014-03-10 05:18:05 +00:00
if _xem_refresh_needed ( indexer_id ) :
_xem_refresh ( indexer_id )
cacheDB = db . DBConnection ( ' cache.db ' )
rows = cacheDB . select ( ''' SELECT season, scene_season
2014-03-20 05:33:34 +00:00
FROM xem_numbering WHERE indexer = ? and indexer_id = ? AND season = ?
ORDER BY season ''' , [indexer, indexer_id, season])
2014-03-10 05:18:05 +00:00
result = { }
if rows :
for row in rows :
result . setdefault ( int ( row [ ' season ' ] ) , [ ] ) . append ( int ( row [ ' scene_season ' ] ) )
else :
result . setdefault ( int ( season ) , [ ] ) . append ( int ( season ) )
return result