import xbmc import xbmcgui import xbmcaddon import urllib import urllib2 import httplib import hashlib import StringIO import gzip import sys import inspect import json as json from random import randrange from uuid import uuid4 as uuid4 from ClientInformation import ClientInformation import encodings import time import traceback class DownloadUtils(): logLevel = 0 addonSettings = None getString = None LogCalls = False TrackLog = "" TotalUrlCalls = 0 def __init__(self, *args): addonId = ClientInformation().getAddonId() self.addonSettings = xbmcaddon.Addon(id=addonId) self.addon = xbmcaddon.Addon(id=addonId) self.getString = self.addonSettings.getLocalizedString level = self.addonSettings.getSetting('logLevel') self.logLevel = 0 if(level != None and level != ""): self.logLevel = int(level) if(self.logLevel == 2): self.LogCalls = True def logMsg(self, msg, level = 1): if(self.logLevel >= level): try: xbmc.log("mb3sync DownloadUtils -> " + str(msg)) except UnicodeEncodeError: try: xbmc.log("mb3sync DownloadUtils -> " + str(msg.encode('utf-8'))) except: pass def getServer(self, prefix=True): # For https support addon = self.addon HTTPS = addon.getSetting('https') host = addon.getSetting('ipaddress') port = addon.getSetting('port') server = host + ":" + port if len(server) < 2: self.logMsg("No server information saved.") return "" # If https is true if prefix and (HTTPS == "true"): server = "https://%s" % server return server # If https is false elif prefix and (HTTPS == "false"): server = "http://%s" % server return server # If only the host:port is required elif (prefix == False): return server def getUserId(self, suppress=True): WINDOW = xbmcgui.Window( 10000 ) self.addonSettings = xbmcaddon.Addon(id='plugin.video.mb3sync') port = self.addonSettings.getSetting('port') host = self.addonSettings.getSetting('ipaddress') userName = self.addonSettings.getSetting('username') userid = WINDOW.getProperty("userid" + userName) if(userid != None and userid != ""): self.logMsg("DownloadUtils -> Returning saved (WINDOW) UserID : " + userid + "UserName: " + userName,2) return userid userid = self.addonSettings.getSetting("userid" + userName) if(userid != None and userid != ""): WINDOW.setProperty("userid" + userName, userid) self.logMsg("DownloadUtils -> Returning saved (SETTING) UserID : " + userid + "UserName: " + userName,2) return userid self.logMsg("Looking for user name: " + userName) authOk = self.authenticate() if(authOk == ""): if(suppress == False): xbmcgui.Dialog().ok(self.getString(30044), self.getString(30044)) return "" userid = WINDOW.getProperty("userid" + userName) if(userid == "" and suppress == False): xbmcgui.Dialog().ok(self.getString(30045),self.getString(30045)) self.logMsg("userid : " + userid) self.postcapabilities() return userid def postcapabilities(self): self.logMsg("postcapabilities called") # Set Capabilities mb3Port = self.addonSettings.getSetting('port') mb3Host = self.addonSettings.getSetting('ipaddress') clientInfo = ClientInformation() machineId = clientInfo.getMachineId() # get session id url = "http://" + mb3Host + ":" + mb3Port + "/mediabrowser/Sessions?DeviceId=" + machineId + "&format=json" self.logMsg("Session URL : " + url); jsonData = self.downloadUrl(url) self.logMsg("Session JsonData : " + jsonData) result = json.loads(jsonData) self.logMsg("Session JsonData : " + str(result)) sessionId = result[0].get("Id") self.logMsg("Session Id : " + str(sessionId)) # post capability data playableMediaTypes = "Audio,Video,Photo" supportedCommands = "Play,Playstate,DisplayContent,GoHome,SendString,GoToSettings,DisplayMessage,PlayNext" url = "http://" + mb3Host + ":" + mb3Port + "/mediabrowser/Sessions/Capabilities?Id=" + sessionId + "&PlayableMediaTypes=" + playableMediaTypes + "&SupportedCommands=" + supportedCommands + "&SupportsMediaControl=True" postData = {} #postData["Id"] = sessionId; #postData["PlayableMediaTypes"] = "Video"; #postData["SupportedCommands"] = "MoveUp"; stringdata = json.dumps(postData) self.logMsg("Capabilities URL : " + url); self.logMsg("Capabilities Data : " + stringdata) self.downloadUrl(url, postBody=stringdata, type="POST") def authenticate(self, retreive=True): WINDOW = xbmcgui.Window(10000) self.addonSettings = xbmcaddon.Addon(id='plugin.video.mb3sync') username = self.addonSettings.getSetting('username') token = WINDOW.getProperty("AccessToken" + username) if(token != None and token != ""): self.logMsg("DownloadUtils -> Returning saved (WINDOW) AccessToken for user:" + username + " token:" + token,2) return token token = self.addonSettings.getSetting("AccessToken" + username) if(token != None and token != ""): WINDOW.setProperty("AccessToken" + username, token) self.logMsg("DownloadUtils -> Returning saved (SETTINGS) AccessToken for user:" + username + " token:" + token,2) return token port = self.addonSettings.getSetting("port") host = self.addonSettings.getSetting("ipaddress") if(host == None or host == "" or host == "" or port == None or port == ""): return "" if(retreive == False): return "" url = "http://" + host + ":" + port + "/mediabrowser/Users/AuthenticateByName?format=json" clientInfo = ClientInformation() txt_mac = clientInfo.getMachineId() version = clientInfo.getVersion() # get user info jsonData = self.downloadUrl("http://" + host + ":" + port + "/mediabrowser/Users/Public?format=json", authenticate=False) users = [] if(jsonData != ""): users = json.loads(jsonData) userHasPassword = False for user in users: name = user.get("Name") if(username == name): if(user.get("HasPassword") == True): userHasPassword = True break password = "" if(userHasPassword): password = xbmcgui.Dialog().input("Enter Password for user : " + username) if (password != ""): sha1 = hashlib.sha1(password) sha1 = sha1.hexdigest() else: sha1 = 'da39a3ee5e6b4b0d3255bfef95601890afd80709' messageData = "username=" + username + "&password=" + sha1 resp = self.downloadUrl(url, postBody=messageData, type="POST", authenticate=False) result = None accessToken = None try: xbmc.log("Auth_Reponce: " + str(resp)) result = json.loads(resp) accessToken = result.get("AccessToken") except: pass if(result != None and accessToken != None): userID = result.get("User").get("Id") self.logMsg("User Authenticated : " + accessToken) WINDOW.setProperty("AccessToken" + username, accessToken) WINDOW.setProperty("userid" + username, userID) self.addonSettings.setSetting("AccessToken" + username, accessToken) self.addonSettings.setSetting("userid" + username, userID) return accessToken else: self.logMsg("User NOT Authenticated") WINDOW.setProperty("AccessToken" + username, "") WINDOW.setProperty("userid" + username, "") self.addonSettings.setSetting("AccessToken" + username, "") self.addonSettings.setSetting("userid" + username, "") return "" def imageUrl(self, id, type, index, width, height): port = self.addonSettings.getSetting('port') host = self.addonSettings.getSetting('ipaddress') server = host + ":" + port return "http://" + server + "/mediabrowser/Items/" + str(id) + "/Images/" + type + "/" + str(index) + "/e3ab56fe27d389446754d0fb04910a34/original/" + str(width) + "/" + str(height) + "/0" def getAuthHeader(self, authenticate=True): clientInfo = ClientInformation() txt_mac = clientInfo.getMachineId() version = clientInfo.getVersion() deviceName = self.addonSettings.getSetting('deviceName') deviceName = deviceName.replace("\"", "_") if(authenticate == False): authString = "MediaBrowser Client=\"Kodi\",Device=\"" + deviceName + "\",DeviceId=\"" + txt_mac + "\",Version=\"" + version + "\"" headers = {"Accept-encoding": "gzip", "Accept-Charset" : "UTF-8,*", "Authorization" : authString} return headers else: userid = self.getUserId() authString = "MediaBrowser UserId=\"" + userid + "\",Client=\"Kodi\",Device=\"" + deviceName + "\",DeviceId=\"" + txt_mac + "\",Version=\"" + version + "\"" headers = {"Accept-encoding": "gzip", "Accept-Charset" : "UTF-8,*", "Authorization" : authString} authToken = self.authenticate() if(authToken != ""): headers["X-MediaBrowser-Token"] = authToken self.logMsg("Authentication Header : " + str(headers),2) return headers def downloadUrl(self, url, suppress=False, postBody=None, type="GET", popup=0, authenticate=True ): self.logMsg("== ENTER: getURL ==",2) if(authenticate == True and suppress == True): token = self.authenticate(retreive=False) if(token == ""): self.logMsg("No auth info set and suppress is true so returning no data!") return "" suppress = False self.TotalUrlCalls = self.TotalUrlCalls + 1 if(self.LogCalls): stackString = "" for f in inspect.stack(): stackString = stackString + "\r - " + str(f) self.TrackLog = self.TrackLog + "HTTP_API_CALL : " + url + stackString + "\r" link = "" try: if url[0:4] == "http": serversplit = 2 urlsplit = 3 else: serversplit = 0 urlsplit = 1 server = url.split('/')[serversplit] urlPath = "/"+"/".join(url.split('/')[urlsplit:]) self.logMsg("DOWNLOAD_URL = " + url,2) self.logMsg("server = " + str(server),2) self.logMsg("urlPath = " + str(urlPath),2) if(server[0:1] == ":" or server[-1:] == ":"): self.logMsg("No server host or port set in url") return "" head = self.getAuthHeader(authenticate) self.logMsg("HEADERS : " + str(head), level=2) conn = httplib.HTTPConnection(server, timeout=5) # make the connection and send the request if(postBody != None): head["Content-Type"] = "application/x-www-form-urlencoded" head["Content-Length"] = str(len(postBody)) self.logMsg("POST DATA : " + postBody,2) conn.request(method=type, url=urlPath, body=postBody, headers=head) else: conn.request(method=type, url=urlPath, headers=head) # get the response tries = 0 while tries <= 4: try: data = conn.getresponse() break except: # TODO: we need to work out which errors we can just quit trying immediately if(xbmc.abortRequested == True): return "" xbmc.sleep(100) if(xbmc.abortRequested == True): return "" tries += 1 if tries == 5: data = conn.getresponse() self.logMsg("GET URL HEADERS : " + str(data.getheaders()), level=2) # process the response contentType = "none" if int(data.status) == 200: retData = data.read() contentType = data.getheader('content-encoding') self.logMsg("Data Len Before : " + str(len(retData)), level=2) if(contentType == "gzip"): retData = StringIO.StringIO(retData) gzipper = gzip.GzipFile(fileobj=retData) link = gzipper.read() else: link = retData self.logMsg("Data Len After : " + str(len(link)), level=2) self.logMsg("====== 200 returned =======", level=2) self.logMsg("Content-Type : " + str(contentType), level=2) self.logMsg(link, level=2) self.logMsg("====== 200 finished ======", level=2) elif ( int(data.status) == 301 ) or ( int(data.status) == 302 ): try: conn.close() except: pass return data.getheader('Location') elif int(data.status) == 401: error = "HTTP response error: " + str(data.status) + " " + str(data.reason) xbmc.log(error) username = self.addonSettings.getSetting("username") WINDOW = xbmcgui.Window(10000) WINDOW.setProperty("AccessToken" + username, "") WINDOW.setProperty("userid" + username, "") self.addonSettings.setSetting("AccessToken" + username, "") self.addonSettings.setSetting("userid" + username, "") xbmcgui.Dialog().ok(self.getString(30135), self.getString(30044), "Reason : " + str(data.reason)) try: conn.close() except: pass return "" elif int(data.status) >= 400: error = "HTTP response error: " + str(data.status) + " " + str(data.reason) xbmc.log(error) stack = self.FormatException() self.logMsg(stack) if suppress is False: if popup == 0: xbmc.executebuiltin("XBMC.Notification(URL error: "+ str(data.reason) +",)") else: xbmcgui.Dialog().ok(self.getString(30135),server) try: conn.close() except: pass return "" else: link = "" except Exception, msg: error = "Unable to connect to " + str(server) + " : " + str(msg) xbmc.log(error) stack = self.FormatException() self.logMsg(stack) if suppress is False: if popup == 0: xbmc.executebuiltin("XBMC.Notification(: Connection Error: Error connecting to server,)") else: xbmcgui.Dialog().ok(self.getString(30204), str(msg)) pass else: try: conn.close() except: pass return link def FormatException(self): exception_list = traceback.format_stack() exception_list = exception_list[:-2] exception_list.extend(traceback.format_tb(sys.exc_info()[2])) exception_list.extend(traceback.format_exception_only(sys.exc_info()[0], sys.exc_info()[1])) exception_str = "Traceback (most recent call last):\n" exception_str += "".join(exception_list) # Removing the last \n exception_str = exception_str[:-1] return exception_str def __del__(self): return # xbmc.log("\rURL_REQUEST_REPORT : Total Calls : " + str(self.TotalUrlCalls) + "\r" + self.TrackLog)