mirror of
https://github.com/SickGear/SickGear.git
synced 2024-12-11 21:53:37 +00:00
47f1c8eaaf
Add support for multiple server hosts to the updated Plex server notifier. Change Plex Media Server settings section for multi server(s) and improve the layout in the config/notifications page. Add logic to Plex notifier to update a single server where its TV section path matches the downloaded show. All server libraries are updated if no single server has a download path match. Change the ui notifications to show the Plex Media Server(s) actioned for library updating. Fix issue where PMS text wasn't initialised on the config/notifications page and added info about Plex clients. Add ability to test Plex Server(s) on config/notifications page.
269 lines
11 KiB
Python
269 lines
11 KiB
Python
# Author: Nic Wolfe <nic@wolfeden.ca>
|
|
# URL: http://code.google.com/p/sickbeard/
|
|
#
|
|
# 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 urllib
|
|
import urllib2
|
|
import base64
|
|
import re
|
|
|
|
import sickbeard
|
|
|
|
from sickbeard import logger
|
|
from sickbeard import common
|
|
from sickbeard.exceptions import ex
|
|
from sickbeard.encodingKludge import fixStupidEncodings
|
|
|
|
try:
|
|
import xml.etree.cElementTree as etree
|
|
except ImportError:
|
|
import elementtree.ElementTree as etree
|
|
|
|
|
|
class PLEXNotifier:
|
|
|
|
def _send_to_plex(self, command, host, username=None, password=None):
|
|
"""Handles communication to Plex hosts via HTTP API
|
|
|
|
Args:
|
|
command: Dictionary of field/data pairs, encoded via urllib and passed to the legacy xbmcCmds HTTP API
|
|
host: Plex host:port
|
|
username: Plex API username
|
|
password: Plex API password
|
|
|
|
Returns:
|
|
Returns 'OK' for successful commands or False if there was an error
|
|
|
|
"""
|
|
|
|
# fill in omitted parameters
|
|
if not username:
|
|
username = sickbeard.PLEX_USERNAME
|
|
if not password:
|
|
password = sickbeard.PLEX_PASSWORD
|
|
|
|
if not host:
|
|
logger.log(u'PLEX: No host specified, check your settings', logger.ERROR)
|
|
return False
|
|
|
|
for key in command:
|
|
if type(command[key]) == unicode:
|
|
command[key] = command[key].encode('utf-8')
|
|
|
|
enc_command = urllib.urlencode(command)
|
|
logger.log(u'PLEX: Encoded API command: ' + enc_command, logger.DEBUG)
|
|
|
|
url = 'http://%s/xbmcCmds/xbmcHttp/?%s' % (host, enc_command)
|
|
try:
|
|
req = urllib2.Request(url)
|
|
# if we have a password, use authentication
|
|
if password:
|
|
base64string = base64.encodestring('%s:%s' % (username, password))[:-1]
|
|
authheader = 'Basic %s' % base64string
|
|
req.add_header('Authorization', authheader)
|
|
logger.log(u'PLEX: Contacting (with auth header) via url: ' + url, logger.DEBUG)
|
|
else:
|
|
logger.log(u'PLEX: Contacting via url: ' + url, logger.DEBUG)
|
|
|
|
response = urllib2.urlopen(req)
|
|
|
|
result = response.read().decode(sickbeard.SYS_ENCODING)
|
|
response.close()
|
|
|
|
logger.log(u'PLEX: HTTP response: ' + result.replace('\n', ''), logger.DEBUG)
|
|
# could return result response = re.compile('<html><li>(.+\w)</html>').findall(result)
|
|
return 'OK'
|
|
|
|
except (urllib2.URLError, IOError), e:
|
|
logger.log(u'PLEX: Warning: Couldn\'t contact Plex at ' + fixStupidEncodings(url) + ' ' + ex(e), logger.WARNING)
|
|
return False
|
|
|
|
def _notify_pmc(self, message, title='SickGear', host=None, username=None, password=None, force=False):
|
|
"""Internal wrapper for the notify_snatch and notify_download functions
|
|
|
|
Args:
|
|
message: Message body of the notice to send
|
|
title: Title of the notice to send
|
|
host: Plex Media Client(s) host:port
|
|
username: Plex username
|
|
password: Plex password
|
|
force: Used for the Test method to override config safety checks
|
|
|
|
Returns:
|
|
Returns a list results in the format of host:ip:result
|
|
The result will either be 'OK' or False, this is used to be parsed by the calling function.
|
|
|
|
"""
|
|
|
|
# suppress notifications if the notifier is disabled but the notify options are checked
|
|
if not sickbeard.USE_PLEX and not force:
|
|
return False
|
|
|
|
# fill in omitted parameters
|
|
if not host:
|
|
host = sickbeard.PLEX_HOST
|
|
if not username:
|
|
username = sickbeard.PLEX_USERNAME
|
|
if not password:
|
|
password = sickbeard.PLEX_PASSWORD
|
|
|
|
result = ''
|
|
for curHost in [x.strip() for x in host.split(',')]:
|
|
logger.log(u'PLEX: Sending notification to \'%s\' - %s' % (curHost, message), logger.MESSAGE)
|
|
|
|
command = {'command': 'ExecBuiltIn', 'parameter': 'Notification(%s,%s)' % (title.encode('utf-8'), message.encode('utf-8'))}
|
|
notify_result = self._send_to_plex(command, curHost, username, password)
|
|
if notify_result:
|
|
result += '%s:%s' % (curHost, str(notify_result))
|
|
|
|
return result
|
|
|
|
##############################################################################
|
|
# Public functions
|
|
##############################################################################
|
|
|
|
def notify_snatch(self, ep_name):
|
|
if sickbeard.PLEX_NOTIFY_ONSNATCH:
|
|
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 notify_git_update(self, new_version='??'):
|
|
if sickbeard.USE_PLEX:
|
|
update_text = common.notifyStrings[common.NOTIFY_GIT_UPDATE_TEXT]
|
|
title = common.notifyStrings[common.NOTIFY_GIT_UPDATE]
|
|
self._notify_pmc(update_text + new_version, title)
|
|
|
|
def test_notify_pmc(self, host, username, password):
|
|
return self._notify_pmc('This is a test notification from SickGear', 'Test', host, username, password, force=True)
|
|
|
|
def test_notify_pms(self, host, username, password):
|
|
return self.update_library(host=host, username=username, password=password, force=False)
|
|
|
|
def update_library(self, ep_obj=None, host=None, username=None, password=None, force=True):
|
|
"""Handles updating the Plex Media Server host via HTTP API
|
|
|
|
Plex Media Server currently only supports updating the whole video library and not a specific path.
|
|
|
|
Returns:
|
|
Returns None for no issue, else a string of host with connection issues
|
|
|
|
"""
|
|
|
|
if sickbeard.USE_PLEX and sickbeard.PLEX_UPDATE_LIBRARY:
|
|
|
|
if not sickbeard.PLEX_SERVER_HOST:
|
|
logger.log(u'PLEX: No Plex Media Server host specified, check your settings', logger.DEBUG)
|
|
return False
|
|
|
|
if not host:
|
|
host = sickbeard.PLEX_SERVER_HOST
|
|
if not username:
|
|
username = sickbeard.PLEX_USERNAME
|
|
if not password:
|
|
password = sickbeard.PLEX_PASSWORD
|
|
|
|
# if username and password were provided, fetch the auth token from plex.tv
|
|
token_arg = ''
|
|
if username and password:
|
|
|
|
logger.log(u'PLEX: fetching plex.tv credentials for user: ' + username, logger.DEBUG)
|
|
req = urllib2.Request('https://plex.tv/users/sign_in.xml', data='')
|
|
authheader = 'Basic %s' % base64.encodestring('%s:%s' % (username, password))[:-1]
|
|
req.add_header('Authorization', authheader)
|
|
req.add_header('X-Plex-Device-Name', 'SickGear')
|
|
req.add_header('X-Plex-Product', 'SickGear Notifier')
|
|
req.add_header('X-Plex-Client-Identifier', '5f48c063eaf379a565ff56c9bb2b401e')
|
|
req.add_header('X-Plex-Version', '1.0')
|
|
|
|
try:
|
|
response = urllib2.urlopen(req)
|
|
auth_tree = etree.parse(response)
|
|
token = auth_tree.findall('.//authentication-token')[0].text
|
|
token_arg = '?X-Plex-Token=' + token
|
|
|
|
except urllib2.URLError as e:
|
|
logger.log(u'PLEX: Error fetching credentials from from plex.tv for user %s: %s' % (username, ex(e)), logger.MESSAGE)
|
|
|
|
except (ValueError, IndexError) as e:
|
|
logger.log(u'PLEX: Error parsing plex.tv response: ' + ex(e), logger.MESSAGE)
|
|
|
|
file_location = '' if None is ep_obj else ep_obj.location
|
|
host_list = [x.strip() for x in host.split(',')]
|
|
hosts_all = {}
|
|
hosts_match = {}
|
|
hosts_failed = []
|
|
for cur_host in host_list:
|
|
|
|
url = 'http://%s/library/sections%s' % (cur_host, token_arg)
|
|
try:
|
|
xml_tree = etree.parse(urllib.urlopen(url))
|
|
media_container = xml_tree.getroot()
|
|
except IOError, e:
|
|
logger.log(u'PLEX: Error while trying to contact Plex Media Server: ' + ex(e), logger.ERROR)
|
|
hosts_failed.append(cur_host)
|
|
continue
|
|
|
|
sections = media_container.findall('.//Directory')
|
|
if not sections:
|
|
logger.log(u'PLEX: Plex Media Server not running on: ' + cur_host, logger.MESSAGE)
|
|
hosts_failed.append(cur_host)
|
|
continue
|
|
|
|
for section in sections:
|
|
if 'show' == section.attrib['type']:
|
|
|
|
keyed_host = [(str(section.attrib['key']), cur_host)]
|
|
hosts_all.update(keyed_host)
|
|
if not file_location:
|
|
continue
|
|
|
|
for section_location in section.findall('.//Location'):
|
|
section_path = re.sub(r'[/\\]+', '/', section_location.attrib['path'].lower())
|
|
section_path = re.sub(r'^(.{,2})[/\\]', '', section_path)
|
|
location_path = re.sub(r'[/\\]+', '/', file_location.lower())
|
|
location_path = re.sub(r'^(.{,2})[/\\]', '', location_path)
|
|
|
|
if section_path in location_path:
|
|
hosts_match.update(keyed_host)
|
|
|
|
hosts_try = (hosts_all.copy(), hosts_match.copy())[len(hosts_match)]
|
|
host_list = []
|
|
for section_key, cur_host in hosts_try.items():
|
|
|
|
url = 'http://%s/library/sections/%s/refresh%s' % (cur_host, section_key, token_arg)
|
|
try:
|
|
force and urllib.urlopen(url)
|
|
host_list.append(cur_host)
|
|
except Exception, e:
|
|
logger.log(u'PLEX: Error updating library section for Plex Media Server: ' + ex(e), logger.ERROR)
|
|
hosts_failed.append(cur_host)
|
|
|
|
if len(hosts_match):
|
|
logger.log(u'PLEX: Updating hosts where TV section paths match the downloaded show: ' + ', '.join(set(host_list)), logger.MESSAGE)
|
|
else:
|
|
logger.log(u'PLEX: Updating all hosts with TV sections: ' + ', '.join(set(host_list)), logger.MESSAGE)
|
|
|
|
return (', '.join(set(hosts_failed)), None)[not len(hosts_failed)]
|
|
|
|
notifier = PLEXNotifier
|