mirror of
https://github.com/SickGear/SickGear.git
synced 2025-01-23 17:53:43 +00:00
197 lines
7.7 KiB
Python
197 lines
7.7 KiB
Python
|
# Author: Nic Wolfe <nic@wolfeden.ca>
|
||
|
# URL: http://code.google.com/p/sickbeard/
|
||
|
#
|
||
|
# 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/>.
|
||
|
|
||
|
import urllib
|
||
|
import urllib2
|
||
|
import base64
|
||
|
|
||
|
import sickbeard
|
||
|
|
||
|
from sickbeard import logger
|
||
|
from sickbeard import common
|
||
|
from sickbeard.exceptions import ex
|
||
|
from sickbeard.encodingKludge import fixStupidEncodings
|
||
|
|
||
|
# TODO: switch over to using ElementTree
|
||
|
from xml.dom import minidom
|
||
|
|
||
|
|
||
|
class PLEXNotifier:
|
||
|
|
||
|
def _send_to_plex(self, command, host, username=None, password=None):
|
||
|
"""Handles communication to Plex hosts via HTTP API
|
||
|
|
||
|
def notify_subtitle_download(self, ep_name, lang):
|
||
|
if sickbeard.PLEX_NOTIFY_ONSUBTITLEDOWNLOAD:
|
||
|
self._notifyXBMC(ep_name + ": " + lang, common.notifyStrings[common.NOTIFY_SUBTITLE_DOWNLOAD])
|
||
|
|
||
|
Args:
|
||
|
command: Dictionary of field/data pairs, encoded via urllib and passed to the legacy xbmcCmds HTTP API
|
||
|
host: Plex host:port
|
||
|
username: Plex API username
|
||
|
password: Plex API password
|
||
|
|
||
|
Returns:
|
||
|
Returns 'OK' for successful commands or False if there was an error
|
||
|
|
||
|
"""
|
||
|
|
||
|
# fill in omitted parameters
|
||
|
if not username:
|
||
|
username = sickbeard.PLEX_USERNAME
|
||
|
if not password:
|
||
|
password = sickbeard.PLEX_PASSWORD
|
||
|
|
||
|
if not host:
|
||
|
logger.log(u"No Plex host specified, check your settings", logger.DEBUG)
|
||
|
return False
|
||
|
|
||
|
for key in command:
|
||
|
if type(command[key]) == unicode:
|
||
|
command[key] = command[key].encode('utf-8')
|
||
|
|
||
|
enc_command = urllib.urlencode(command)
|
||
|
logger.log(u"Plex encoded API command: " + enc_command, logger.DEBUG)
|
||
|
|
||
|
url = 'http://%s/xbmcCmds/xbmcHttp/?%s' % (host, enc_command)
|
||
|
try:
|
||
|
req = urllib2.Request(url)
|
||
|
# if we have a password, use authentication
|
||
|
if password:
|
||
|
base64string = base64.encodestring('%s:%s' % (username, password))[:-1]
|
||
|
authheader = "Basic %s" % base64string
|
||
|
req.add_header("Authorization", authheader)
|
||
|
logger.log(u"Contacting Plex (with auth header) via url: " + url, logger.DEBUG)
|
||
|
else:
|
||
|
logger.log(u"Contacting Plex via url: " + url, logger.DEBUG)
|
||
|
|
||
|
response = urllib2.urlopen(req)
|
||
|
|
||
|
result = response.read().decode(sickbeard.SYS_ENCODING)
|
||
|
response.close()
|
||
|
|
||
|
logger.log(u"Plex HTTP response: " + result.replace('\n', ''), logger.DEBUG)
|
||
|
# could return result response = re.compile('<html><li>(.+\w)</html>').findall(result)
|
||
|
return 'OK'
|
||
|
|
||
|
except (urllib2.URLError, IOError), e:
|
||
|
logger.log(u"Warning: Couldn't contact Plex at " + fixStupidEncodings(url) + " " + ex(e), logger.WARNING)
|
||
|
return False
|
||
|
|
||
|
def _notify_pmc(self, message, title="Sick Beard", host=None, username=None, password=None, force=False):
|
||
|
"""Internal wrapper for the notify_snatch and notify_download functions
|
||
|
|
||
|
Args:
|
||
|
message: Message body of the notice to send
|
||
|
title: Title of the notice to send
|
||
|
host: Plex Media Client(s) host:port
|
||
|
username: Plex username
|
||
|
password: Plex password
|
||
|
force: Used for the Test method to override config saftey checks
|
||
|
|
||
|
Returns:
|
||
|
Returns a list results in the format of host:ip:result
|
||
|
The result will either be 'OK' or False, this is used to be parsed by the calling function.
|
||
|
|
||
|
"""
|
||
|
|
||
|
# fill in omitted parameters
|
||
|
if not host:
|
||
|
host = sickbeard.PLEX_HOST
|
||
|
if not username:
|
||
|
username = sickbeard.PLEX_USERNAME
|
||
|
if not password:
|
||
|
password = sickbeard.PLEX_PASSWORD
|
||
|
|
||
|
# suppress notifications if the notifier is disabled but the notify options are checked
|
||
|
if not sickbeard.USE_PLEX and not force:
|
||
|
logger.log("Notification for Plex not enabled, skipping this notification", logger.DEBUG)
|
||
|
return False
|
||
|
|
||
|
result = ''
|
||
|
for curHost in [x.strip() for x in host.split(",")]:
|
||
|
logger.log(u"Sending Plex notification to '" + curHost + "' - " + message, logger.MESSAGE)
|
||
|
|
||
|
command = {'command': 'ExecBuiltIn', 'parameter': 'Notification(' + title.encode("utf-8") + ',' + message.encode("utf-8") + ')'}
|
||
|
notifyResult = self._send_to_plex(command, curHost, username, password)
|
||
|
if notifyResult:
|
||
|
result += curHost + ':' + str(notifyResult)
|
||
|
|
||
|
return result
|
||
|
|
||
|
##############################################################################
|
||
|
# Public functions
|
||
|
##############################################################################
|
||
|
|
||
|
def notify_snatch(self, ep_name):
|
||
|
if sickbeard.PLEX_NOTIFY_ONSNATCH:
|
||
|
self._notify_pmc(ep_name, common.notifyStrings[common.NOTIFY_SNATCH])
|
||
|
|
||
|
def notify_download(self, ep_name):
|
||
|
if sickbeard.PLEX_NOTIFY_ONDOWNLOAD:
|
||
|
self._notify_pmc(ep_name, common.notifyStrings[common.NOTIFY_DOWNLOAD])
|
||
|
|
||
|
def notify_subtitle_download(self, ep_name, lang):
|
||
|
if sickbeard.PLEX_NOTIFY_ONSUBTITLEDOWNLOAD:
|
||
|
self._notify_pmc(ep_name + ": " + lang, common.notifyStrings[common.NOTIFY_SUBTITLE_DOWNLOAD])
|
||
|
|
||
|
def test_notify(self, host, username, password):
|
||
|
return self._notify_pmc("Testing Plex notifications from Sick Beard", "Test Notification", host, username, password, force=True)
|
||
|
|
||
|
def update_library(self):
|
||
|
"""Handles updating the Plex Media Server host via HTTP API
|
||
|
|
||
|
Plex Media Server currently only supports updating the whole video library and not a specific path.
|
||
|
|
||
|
Returns:
|
||
|
Returns True or False
|
||
|
|
||
|
"""
|
||
|
|
||
|
if sickbeard.USE_PLEX and sickbeard.PLEX_UPDATE_LIBRARY:
|
||
|
if not sickbeard.PLEX_SERVER_HOST:
|
||
|
logger.log(u"No Plex Server host specified, check your settings", logger.DEBUG)
|
||
|
return False
|
||
|
|
||
|
logger.log(u"Updating library for the Plex Media Server host: " + sickbeard.PLEX_SERVER_HOST, logger.MESSAGE)
|
||
|
|
||
|
url = "http://%s/library/sections" % sickbeard.PLEX_SERVER_HOST
|
||
|
try:
|
||
|
xml_sections = minidom.parse(urllib.urlopen(url))
|
||
|
except IOError, e:
|
||
|
logger.log(u"Error while trying to contact Plex Media Server: " + ex(e), logger.ERROR)
|
||
|
return False
|
||
|
|
||
|
sections = xml_sections.getElementsByTagName('Directory')
|
||
|
if not sections:
|
||
|
logger.log(u"Plex Media Server not running on: " + sickbeard.PLEX_SERVER_HOST, logger.MESSAGE)
|
||
|
return False
|
||
|
|
||
|
for s in sections:
|
||
|
if s.getAttribute('type') == "show":
|
||
|
url = "http://%s/library/sections/%s/refresh" % (sickbeard.PLEX_SERVER_HOST, s.getAttribute('key'))
|
||
|
try:
|
||
|
urllib.urlopen(url)
|
||
|
except Exception, e:
|
||
|
logger.log(u"Error updating library section for Plex Media Server: " + ex(e), logger.ERROR)
|
||
|
return False
|
||
|
|
||
|
return True
|
||
|
|
||
|
notifier = PLEXNotifier
|