PlexKodiConnect/resources/lib/downloadutils.py

396 lines
14 KiB
Python
Raw Normal View History

2015-12-24 14:07:00 -06:00
# -*- coding: utf-8 -*-
##################################################################################################
import json
import requests
import logging
import xbmc
import xbmcgui
import clientinfo
2016-06-17 22:03:28 -05:00
from utils import Logging, window, settings
2015-12-24 14:07:00 -06:00
##################################################################################################
# Disable requests logging
2016-06-17 22:03:28 -05:00
from requests.packages.urllib3.exceptions import InsecureRequestWarning, InsecurePlatformWarning
2015-12-24 14:07:00 -06:00
requests.packages.urllib3.disable_warnings(InsecureRequestWarning)
2016-06-17 22:03:28 -05:00
requests.packages.urllib3.disable_warnings(InsecurePlatformWarning)
2015-12-24 14:07:00 -06:00
#logging.getLogger('requests').setLevel(logging.WARNING)
##################################################################################################
class DownloadUtils():
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
# Borg - multiple instances, shared state
_shared_state = {}
clientInfo = clientinfo.ClientInfo()
addonName = clientInfo.getAddonName()
# Requests session
s = None
timeout = 30
def __init__(self):
2016-06-17 22:03:28 -05:00
global log
log = Logging(self.__class__.__name__).log
2015-12-24 14:07:00 -06:00
2016-06-17 22:03:28 -05:00
self.__dict__ = self._shared_state
2015-12-24 14:07:00 -06:00
def setUsername(self, username):
# Reserved for userclient only
self.username = username
2016-06-17 22:03:28 -05:00
log("Set username: %s" % username, 2)
2015-12-24 14:07:00 -06:00
def setUserId(self, userId):
# Reserved for userclient only
self.userId = userId
2016-06-17 22:03:28 -05:00
log("Set userId: %s" % userId, 2)
2015-12-24 14:07:00 -06:00
def setServer(self, server):
# Reserved for userclient only
self.server = server
2016-06-17 22:03:28 -05:00
log("Set server: %s" % server, 2)
2015-12-24 14:07:00 -06:00
def setToken(self, token):
# Reserved for userclient only
self.token = token
2016-06-17 22:03:28 -05:00
log("Set token: %s" % token, 2)
2015-12-24 14:07:00 -06:00
def setSSL(self, ssl, sslclient):
# Reserved for userclient only
self.sslverify = ssl
self.sslclient = sslclient
2016-06-17 22:03:28 -05:00
log("Verify SSL host certificate: %s" % ssl, 2)
log("SSL client side certificate: %s" % sslclient, 2)
2015-12-24 14:07:00 -06:00
def postCapabilities(self, deviceId):
# Post settings to session
url = "{server}/emby/Sessions/Capabilities/Full?format=json"
data = {
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
'PlayableMediaTypes': "Audio,Video",
'SupportsMediaControl': True,
'SupportedCommands': (
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
"MoveUp,MoveDown,MoveLeft,MoveRight,Select,"
"Back,ToggleContextMenu,ToggleFullscreen,ToggleOsdMenu,"
"GoHome,PageUp,NextLetter,GoToSearch,"
"GoToSettings,PageDown,PreviousLetter,TakeScreenshot,"
"VolumeUp,VolumeDown,ToggleMute,SendString,DisplayMessage,"
"SetAudioStreamIndex,SetSubtitleStreamIndex,"
"Mute,Unmute,SetVolume,"
"Play,Playstate,PlayNext"
)
}
2016-06-17 22:03:28 -05:00
log("Capabilities URL: %s" % url, 2)
log("Postdata: %s" % data, 2)
2015-12-24 14:07:00 -06:00
2016-04-04 16:21:05 -05:00
self.downloadUrl(url, postBody=data, action_type="POST")
2016-06-17 22:03:28 -05:00
log("Posted capabilities to %s" % self.server, 2)
2015-12-24 14:07:00 -06:00
# Attempt at getting sessionId
url = "{server}/emby/Sessions?DeviceId=%s&format=json" % deviceId
result = self.downloadUrl(url)
try:
sessionId = result[0]['Id']
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
except (KeyError, TypeError):
2016-06-17 22:03:28 -05:00
log("Failed to retrieve sessionId.", 1)
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
else:
2016-06-17 22:03:28 -05:00
log("Session: %s" % result, 2)
log("SessionId: %s" % sessionId, 1)
window('emby_sessionId', value=sessionId)
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
# Post any permanent additional users
2016-06-17 22:03:28 -05:00
additionalUsers = settings('additionalUsers')
2015-12-24 14:07:00 -06:00
if additionalUsers:
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
additionalUsers = additionalUsers.split(',')
2016-06-17 22:03:28 -05:00
log("List of permanent users added to the session: %s"
2015-12-24 14:07:00 -06:00
% additionalUsers, 1)
# Get the user list from server to get the userId
url = "{server}/emby/Users?format=json"
result = self.downloadUrl(url)
for additional in additionalUsers:
addUser = additional.decode('utf-8').lower()
# Compare to server users to list of permanent additional users
for user in result:
username = user['Name'].lower()
if username in addUser:
userId = user['Id']
url = (
"{server}/emby/Sessions/%s/Users/%s?format=json"
% (sessionId, userId)
)
2016-04-04 16:21:05 -05:00
self.downloadUrl(url, postBody={}, action_type="POST")
2015-12-24 14:07:00 -06:00
def startSession(self):
self.deviceId = self.clientInfo.getDeviceId()
# User is identified from this point
# Attach authenticated header to the session
2016-02-24 22:45:25 -06:00
verify = False
2015-12-24 14:07:00 -06:00
header = self.getHeader()
# If user enabled host certificate verification
try:
verify = self.sslverify
2016-02-24 22:45:25 -06:00
if self.sslclient is not None:
verify = self.sslclient
2015-12-24 14:07:00 -06:00
except:
2016-06-17 22:03:28 -05:00
log("Could not load SSL settings.", 1)
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
# Start session
self.s = requests.Session()
self.s.headers = header
self.s.verify = verify
# Retry connections to the server
self.s.mount("http://", requests.adapters.HTTPAdapter(max_retries=1))
self.s.mount("https://", requests.adapters.HTTPAdapter(max_retries=1))
2016-06-17 22:03:28 -05:00
log("Requests session started on: %s" % self.server, 1)
2015-12-24 14:07:00 -06:00
def stopSession(self):
try:
self.s.close()
except:
2016-06-17 22:03:28 -05:00
log("Requests session could not be terminated.", 1)
2015-12-24 14:07:00 -06:00
def getHeader(self, authenticate=True):
2016-03-31 17:42:49 -05:00
deviceName = self.clientInfo.getDeviceName()
2016-06-17 22:03:28 -05:00
deviceName = deviceName.encode('utf-8')
2016-03-31 17:42:49 -05:00
deviceId = self.clientInfo.getDeviceId()
version = self.clientInfo.getVersion()
2015-12-24 14:07:00 -06:00
if not authenticate:
# If user is not authenticated
auth = (
'MediaBrowser Client="Kodi", Device="%s", DeviceId="%s", Version="%s"'
% (deviceName, deviceId, version))
2015-12-24 14:07:00 -06:00
header = {
'Content-type': 'application/json',
'Accept-encoding': 'gzip',
'Accept-Charset': 'UTF-8,*',
'Authorization': auth
2016-03-30 22:37:37 -05:00
}
2016-06-17 22:03:28 -05:00
log("Header: %s" % header, 2)
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
else:
userId = self.userId
token = self.token
# Attached to the requests session
auth = (
'MediaBrowser UserId="%s", Client="Kodi", Device="%s", DeviceId="%s", Version="%s"'
% (userId, deviceName, deviceId, version))
2015-12-24 14:07:00 -06:00
header = {
'Content-type': 'application/json',
'Accept-encoding': 'gzip',
'Accept-Charset': 'UTF-8,*',
'Authorization': auth,
'X-MediaBrowser-Token': token
2016-03-30 22:37:37 -05:00
}
2016-06-17 22:03:28 -05:00
log("Header: %s" % header, 2)
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
return header
2016-06-17 22:03:28 -05:00
def downloadUrl(self, url, postBody=None, action_type="GET", parameters=None,
authenticate=True):
2016-03-30 22:37:37 -05:00
2016-06-17 22:03:28 -05:00
log("=== ENTER downloadUrl ===", 2)
2015-12-24 14:07:00 -06:00
default_link = ""
try:
# If user is authenticated
2016-06-17 22:03:28 -05:00
if authenticate:
2015-12-24 14:07:00 -06:00
# Get requests session
2016-03-30 22:37:37 -05:00
try:
2015-12-24 14:07:00 -06:00
s = self.s
# Replace for the real values
url = url.replace("{server}", self.server)
url = url.replace("{UserId}", self.userId)
# Prepare request
2016-03-31 17:59:29 -05:00
if action_type == "GET":
2016-03-31 17:42:49 -05:00
r = s.get(url, json=postBody, params=parameters, timeout=self.timeout)
2016-03-31 17:59:29 -05:00
elif action_type == "POST":
2016-03-31 17:42:49 -05:00
r = s.post(url, json=postBody, timeout=self.timeout)
2016-03-31 17:59:29 -05:00
elif action_type == "DELETE":
2016-03-31 17:42:49 -05:00
r = s.delete(url, json=postBody, timeout=self.timeout)
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
except AttributeError:
# request session does not exists
# Get user information
2016-06-17 22:03:28 -05:00
self.userId = window('emby_currUser')
self.server = window('emby_server%s' % self.userId)
self.token = window('emby_accessToken%s' % self.userId)
2015-12-24 14:07:00 -06:00
header = self.getHeader()
verifyssl = False
cert = None
# IF user enables ssl verification
2016-06-17 22:03:28 -05:00
if settings('sslverify') == "true":
2015-12-24 14:07:00 -06:00
verifyssl = True
2016-06-17 22:03:28 -05:00
if settings('sslcert') != "None":
verifyssl = settings('sslcert')
2015-12-24 14:07:00 -06:00
# Replace for the real values
url = url.replace("{server}", self.server)
url = url.replace("{UserId}", self.userId)
# Prepare request
2016-03-31 17:59:29 -05:00
if action_type == "GET":
2015-12-24 14:07:00 -06:00
r = requests.get(url,
json=postBody,
params=parameters,
headers=header,
2016-03-31 17:42:49 -05:00
timeout=self.timeout,
2015-12-24 14:07:00 -06:00
verify=verifyssl)
2016-03-31 17:59:29 -05:00
elif action_type == "POST":
2015-12-24 14:07:00 -06:00
r = requests.post(url,
json=postBody,
headers=header,
2016-03-31 17:42:49 -05:00
timeout=self.timeout,
2015-12-24 14:07:00 -06:00
verify=verifyssl)
2016-03-31 17:59:29 -05:00
elif action_type == "DELETE":
2015-12-24 14:07:00 -06:00
r = requests.delete(url,
json=postBody,
headers=header,
2016-03-31 17:42:49 -05:00
timeout=self.timeout,
2015-12-24 14:07:00 -06:00
verify=verifyssl)
# If user is not authenticated
elif not authenticate:
header = self.getHeader(authenticate=False)
verifyssl = False
# If user enables ssl verification
try:
verifyssl = self.sslverify
2016-02-24 22:45:25 -06:00
if self.sslclient is not None:
verifyssl = self.sslclient
2015-12-24 14:07:00 -06:00
except AttributeError:
pass
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
# Prepare request
2016-03-31 17:59:29 -05:00
if action_type == "GET":
2015-12-24 14:07:00 -06:00
r = requests.get(url,
json=postBody,
params=parameters,
headers=header,
2016-03-31 17:42:49 -05:00
timeout=self.timeout,
2015-12-24 14:07:00 -06:00
verify=verifyssl)
2016-03-31 17:59:29 -05:00
elif action_type == "POST":
2015-12-24 14:07:00 -06:00
r = requests.post(url,
json=postBody,
headers=header,
2016-03-31 17:42:49 -05:00
timeout=self.timeout,
2015-12-24 14:07:00 -06:00
verify=verifyssl)
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
##### THE RESPONSE #####
2016-06-17 22:03:28 -05:00
log(r.url, 2)
2015-12-24 14:07:00 -06:00
if r.status_code == 204:
# No body in the response
2016-06-17 22:03:28 -05:00
log("====== 204 Success ======", 2)
2015-12-24 14:07:00 -06:00
elif r.status_code == requests.codes.ok:
2016-03-30 22:37:37 -05:00
try:
# UNICODE - JSON object
2015-12-24 14:07:00 -06:00
r = r.json()
2016-06-17 22:03:28 -05:00
log("====== 200 Success ======", 2)
log("Response: %s" % r, 2)
2015-12-24 14:07:00 -06:00
return r
except:
if r.headers.get('content-type') != "text/html":
2016-06-17 22:03:28 -05:00
log("Unable to convert the response for: %s" % url, 1)
2015-12-24 14:07:00 -06:00
else:
r.raise_for_status()
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
##### EXCEPTIONS #####
except requests.exceptions.ConnectionError as e:
# Make the addon aware of status
2016-06-17 22:03:28 -05:00
if window('emby_online') != "false":
log("Server unreachable at: %s" % url, 0)
log(e, 2)
window('emby_online', value="false")
2015-12-24 14:07:00 -06:00
except requests.exceptions.ConnectTimeout as e:
2016-06-17 22:03:28 -05:00
log("Server timeout at: %s" % url, 0)
log(e, 1)
2015-12-24 14:07:00 -06:00
except requests.exceptions.HTTPError as e:
if r.status_code == 401:
# Unauthorized
2016-06-17 22:03:28 -05:00
status = window('emby_serverStatus')
2015-12-24 14:07:00 -06:00
if 'X-Application-Error-Code' in r.headers:
# Emby server errors
if r.headers['X-Application-Error-Code'] == "ParentalControl":
# Parental control - access restricted
2016-06-17 22:03:28 -05:00
window('emby_serverStatus', value="restricted")
2015-12-24 14:07:00 -06:00
xbmcgui.Dialog().notification(
heading="Emby server",
message="Access restricted.",
icon=xbmcgui.NOTIFICATION_ERROR,
time=5000)
return False
2016-03-30 22:37:37 -05:00
2015-12-24 14:07:00 -06:00
elif r.headers['X-Application-Error-Code'] == "UnauthorizedAccessException":
# User tried to do something his emby account doesn't allow
pass
elif status not in ("401", "Auth"):
# Tell userclient token has been revoked.
2016-06-17 22:03:28 -05:00
window('emby_serverStatus', value="401")
log("HTTP Error: %s" % e, 0)
2015-12-24 14:07:00 -06:00
xbmcgui.Dialog().notification(
heading="Error connecting",
message="Unauthorized.",
icon=xbmcgui.NOTIFICATION_ERROR)
return 401
elif r.status_code in (301, 302):
# Redirects
pass
elif r.status_code == 400:
# Bad requests
pass
except requests.exceptions.SSLError as e:
2016-06-17 22:03:28 -05:00
log("Invalid SSL certificate for: %s" % url, 0)
log(e, 1)
2015-12-24 14:07:00 -06:00
except requests.exceptions.RequestException as e:
2016-06-17 22:03:28 -05:00
log("Unknown error connecting to: %s" % url, 0)
log(e, 1)
2015-12-24 14:07:00 -06:00
return default_link