using System ;
using System.Collections.Generic ;
using System.IO ;
using System.Linq ;
using System.Text ;
using System.Xml.Linq ;
using NLog ;
using NzbDrone.Core.Helpers ;
using NzbDrone.Core.Providers.Core ;
namespace NzbDrone.Core.Providers
{
public class XbmcProvider
{
private readonly IConfigProvider _configProvider ;
private readonly HttpProvider _httpProvider ;
private static readonly Logger Logger = LogManager . GetCurrentClassLogger ( ) ;
public XbmcProvider ( IConfigProvider configProvider , HttpProvider httpProvider )
{
_configProvider = configProvider ;
_httpProvider = httpProvider ;
}
#region XbmcProvider Members
public virtual void Notify ( string header , string message )
{
//Get time in seconds and convert to ms
var time = Convert . ToInt32 ( _configProvider . GetValue ( "XbmcDisplayTime" , "3" , true ) ) * 1000 ;
var command = String . Format ( "ExecBuiltIn(Notification({0},{1},{2}))" , header , message , time ) ;
if ( Convert . ToBoolean ( _configProvider . GetValue ( "XbmcNotificationImage" , false , true ) ) )
{
//Todo: Get the actual port that NzbDrone is running on...
var serverInfo = String . Format ( "http://{0}:{1}" , ServerHelper . GetServerHostname ( ) , "8989" ) ;
var imageUrl = String . Format ( "{0}/Content/XbmcNotification.png" , serverInfo ) ;
command = String . Format ( "ExecBuiltIn(Notification({0},{1},{2}, {3}))" , header , message , time , imageUrl ) ;
}
foreach ( var host in _configProvider . GetValue ( "XbmcHosts" , "localhost:80" , true ) . Split ( ',' ) )
{
Logger . Trace ( "Sending Notifcation to XBMC Host: {0}" , host ) ;
SendCommand ( host , command ) ;
}
}
public virtual void Update ( int seriesId )
{
foreach ( var host in _configProvider . GetValue ( "XbmcHosts" , "localhost:80" , true ) . Split ( ',' ) )
{
Logger . Trace ( "Sending Update DB Request to XBMC Host: {0}" , host ) ;
var xbmcSeriesPath = GetXbmcSeriesPath ( host , seriesId ) ;
//If the path is not found & the user wants to update the entire library, do it now.
if ( String . IsNullOrEmpty ( xbmcSeriesPath ) & & Convert . ToBoolean ( _configProvider . GetValue ( "XbmcFullUpdate" , false , true ) ) )
{
//Update the entire library
Logger . Trace ( "Series [{0}] doesn't exist on XBMC host: {1}, Updating Entire Library" , seriesId , host ) ;
SendCommand ( host , "ExecBuiltIn(UpdateLibrary(video))" ) ;
return ;
}
var command = String . Format ( "ExecBuiltIn(UpdateLibrary(video,{0}))" , xbmcSeriesPath ) ;
SendCommand ( host , command ) ;
}
}
public virtual void Clean ( )
{
foreach ( var host in _configProvider . GetValue ( "XbmcHosts" , "localhost:80" , true ) . Split ( ',' ) )
{
Logger . Trace ( "Sending DB Clean Request to XBMC Host: {0}" , host ) ;
var command = String . Format ( "ExecBuiltIn(CleanLibrary(video))" ) ;
SendCommand ( host , command ) ;
}
}
# endregion
private string SendCommand ( string host , string command )
{
var username = _configProvider . GetValue ( "XbmcUsername" , String . Empty , true ) ;
var password = _configProvider . GetValue ( "XbmcPassword" , String . Empty , true ) ;
var url = String . Format ( "http://{0}/xbmcCmds/xbmcHttp?command={1}" , host , command ) ;
if ( ! String . IsNullOrEmpty ( username ) )
{
return _httpProvider . DownloadString ( url , username , password ) ;
}
return _httpProvider . DownloadString ( url ) ;
}
private string GetXbmcSeriesPath ( string host , int seriesId )
{
var query = String . Format ( "select path.strPath from path, tvshow, tvshowlinkpath where tvshow.c12 = {0} and tvshowlinkpath.idShow = tvshow.idShow and tvshowlinkpath.idPath = path.idPath" , seriesId ) ;
var command = String . Format ( "QueryVideoDatabase({0})" , query ) ;
var setResponseCommand = "SetResponseFormat(webheader;false;webfooter;false;header;<xml>;footer;</xml>;opentag;<tag>;closetag;</tag>;closefinaltag;false)" ;
var resetResponseCommand = "SetResponseFormat()" ;
SendCommand ( host , setResponseCommand ) ;
var response = SendCommand ( host , command ) ;
SendCommand ( host , resetResponseCommand ) ;
if ( String . IsNullOrEmpty ( response ) )
return String . Empty ;
var xDoc = XDocument . Load ( new StringReader ( response ) ) ;
var xml = ( from x in xDoc . Descendants ( "xml" ) select x ) . FirstOrDefault ( ) ;
if ( xml = = null )
return String . Empty ;
var field = xml . Descendants ( "field" ) . FirstOrDefault ( ) ;
if ( field = = null )
return String . Empty ;
return field . Value ;
}
}
}